/* * Copyright (C) 2011 Network RADIUS SARL <info@networkradius.com> * * This software may not be redistributed in any form without the prior * written consent of Network RADIUS. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include <networkradius-devel/client.h> #include <ctype.h> #include <assert.h> static int packet_code = PW_ACCESS_REQUEST; static int packet_id = 1; static uint8_t packet_vector[16] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; static char secret[256] = "testing123"; static int encode_tlv(char *buffer, uint8_t *output, size_t outlen); static const char *hextab = "0123456789abcdef"; static int encode_data_string(char *buffer, uint8_t *output, size_t outlen) { int length = 0; char *p; p = buffer + 1; while (*p && (outlen > 0)) { if (*p == '"') { return length; } if (*p != '\\') { *(output++) = *(p++); outlen--; length++; continue; } switch (p[1]) { default: *(output++) = p[1]; break; case 'n': *(output++) = '\n'; break; case 'r': *(output++) = '\r'; break; case 't': *(output++) = '\t'; break; } outlen--; length++; } fprintf(stderr, "String is not terminated\n"); return 0; } static int encode_data_tlv(char *buffer, char **endptr, uint8_t *output, size_t outlen) { int depth = 0; int length; char *p; for (p = buffer; *p != '\0'; p++) { if (*p == '{') depth++; if (*p == '}') { depth--; if (depth == 0) break; } } if (*p != '}') { fprintf(stderr, "No trailing '}' in string starting " "with \"%s\"\n", buffer); return 0; } *endptr = p + 1; *p = '\0'; p = buffer + 1; while (isspace((int) *p)) p++; length = encode_tlv(p, output, outlen); if (length == 0) return 0; return length; } static int encode_hex(char *p, uint8_t *output, size_t outlen) { int length = 0; while (*p) { char *c1, *c2; while (isspace((int) *p)) p++; if (!*p) break; if(!(c1 = memchr(hextab, tolower((int) p[0]), 16)) || !(c2 = memchr(hextab, tolower((int) p[1]), 16))) { fprintf(stderr, "Invalid data starting at " "\"%s\"\n", p); return 0; } *output = ((c1 - hextab) << 4) + (c2 - hextab); output++; length++; p += 2; outlen--; if (outlen == 0) { fprintf(stderr, "Too much data\n"); return 0; } } return length; } static int encode_data(char *p, uint8_t *output, size_t outlen) { int length; if (!isspace((int) *p)) { fprintf(stderr, "Invalid character following attribute " "definition\n"); return 0; } while (isspace((int) *p)) p++; if (*p == '{') { int sublen; char *q; length = 0; do { while (isspace((int) *p)) p++; if (!*p) { if (length == 0) { fprintf(stderr, "No data\n"); return 0; } break; } sublen = encode_data_tlv(p, &q, output, outlen); if (sublen == 0) return 0; length += sublen; output += sublen; outlen -= sublen; p = q; } while (*q); return length; } if (*p == '"') { length = encode_data_string(p, output, outlen); return length; } length = encode_hex(p, output, outlen); if (length == 0) { fprintf(stderr, "Empty string\n"); return 0; } return length; } static int decode_attr(char *buffer, char **endptr) { long attr; attr = strtol(buffer, endptr, 10); if (*endptr == buffer) { fprintf(stderr, "No valid number found in string " "starting with \"%s\"\n", buffer); return 0; } if (!**endptr) { fprintf(stderr, "Nothing follows attribute number\n"); return 0; } if ((attr <= 0) || (attr > 256)) { fprintf(stderr, "Attribute number is out of valid " "range\n"); return 0; } return (int) attr; } static int decode_vendor(char *buffer, char **endptr) { long vendor; if (*buffer != '.') { fprintf(stderr, "Invalid separator before vendor id\n"); return 0; } vendor = strtol(buffer + 1, endptr, 10); if (*endptr == (buffer + 1)) { fprintf(stderr, "No valid vendor number found\n"); return 0; } if (!**endptr) { fprintf(stderr, "Nothing follows vendor number\n"); return 0; } if ((vendor <= 0) || (vendor > (1 << 24))) { fprintf(stderr, "Vendor number is out of valid range\n"); return 0; } if (**endptr != '.') { fprintf(stderr, "Invalid data following vendor number\n"); return 0; } (*endptr)++; return (int) vendor; } static int encode_tlv(char *buffer, uint8_t *output, size_t outlen) { int attr; int length; char *p; attr = decode_attr(buffer, &p); if (attr == 0) return 0; output[0] = attr; output[1] = 2; if (*p == '.') { p++; length = encode_tlv(p, output + 2, outlen - 2); } else { length = encode_data(p, output + 2, outlen - 2); } if (length == 0) return 0; if (length > (255 - 2)) { fprintf(stderr, "TLV data is too long\n"); return 0; } output[1] += length; return length + 2; } static int encode_vsa(char *buffer, uint8_t *output, size_t outlen) { int vendor; int length; char *p; vendor = decode_vendor(buffer, &p); if (vendor == 0) return 0; output[0] = 0; output[1] = (vendor >> 16) & 0xff; output[2] = (vendor >> 8) & 0xff; output[3] = vendor & 0xff; length = encode_tlv(p, output + 4, outlen - 4); if (length == 0) return 0; if (length > (255 - 6)) { fprintf(stderr, "VSA data is too long\n"); return 0; } return length + 4; } static int encode_evs(char *buffer, uint8_t *output, size_t outlen) { int vendor; int attr; int length; char *p; vendor = decode_vendor(buffer, &p); if (vendor == 0) return 0; attr = decode_attr(p, &p); if (attr == 0) return 0; output[0] = 0; output[1] = (vendor >> 16) & 0xff; output[2] = (vendor >> 8) & 0xff; output[3] = vendor & 0xff; output[4] = attr; length = encode_data(p, output + 5, outlen - 5); if (length == 0) return 0; return length + 5; } static int encode_extended(char *buffer, uint8_t *output, size_t outlen) { int attr; int length; char *p; attr = decode_attr(buffer, &p); if (attr == 0) return 0; output[0] = attr; if (attr == 26) { length = encode_evs(p, output + 1, outlen - 1); } else { length = encode_data(p, output + 1, outlen - 1); } if (length == 0) return 0; if (length > (255 - 3)) { fprintf(stderr, "Extended Attr data is too long\n"); return 0; } return length + 1; } static int encode_extended_flags(char *buffer, uint8_t *output, size_t outlen) { int attr; int length, total; char *p; attr = decode_attr(buffer, &p); if (attr == 0) return 0; /* output[0] is the extended attribute */ output[1] = 4; output[2] = attr; output[3] = 0; if (attr == 26) { length = encode_evs(p, output + 4, outlen - 4); if (length == 0) return 0; output[1] += 5; length -= 5; } else { length = encode_data(p, output + 4, outlen - 4); } if (length == 0) return 0; total = 0; while (1) { int sublen = 255 - output[1]; if (length <= sublen) { output[1] += length; total += output[1]; break; } length -= sublen; memmove(output + 255 + 4, output + 255, length); memcpy(output + 255, output, 4); output[1] = 255; output[3] |= 0x80; output += 255; output[1] = 4; total += 255; } return total; } static int encode_rfc(char *buffer, uint8_t *output, size_t outlen) { int attr; int length, sublen; char *p; attr = decode_attr(buffer, &p); if (attr == 0) return 0; length = 2; output[0] = attr; output[1] = 2; if (attr == 26) { sublen = encode_vsa(p, output + 2, outlen - 2); } else if ((attr < 241) || (attr > 246)) { sublen = encode_data(p, output + 2, outlen - 2); } else { if (*p != '.') { fprintf(stderr, "Invalid data following " "attribute number\n"); return 0; } if (attr < 245) { sublen = encode_extended(p + 1, output + 2, outlen - 2); } else { /* * Not like the others! */ return encode_extended_flags(p + 1, output, outlen); } } if (sublen == 0) return 0; if (sublen > (255 -2)) { fprintf(stderr, "RFC Data is too long\n"); return 0; } output[1] += sublen; return length + sublen; } static int walk_callback(void *ctx, const DICT_ATTR *da, const uint8_t *data, size_t sizeof_data) { char **p = ctx; sprintf(*p, "v%u a%u l%ld,", da->vendor, da->attr, sizeof_data); *p += strlen(*p); } static void process_file(const char *filename) { int lineno, rcode; size_t i, outlen; ssize_t len, data_len; FILE *fp; RADIUS_PACKET packet; char input[8192], buffer[8192]; char output[8192]; uint8_t *attr, data[2048]; if (strcmp(filename, "-") == 0) { fp = stdin; filename = "<stdin>"; } else { fp = fopen(filename, "r"); if (!fp) { fprintf(stderr, "Error opening %s: %s\n", filename, strerror(errno)); exit(1); } } lineno = 0; *output = '\0'; data_len = 0; while (fgets(buffer, sizeof(buffer), fp) != NULL) { char *p = strchr(buffer, '\n'); VALUE_PAIR *vp, *head = NULL; VALUE_PAIR **tail = &head; lineno++; if (!p) { if (!feof(fp)) { fprintf(stderr, "Line %d too long in %s\n", lineno, filename); exit(1); } } else { *p = '\0'; } p = strchr(buffer, '#'); if (p) *p = '\0'; p = buffer; while (isspace((int) *p)) p++; if (!*p) continue; strcpy(input, p); if (strncmp(p, "raw ", 4) == 0) { outlen = encode_rfc(p + 4, data, sizeof(data)); if (outlen == 0) { fprintf(stderr, "Parse error in line %d of %s\n", lineno, filename); exit(1); } print_hex: if (outlen == 0) { output[0] = 0; continue; } data_len = outlen; for (i = 0; i < outlen; i++) { snprintf(output + 3*i, sizeof(output), "%02x ", data[i]); } outlen = strlen(output); output[outlen - 1] = '\0'; continue; } if (strncmp(p, "data ", 5) == 0) { if (strcmp(p + 5, output) != 0) { fprintf(stderr, "Mismatch in line %d of %s, expected: %s\n", lineno, filename, output); exit(1); } continue; } head = NULL; if (strncmp(p, "encode ", 7) == 0) { if (strcmp(p + 7, "-") == 0) { p = output; } else { p += 7; } rcode = nr_vp_sscanf(p, &head); if (rcode < 0) { strcpy(output, nr_strerror(rcode)); continue; } attr = data; vp = head; while (vp != NULL) { len = nr_vp2attr(NULL, NULL, &vp, attr, sizeof(data) - (attr - data)); if (len < 0) { fprintf(stderr, "Failed encoding %s: %s\n", vp->da->name, nr_strerror(len)); exit(1); } attr += len; if (len == 0) break; } nr_vp_free(&head); outlen = len; goto print_hex; } if (strncmp(p, "decode ", 7) == 0) { ssize_t my_len; if (strcmp(p + 7, "-") == 0) { attr = data; len = data_len; } else { attr = data; len = encode_hex(p + 7, data, sizeof(data)); if (len == 0) { fprintf(stderr, "Failed decoding hex string at line %d of %s\n", lineno, filename); exit(1); } } while (len > 0) { vp = NULL; my_len = nr_attr2vp(NULL, NULL, attr, len, &vp); if (my_len < 0) { nr_vp_free(&head); break; } if (my_len > len) { fprintf(stderr, "Internal sanity check failed at %d\n", __LINE__); exit(1); } *tail = vp; while (vp) { tail = &(vp->next); vp = vp->next; } attr += my_len; len -= my_len; } /* * Output may be an error, and we ignore * it if so. */ if (head) { p = output; for (vp = head; vp != NULL; vp = vp->next) { nr_vp_snprintf(p, sizeof(output) - (p - output), vp); p += strlen(p); if (vp->next) {strcpy(p, ", "); p += 2; } } nr_vp_free(&head); } else if (my_len < 0) { strcpy(output, nr_strerror(my_len)); } else { /* zero-length attribute */ *output = '\0'; } continue; } if (strncmp(p, "walk ", 5) == 0) { len = encode_hex(p + 5, data + 20, sizeof(data) - 20); if (len == 0) { fprintf(stderr, "Failed decoding hex string at line %d of %s\n", lineno, filename); exit(1); } memset(data, 0, 20); packet.data = data; packet.length = len + 20; packet.data[2] = ((len + 20) >> 8) & 0xff; packet.data[3] = (len + 20) & 0xff; *output = '\0'; p = output; rcode = nr_packet_walk(&packet, &p, walk_callback); if (rcode < 0) { snprintf(output, sizeof(output), "%d", rcode); continue; } if (*output) output[strlen(output) - 1] = '\0'; continue; } if (strncmp(p, "$INCLUDE ", 9) == 0) { p += 9; while (isspace((int) *p)) p++; process_file(p); continue; } if (strncmp(p, "secret ", 7) == 0) { strlcpy(secret, p + 7, sizeof(secret)); strlcpy(output, secret, sizeof(output)); continue; } if (strncmp(p, "code ", 5) == 0) { packet_code = atoi(p + 5); snprintf(output, sizeof(output), "%u", packet_code); continue; } if (strncmp(p, "sign ", 5) == 0) { len = encode_hex(p + 5, data + 20, sizeof(data) - 20); if (len == 0) { fprintf(stderr, "Failed decoding hex string at line %d of %s\n", lineno, filename); exit(1); } memset(&packet, 0, sizeof(packet)); packet.secret = secret; packet.sizeof_secret = strlen(secret); packet.code = packet_code; packet.id = packet_id; memcpy(packet.vector, packet_vector, 16); packet.data = data; packet.length = len + 20; /* * Hack encode the packet. */ packet.data[0] = packet_code; packet.data[1] = packet_id; packet.data[2] = ((len + 20) >> 8) & 0xff; packet.data[3] = (len + 20) & 0xff; memcpy(packet.data + 4, packet_vector, 16); rcode = nr_packet_sign(&packet, NULL); if (rcode < 0) { snprintf(output, sizeof(output), "%d", rcode); continue; } memcpy(data, packet.vector, sizeof(packet.vector)); outlen = sizeof(packet.vector); goto print_hex; } fprintf(stderr, "Unknown input at line %d of %s\n", lineno, filename); exit(1); } if (fp != stdin) fclose(fp); } int main(int argc, char *argv[]) { int c; if (argc < 2) { process_file("-"); } else { process_file(argv[1]); } return 0; }