| line | stmt | bran | cond | sub | pod | time | code | 
| 1 |  |  |  |  |  |  | /* | 
| 2 |  |  |  |  |  |  | * | 
| 3 |  |  |  |  |  |  | * Copyright (c) 2018, cPanel, LLC. | 
| 4 |  |  |  |  |  |  | * All rights reserved. | 
| 5 |  |  |  |  |  |  | * http://cpanel.net | 
| 6 |  |  |  |  |  |  | * | 
| 7 |  |  |  |  |  |  | * This is free software; you can redistribute it and/or modify it under the | 
| 8 |  |  |  |  |  |  | * same terms as Perl itself. | 
| 9 |  |  |  |  |  |  | * | 
| 10 |  |  |  |  |  |  | */ | 
| 11 |  |  |  |  |  |  |  | 
| 12 |  |  |  |  |  |  | #include | 
| 13 |  |  |  |  |  |  | #include | 
| 14 |  |  |  |  |  |  | #include | 
| 15 |  |  |  |  |  |  | #include | 
| 16 |  |  |  |  |  |  |  | 
| 17 |  |  |  |  |  |  | /* prototypes */ | 
| 18 |  |  |  |  |  |  | SV* _parse_string_field(SV *sv, int need_field); | 
| 19 |  |  |  |  |  |  |  | 
| 20 | 49 |  |  |  |  |  | SV* _parse_string_field(SV *sv, int need_field) { | 
| 21 | 49 |  |  |  |  |  | int len = SvCUR(sv); | 
| 22 | 49 |  |  |  |  |  | char *ptr = (char *) SvPVX_const(sv); /* todo: preserve the const state of the pointer */ | 
| 23 |  |  |  |  |  |  | AV   *av; | 
| 24 |  |  |  |  |  |  | char *start_key, *end_key; | 
| 25 |  |  |  |  |  |  | char *start_val, *end_val; | 
| 26 |  |  |  |  |  |  | char *max; | 
| 27 |  |  |  |  |  |  |  | 
| 28 | 49 |  |  |  |  |  | int is_utf8 = SvUTF8(sv); | 
| 29 |  |  |  |  |  |  |  | 
| 30 | 49 |  |  |  |  |  | av = newAV(); | 
| 31 |  |  |  |  |  |  |  | 
| 32 | 49 |  |  |  |  |  | const char eol      = '\n'; | 
| 33 | 49 |  |  |  |  |  | const char sep      = ':'; /* customize it later */ | 
| 34 | 49 |  |  |  |  |  | const char comment  = '#'; | 
| 35 | 49 |  |  |  |  |  | const char line_feed = '\r'; | 
| 36 |  |  |  |  |  |  |  | 
| 37 | 49 |  |  |  |  |  | start_key = ptr; | 
| 38 | 49 |  |  |  |  |  | end_key   = 0; | 
| 39 | 49 |  |  |  |  |  | start_val = 0; | 
| 40 | 49 |  |  |  |  |  | end_val   = 0; | 
| 41 |  |  |  |  |  |  |  | 
| 42 | 49 |  |  |  |  |  | int found_eol = 1; | 
| 43 | 49 |  |  |  |  |  | int found_comment = 0; | 
| 44 | 49 |  |  |  |  |  | int found_sep  = 0; | 
| 45 | 49 |  |  |  |  |  | int found_field = 0; | 
| 46 |  |  |  |  |  |  |  | 
| 47 | 6828 | 100 |  |  |  |  | for ( max = ptr + len ; ptr < max; ++ptr ) { | 
| 48 | 6779 | 100 |  |  |  |  | if ( ! *ptr ) continue; /* skip \0 so we can parse binaries strings */ | 
| 49 | 6778 | 100 |  |  |  |  | if ( *ptr == line_feed ) continue; /* ignore \r */ | 
| 50 |  |  |  |  |  |  |  | 
| 51 |  |  |  |  |  |  | //printf( "# %c\n", *ptr ); | 
| 52 |  |  |  |  |  |  |  | 
| 53 |  |  |  |  |  |  | /* skip all characters in a comment block */ | 
| 54 | 6742 | 100 |  |  |  |  | if ( found_comment ) { | 
| 55 | 522 | 100 |  |  |  |  | if ( *ptr == eol ) found_comment = 0; | 
| 56 | 522 |  |  |  |  |  | continue; | 
| 57 |  |  |  |  |  |  | } | 
| 58 |  |  |  |  |  |  |  | 
| 59 | 6220 | 100 |  |  |  |  | if ( (need_field == 0 && found_sep) || (need_field && found_sep == need_field) ) { | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
| 60 | 215 | 100 |  |  |  |  | if ( *ptr == ' ' || *ptr == '\t' ) continue; | 
|  |  | 50 |  |  |  |  |  | 
| 61 | 198 | 100 |  |  |  |  | if (need_field == 0) found_sep = 0; | 
| 62 | 98 |  |  |  |  |  | else ++found_sep; /* moving it away */ | 
| 63 | 198 |  |  |  |  |  | end_val = start_val = ptr; | 
| 64 | 198 |  |  |  |  |  | found_field = 0; | 
| 65 |  |  |  |  |  |  | } | 
| 66 |  |  |  |  |  |  |  | 
| 67 |  |  |  |  |  |  | /* get to the first valuable char of the line */ | 
| 68 | 6203 | 100 |  |  |  |  | if ( found_eol ) { /* starting a line */ | 
| 69 |  |  |  |  |  |  | /* spaces at the beginning of a line */ | 
| 70 | 331 | 50 |  |  |  |  | if ( *ptr == ' ' || *ptr == '\t' || *ptr == line_feed ) | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
| 71 | 0 |  |  |  |  |  | continue; | 
| 72 | 331 | 100 |  |  |  |  | if ( *ptr == comment ) { | 
| 73 | 36 |  |  |  |  |  | found_comment = 1; | 
| 74 | 36 |  |  |  |  |  | continue; | 
| 75 |  |  |  |  |  |  | } | 
| 76 |  |  |  |  |  |  | /* we have a real character to start the line */ | 
| 77 | 295 |  |  |  |  |  | found_eol = 0; | 
| 78 | 295 |  |  |  |  |  | start_key = ptr; | 
| 79 | 295 |  |  |  |  |  | end_key   = 0; | 
| 80 | 295 |  |  |  |  |  | end_val   = 0; | 
| 81 | 295 |  |  |  |  |  | found_sep = 0; | 
| 82 | 295 |  |  |  |  |  | start_val = 0; | 
| 83 | 295 |  |  |  |  |  | found_field = 0; | 
| 84 |  |  |  |  |  |  | } | 
| 85 |  |  |  |  |  |  |  | 
| 86 | 6167 | 100 |  |  |  |  | if ( *ptr == sep ) { | 
| 87 |  |  |  |  |  |  | //printf ("# separator key/value\n" ); | 
| 88 | 990 | 100 |  |  |  |  | if (need_field) ++found_sep; | 
| 89 | 990 | 100 |  |  |  |  | if ( !end_key  ) { | 
| 90 | 234 |  |  |  |  |  | end_key = ptr; | 
| 91 | 234 | 100 |  |  |  |  | if ( !need_field) found_sep = 1; | 
| 92 |  |  |  |  |  |  | } | 
| 93 |  |  |  |  |  |  |  | 
| 94 | 990 | 100 |  |  |  |  | if ( need_field && found_sep == need_field + 2 ) { | 
|  |  | 100 |  |  |  |  |  | 
| 95 | 80 |  |  |  |  |  | end_val = ptr; | 
| 96 | 990 |  |  |  |  |  | found_field = 1; | 
| 97 |  |  |  |  |  |  | } | 
| 98 |  |  |  |  |  |  |  | 
| 99 | 5177 | 100 |  |  |  |  | }  else if ( *ptr == eol ) { /* only handle the line once we reach a \n */ | 
| 100 |  |  |  |  |  |  |  | 
| 101 |  |  |  |  |  |  | #define __PARSE_STRING_LINE_FIELD /* reuse code for the last line */ \ | 
| 102 |  |  |  |  |  |  | if ( ( need_field == 0 || found_field == 0) && end_val == start_val) end_val = ptr; \ | 
| 103 |  |  |  |  |  |  | if (end_val && *end_val == line_feed) end_val = ptr - 1; \ | 
| 104 |  |  |  |  |  |  | found_eol = 1; \ | 
| 105 |  |  |  |  |  |  | \ | 
| 106 |  |  |  |  |  |  | /* check if we got a key */ \ | 
| 107 |  |  |  |  |  |  | if ( end_key > start_key ) { \ | 
| 108 |  |  |  |  |  |  | /* we got a key */ \ | 
| 109 |  |  |  |  |  |  | av_push(av, newSVpvn_flags( start_key, (int) (end_key - start_key), is_utf8 )); \ | 
| 110 |  |  |  |  |  |  | \ | 
| 111 |  |  |  |  |  |  | /* remove the line_feed chars if any */ \ | 
| 112 |  |  |  |  |  |  | while ( start_val && end_val > start_val && \ | 
| 113 |  |  |  |  |  |  | ( ( *(end_val - 1) == line_feed ) || ( *(end_val - 1) == ' ' ) || ( *(end_val - 1) == '\t' ) ) \ | 
| 114 |  |  |  |  |  |  | ) { \ | 
| 115 |  |  |  |  |  |  | --end_val; \ | 
| 116 |  |  |  |  |  |  | }  \ | 
| 117 |  |  |  |  |  |  | /* only add the value if we have a key */ \ | 
| 118 |  |  |  |  |  |  | if ( start_val && ( (int ) ( end_val - start_val ) ) ) { \ | 
| 119 |  |  |  |  |  |  | av_push(av, newSVpvn_flags( start_val, (int) (end_val - start_val), is_utf8 )); \ | 
| 120 |  |  |  |  |  |  | } else { \ | 
| 121 |  |  |  |  |  |  | av_push(av, &PL_sv_undef); \ | 
| 122 |  |  |  |  |  |  | } \ | 
| 123 |  |  |  |  |  |  | } \ | 
| 124 |  |  |  |  |  |  | /* end of __PARSE_STRING_LINE_FIELD */ | 
| 125 |  |  |  |  |  |  |  | 
| 126 | 284 | 100 |  |  |  |  | __PARSE_STRING_LINE_FIELD | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
|  |  | 100 |  |  |  |  |  | 
| 127 |  |  |  |  |  |  |  | 
| 128 | 277 |  |  |  |  |  | start_key = 0; | 
| 129 |  |  |  |  |  |  | } | 
| 130 |  |  |  |  |  |  |  | 
| 131 |  |  |  |  |  |  | } /* end main for loop for *ptr */ | 
| 132 |  |  |  |  |  |  |  | 
| 133 |  |  |  |  |  |  | /* handle the last entry */ | 
| 134 | 49 | 100 |  |  |  |  | if ( start_key ) { | 
| 135 | 18 | 50 |  |  |  |  | __PARSE_STRING_LINE_FIELD | 
|  |  | 0 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
| 136 |  |  |  |  |  |  | } | 
| 137 |  |  |  |  |  |  |  | 
| 138 | 49 |  |  |  |  |  | return (SV*) (newRV_noinc((SV*) av)); | 
| 139 |  |  |  |  |  |  | } | 
| 140 |  |  |  |  |  |  |  | 
| 141 |  |  |  |  |  |  | MODULE = Colon__Config       PACKAGE = Colon::Config | 
| 142 |  |  |  |  |  |  |  | 
| 143 |  |  |  |  |  |  | SV* | 
| 144 |  |  |  |  |  |  | read(sv, ...) | 
| 145 |  |  |  |  |  |  | SV *sv; | 
| 146 |  |  |  |  |  |  | CODE: | 
| 147 | 99 | 50 |  |  |  |  | if ( sv && SvPOK(sv) ) { | 
|  |  | 50 |  |  |  |  |  | 
| 148 | 50 |  |  |  |  |  | int field = 0; | 
| 149 | 50 | 100 |  |  |  |  | if ( items > 2 ) | 
| 150 | 1 |  |  |  |  |  | Perl_croak( "Too many arguments when calling 'Config::Colon::read'." ); | 
| 151 | 49 | 100 |  |  |  |  | if ( items == 2 ) { | 
| 152 | 25 |  |  |  |  |  | SV *sv_field = ST(1); | 
| 153 | 25 | 50 |  |  |  |  | if ( !SvOK(sv_field) || !SvIOK(sv_field) ) | 
|  |  | 0 |  |  |  |  |  | 
|  |  | 0 |  |  |  |  |  | 
|  |  | 50 |  |  |  |  |  | 
| 154 | 0 |  |  |  |  |  | Perl_croak( "Config::Colon::read - Second argument must be one integer." ); | 
| 155 | 25 | 50 |  |  |  |  | field = SvIV(sv_field); | 
| 156 |  |  |  |  |  |  | } | 
| 157 | 49 |  |  |  |  |  | RETVAL = _parse_string_field( sv, field ); | 
| 158 |  |  |  |  |  |  | } else { | 
| 159 | 0 |  |  |  |  |  | RETVAL = &PL_sv_undef; | 
| 160 |  |  |  |  |  |  | } | 
| 161 |  |  |  |  |  |  | OUTPUT: | 
| 162 |  |  |  |  |  |  | RETVAL | 
| 163 |  |  |  |  |  |  |  |