Annotation of gpl/axl/knife/axl-knife.c, revision 1.1.1.1

1.1       misho       1: /*
                      2:  *  Axl Knife: Console tool on top of Axl Library
                      3:  *  Copyright (C) 2007 Advanced Software Production Line, S.L.
                      4:  *
                      5:  *  This program is free software; you can redistribute it and/or
                      6:  *  modify it under the terms of the GNU General Public License as
                      7:  *  published by the Free Software Foundation; either version 2.1 of
                      8:  *  the License, or (at your option) any later version.
                      9:  *
                     10:  *  This program is distributed in the hope that it will be useful,
                     11:  *  but WITHOUT ANY WARRANTY; without even the implied warranty of 
                     12:  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the  
                     13:  *  GNU Lesser General Public License for more details.
                     14:  *
                     15:  *  You should have received a copy of the GNU General Public
                     16:  *  License along with this program; if not, write to the Free
                     17:  *  Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
                     18:  *  02111-1307 USA
                     19:  *  
                     20:  *  For commercial support on build XML enabled solutions contact us:
                     21:  *          
                     22:  *      Postal address:
                     23:  *         Advanced Software Production Line, S.L.
                     24:  *         Edificio Alius A, Oficina 102,
                     25:  *         C/ Antonio Suarez Nº 10,
                     26:  *         Alcalá de Henares 28802 Madrid
                     27:  *         Spain
                     28:  *
                     29:  *      Email address:
                     30:  *         info@aspl.es - http://www.aspl.es/xml
                     31:  */
                     32: /* main include */
                     33: #include <axl-knife.h>
                     34: #include <errno.h>
                     35: 
                     36: /* internal errno redefinition */
                     37: #if defined(AXL_OS_WIN32)
                     38: #define S_ISLNK(m) (0)
                     39: #endif
                     40: 
                     41: #define HELP_HEADER "Axl Knife: Console Tool on top of Axl Library\n\
                     42: Copyright (C) 2007  Advanced Software Production Line, S.L.\n\n"
                     43: 
                     44: #define POST_HEADER "\n\
                     45: If you have question, bugs to report, patches, you can reach us\n\
                     46: at <axl@lists.aspl.es>."
                     47: 
                     48: /** 
                     49:  * @internal Controls if messages must be send to the console log.
                     50:  */
                     51: axl_bool        console_enabled        = axl_true;
                     52: axl_bool        console_debug          = axl_false;
                     53: axl_bool        console_debug2         = axl_false;
                     54: axl_bool        console_debug3         = axl_false;
                     55: axl_bool        console_color_debug    = axl_false;
                     56: int         axl_knife_pid          = -1;
                     57: 
                     58: /** 
                     59:  * @internal Simple macro to check if the console output is activated
                     60:  * or not.
                     61:  */
                     62: #define CONSOLE if (console_enabled) fprintf
                     63: 
                     64: /** 
                     65:  * @internal Simple macro to check if the console output is activated
                     66:  * or not.
                     67:  */
                     68: #define CONSOLEV if (console_enabled) vfprintf
                     69: 
                     70: /** 
                     71:  * @brief Allows to get the file part from the provided path.
                     72:  * 
                     73:  * @param path The path that is provided to return the file part.
                     74:  * 
                     75:  * @return A reference newly allocated to the file, or an empty string
                     76:  * that is also required to deallocate.
                     77:  */
                     78: char  * axl_knife_file_name           (const char * path)
                     79: {
                     80:        int    iterator;
                     81:        axl_return_val_if_fail (path, NULL);
                     82: 
                     83:        /* start with string length */
                     84:        iterator = strlen (path) - 1;
                     85: 
                     86:        /* lookup for the back-slash */
                     87:        while ((iterator >= 0) && (path [iterator] != '/'))
                     88:                iterator--;
                     89: 
                     90:        /* check if the file provided doesn't have any file part */
                     91:        if (iterator == -1) {
                     92:                /* return the an empty file part */
                     93:                return axl_strdup (path);
                     94:        }
                     95: 
                     96:        /* copy the base dir found */
                     97:        return axl_strdup (path + iterator + 1);
                     98: }
                     99: 
                    100: /** 
                    101:  * @brief Cleans the string provided removing all values found "-" and
                    102:  * ".", replacing them with "_".
                    103:  * 
                    104:  * @param path The path that must be cleaned.
                    105:  * 
                    106:  */
                    107: void axl_knife_clean_name           (char * path)
                    108: {
                    109:        int    iterator;
                    110:        
                    111:        /* lookup for the back-slash */
                    112:        iterator = 0;
                    113:        while (path [iterator]) {
                    114:                /* change values found */
                    115:                if (path [iterator] == '.')
                    116:                        path [iterator] = '_';
                    117:                if (path [iterator] == '-')
                    118:                        path [iterator] = '_';
                    119: 
                    120:                /* next position */
                    121:                iterator++;
                    122:        } /* end while */
                    123: 
                    124:        return;
                    125: }
                    126: 
                    127: 
                    128: /** 
                    129:  * @internal function that actually handles the console msg.
                    130:  */
                    131: void axl_knife_msg (const char * file, int line, const char * format, ...)
                    132: {
                    133:        va_list args;
                    134: 
                    135:        /* check extended console log */
                    136:        if (console_debug3) {
                    137: #if defined(AXL_OS_UNIX)       
                    138:                if (console_color_debug) {
                    139:                        CONSOLE (stderr, "(proc:%d) [\e[1;32mmsg\e[0m] (%s:%d) ", axl_knife_pid, file, line);
                    140:                } else
                    141: #endif
                    142:                        CONSOLE (stderr, "(proc:%d) [msg] (%s:%d) ", axl_knife_pid, file, line);
                    143:        } else {
                    144: #if defined(AXL_OS_UNIX)       
                    145:                if (console_color_debug) {
                    146:                        CONSOLE (stderr, "\e[1;32mI: \e[0m");
                    147:                } else
                    148: #endif
                    149:                        CONSOLE (stderr, "I: ");
                    150:        } /* end if */
                    151:        
                    152:        va_start (args, format);
                    153:        
                    154:        /* report to console */
                    155:        CONSOLEV (stderr, format, args);
                    156: 
                    157:        va_end (args);
                    158: 
                    159:        CONSOLE (stderr, "\n");
                    160:        
                    161:        fflush (stderr);
                    162:        
                    163:        return;
                    164: }
                    165: 
                    166: /** 
                    167:  * @internal function that actually handles the console access
                    168:  */
                    169: void  axl_knife_access   (const char * file, int line, const char * format, ...)
                    170: {
                    171:        va_list args;
                    172: 
                    173:        /* check extended console log */
                    174:        if (console_debug3) {
                    175: #if defined(AXL_OS_UNIX)       
                    176:                if (console_color_debug) {
                    177:                        CONSOLE (stderr, "(proc:%d) [\e[1;32mmsg\e[0m] (%s:%d) ", axl_knife_pid, file, line);
                    178:                } else
                    179: #endif
                    180:                        CONSOLE (stderr, "(proc:%d) [msg] (%s:%d) ", axl_knife_pid, file, line);
                    181:        } else {
                    182: #if defined(AXL_OS_UNIX)       
                    183:                if (console_color_debug) {
                    184:                        CONSOLE (stderr, "\e[1;32mI: \e[0m");
                    185:                } else
                    186: #endif
                    187:                        CONSOLE (stderr, "I: ");
                    188:        } /* end if */
                    189:        
                    190:        va_start (args, format);
                    191:        
                    192:        /* report to console */
                    193:        CONSOLEV (stderr, format, args);
                    194: 
                    195:        va_end (args);
                    196: 
                    197:        CONSOLE (stderr, "\n");
                    198:        
                    199:        fflush (stderr);
                    200:        
                    201:        return;
                    202: }
                    203: 
                    204: /** 
                    205:  * @internal function that actually handles the console msg (second level debug)
                    206:  */
                    207: void axl_knife_msg2 (const char * file, int line, const char * format, ...)
                    208: {
                    209:        va_list args;
                    210: 
                    211:        /* check second level debug */
                    212:        if (! console_debug2)
                    213:                return;
                    214: 
                    215:        /* check extended console log */
                    216:        if (console_debug3) {
                    217: #if defined(AXL_OS_UNIX)       
                    218:                if (exarg_is_defined ("color-debug")) {
                    219:                        CONSOLE (stderr, "(proc:%d) [\e[1;32mmsg\e[0m] (%s:%d) ", axl_knife_pid, file, line);
                    220:                } else
                    221: #endif
                    222:                        CONSOLE (stderr, "(proc:%d) [msg] (%s:%d) ", axl_knife_pid, file, line);
                    223:        } else {
                    224: #if defined(AXL_OS_UNIX)       
                    225:                if (console_color_debug) {
                    226:                        CONSOLE (stderr, "\e[1;32mI: \e[0m");
                    227:                } else
                    228: #endif
                    229:                        CONSOLE (stderr, "I: ");
                    230:        } /* end if */
                    231:        
                    232:        va_start (args, format);
                    233:        
                    234:        /* report to console */
                    235:        CONSOLEV (stderr, format, args);
                    236: 
                    237:        va_end (args);
                    238: 
                    239:        CONSOLE (stderr, "\n");
                    240:        
                    241:        fflush (stderr);
                    242:        
                    243:        return;
                    244: }
                    245: 
                    246: /** 
                    247:  * @internal function that actually handles the console wrn.
                    248:  */
                    249: void axl_knife_wrn (const char * file, int line, const char * format, ...)
                    250: {
                    251:        va_list args;
                    252:        
                    253:        /* check extended console log */
                    254:        if (console_debug3) {
                    255: #if defined(AXL_OS_UNIX)       
                    256:                if (exarg_is_defined ("color-debug")) {
                    257:                        CONSOLE (stderr, "(proc:%d) [\e[1;33m!!!\e[0m] (%s:%d) ", axl_knife_pid, file, line);
                    258:                } else
                    259: #endif
                    260:                        CONSOLE (stderr, "(proc:%d) [!!!] (%s:%d) ", axl_knife_pid, file, line);
                    261:        } else {
                    262: #if defined(AXL_OS_UNIX)       
                    263:                if (console_color_debug) {
                    264:                        CONSOLE (stderr, "\e[1;33m!: \e[0m");
                    265:                } else
                    266: #endif
                    267:                        CONSOLE (stderr, "!: ");
                    268:        } /* end if */
                    269:        
                    270:        va_start (args, format);
                    271: 
                    272:        CONSOLEV (stderr, format, args);
                    273: 
                    274:        va_end (args);
                    275: 
                    276:        CONSOLE (stderr, "\n");
                    277:        
                    278:        fflush (stderr);
                    279:        
                    280:        return;
                    281: }
                    282: 
                    283: 
                    284: /** 
                    285:  * @internal function that actually handles the console error.
                    286:  */
                    287: void axl_knife_error (const char * file, int line, const char * format, ...)
                    288: {
                    289:        va_list args;
                    290:        
                    291:        
                    292:        /* check extended console log */
                    293:        if (console_debug3) {
                    294: #if defined(AXL_OS_UNIX)       
                    295:                if (exarg_is_defined ("color-debug")) {
                    296:                        CONSOLE (stderr, "(proc:%d) [\e[1;31merr\e[0m] (%s:%d) ", axl_knife_pid, file, line);
                    297:                } else
                    298: #endif
                    299:                        CONSOLE (stderr, "(proc:%d) [err] (%s:%d) ", axl_knife_pid, file, line);
                    300:        } else {
                    301: #if defined(AXL_OS_UNIX)       
                    302:                if (console_color_debug) {
                    303:                        CONSOLE (stderr, "\e[1;31mE: \e[0m");
                    304:                } else
                    305: #endif
                    306:                        CONSOLE (stderr, "E: ");
                    307:        } /* end if */
                    308:        
                    309:        va_start (args, format);
                    310: 
                    311:        /* report to the console */
                    312:        CONSOLEV (stderr, format, args);
                    313: 
                    314:        va_end (args);
                    315: 
                    316:        CONSOLE (stderr, "\n");
                    317:        
                    318:        fflush (stderr);
                    319:        
                    320:        return;
                    321: }
                    322: 
                    323: void axl_knife_introduce_indentation (FILE * fstream, int level)
                    324: {
                    325:        int iterator;
                    326: 
                    327:        iterator = 0;
                    328:        while (iterator < level) {
                    329:                fprintf (fstream, "   ");
                    330:                iterator++;
                    331:        } /* end while */
                    332: 
                    333:        return;
                    334: }
                    335: 
                    336: axl_bool axl_knife_htmlize_iterator_node (FILE * fstream, axlNode * node, int level)
                    337: {
                    338:        
                    339:        axlItem       * item;
                    340:        int             size;
                    341:        char          * content;
                    342:        axlAttrCursor * cursor;
                    343:        int             iterator;
                    344:        axl_bool        found_content;
                    345:        
                    346:        /* introduce indentation level */
                    347:        axl_knife_introduce_indentation (fstream, level);
                    348: 
                    349:        /* print document node */
                    350:        if (axl_node_has_attributes (node)) {
                    351:                fprintf (fstream, "&lt;<span class=\"node\">%s</span> ",
                    352:                         axl_node_get_name (node));
                    353: 
                    354:                /* get the first cursor */
                    355:                cursor   = axl_node_attr_cursor_new (node);
                    356: 
                    357:                if (axl_node_num_attributes (node)  < 3) {
                    358:                        while (axl_node_attr_cursor_has_item (cursor)) {
                    359:                                
                    360:                                /* print values */
                    361:                                fprintf (fstream, "%s=<span class=\"attrvalue\">\"%s\"</span> ",
                    362:                                         axl_node_attr_cursor_get_key (cursor),
                    363:                                         axl_node_attr_cursor_get_value (cursor));
                    364:                                
                    365:                                /* get the next cursor */
                    366:                                axl_node_attr_cursor_next (cursor);
                    367:                        } /* end while */
                    368:                } else {
                    369:                        while (axl_node_attr_cursor_has_item (cursor)) {
                    370: 
                    371:                                /* print values */
                    372:                                fprintf (fstream, "%s=<span class=\"attrvalue\">\"%s\"</span> ",
                    373:                                         axl_node_attr_cursor_get_key (cursor),
                    374:                                         axl_node_attr_cursor_get_value (cursor));
                    375: 
                    376:                                /* get the next cursor */
                    377:                                axl_node_attr_cursor_next (cursor);
                    378: 
                    379:                                /* before getting the next */
                    380:                                if (axl_node_attr_cursor_has_item (cursor)) {
                    381:                                        fprintf (fstream, "\n");
                    382: 
                    383:                                        /* introduce indentation level */
                    384:                                        axl_knife_introduce_indentation (fstream, level);
                    385: 
                    386:                                        iterator = 0;
                    387:                                        while (iterator < (strlen (axl_node_get_name (node)) + 2)) {
                    388:                                                fprintf (fstream, " ");
                    389:                                                iterator++;
                    390:                                        } /* while */
                    391:                                } /* end if */
                    392: 
                    393:                        } /* end while */
                    394:                }
                    395:                        
                    396:                /* free cursor */
                    397:                axl_node_attr_cursor_free (cursor);
                    398: 
                    399:                /* check if the node have not child on any kind ... */
                    400:                if (axl_item_get_first_child (node) == NULL) {
                    401:                        fprintf (fstream, "/>\n");
                    402:                        return axl_true;
                    403:                } else {
                    404:                        fprintf (fstream, ">\n");
                    405:                }
                    406:                
                    407:        } else {
                    408:                /* check if the node has childs or only content to avoid placing a \n */
                    409:                fprintf (fstream, "&lt;<span class=\"node\">%s</span>>%s",
                    410:                         axl_node_get_name (node),
                    411:                         axl_node_have_childs (node) ? "\n" : "");
                    412:        }
                    413: 
                    414:        /* call to produce internal content representation */
                    415:        item          = axl_item_get_first_child (node);
                    416:        found_content = axl_false;
                    417:        while (item != NULL) {
                    418:                found_content = axl_true;
                    419:                /* according to the type do */
                    420:                switch (axl_item_get_type (item)) {
                    421:                case ITEM_NODE:
                    422:                        /* found node, call to represent this node */
                    423:                        axl_knife_htmlize_iterator_node (fstream, axl_item_get_data (item), level + 1);
                    424:                        break;
                    425:                case ITEM_CONTENT_FROM_FACTORY:
                    426:                case ITEM_CONTENT:
                    427:                        size    = 0;
                    428:                        content = axl_item_get_content (item, &size);
                    429:                        fprintf (fstream, "%s", axl_item_get_content (item, NULL));
                    430:                        break;
                    431:                case ITEM_PI:
                    432:                        break;
                    433:                case ITEM_FROM_FACTORY:
                    434:                        /* never reached */
                    435:                        break;
                    436:                case ITEM_COMMENT:
                    437:                        /* introduce indentation level */
                    438:                        axl_knife_introduce_indentation (fstream, level + 1);
                    439:                        fprintf (fstream, "<span class=\"comment\">&lt;!-- %s --></span>\n",
                    440:                                 axl_item_get_content (item, NULL));
                    441:                        break;
                    442:                case ITEM_REF:
                    443:                        break;
                    444:                case ITEM_CDATA:
                    445:                        fprintf (fstream, "<span class=\"cdata\">&lt;![CDATA[</span>%s<span class=\"cdata\">]]></span>",
                    446:                                 axl_item_get_content (item, NULL));
                    447:                        break;
                    448:                } /* end switch */
                    449: 
                    450:                /* next item */
                    451:                item = axl_item_get_next (item);
                    452:        }
                    453: 
                    454: 
                    455:        if (axl_node_have_childs (node)) {
                    456:                /* introduce indentation level */
                    457:                axl_knife_introduce_indentation (fstream, level);
                    458:        }
                    459:                
                    460:        if (found_content) {
                    461:                fprintf (fstream, "&lt;/<span class=\"node\">%s</span>>\n",
                    462:                         axl_node_get_name (node));
                    463:        } /* end if */
                    464: 
                    465:        /* don't stop iteration */
                    466:        return axl_true;
                    467: }
                    468: 
                    469: axl_bool axl_knife_htmlize (axlDoc * doc)
                    470: {
                    471:        FILE       * fstream = stdout;
                    472: 
                    473:        /* currently we only support stderr so reached this place
                    474:         * means it is already checked */
                    475: 
                    476:        /* check output argument */
                    477:        if (exarg_is_defined ("output")) {
                    478:                /* abrimos el fichero de salida */
                    479:                fstream = fopen (exarg_get_string ("output"), "w");
                    480:                if (fstream == NULL) {
                    481:                        error ("unable to open output document: %s, errno=%d:%s", exarg_get_string ("output"),
                    482:                               errno, strerror (errno));
                    483:                        return axl_false;
                    484:                } /* end if */
                    485:        } /* end if */
                    486: 
                    487:        /* call to iterate */
                    488:        if (exarg_is_defined ("pre-class")) 
                    489:                fprintf (fstream, "<pre class='%s'>\n", exarg_get_string ("pre-class"));
                    490:        else
                    491:                fprintf (fstream, "<pre>\n");
                    492:        axl_knife_htmlize_iterator_node (fstream, axl_doc_get_root (doc), 0);
                    493:        fprintf (fstream, "</pre>\n");
                    494: 
                    495:        return axl_true;
                    496: }
                    497: 
                    498: axl_bool axl_knife_dtd_to_c ()
                    499: {
                    500:        
                    501:        /* check the document received is a DTD */
                    502:        axlError   * err = NULL;
                    503:        axlDtd     * dtd = axl_dtd_parse_from_file (exarg_get_string ("input"), &err);
                    504:        axlStream  * stream;
                    505:        char       * file;
                    506:        int          line_length;
                    507:        int          max;
                    508:        int          chunk_matched;
                    509:        char       * ref;
                    510:        char       * format;
                    511:        int          iterator;
                    512:        FILE       * fstream = stdout;
                    513: 
                    514:        if (dtd == NULL) {
                    515:                error ("unable to translate document into C, found invalid DTD: %s",
                    516:                       axl_error_get (err));
                    517:                axl_error_free (err);
                    518:                return axl_false;
                    519:        } /* end if */
                    520:        
                    521:        /* free dtd document */
                    522:        axl_dtd_free (dtd);
                    523: 
                    524:        /* open the stream */
                    525:        stream = axl_stream_new (NULL, -1, exarg_get_string ("input"), -1, &err);
                    526: 
                    527:        /* check output argument */
                    528:        if (exarg_is_defined ("output")) {
                    529:                /* abrimos el fichero de salida */
                    530:                fstream = fopen (exarg_get_string ("output"), "w");
                    531:                if (fstream == NULL) {
                    532:                        error ("unable to open output document: %s, errno=%d:%s", exarg_get_string ("output"),
                    533:                               errno, strerror (errno));
                    534:                        return axl_false;
                    535:                } /* end if */
                    536:        } /* end if */
                    537:        
                    538:        /* try to get the maximum lenght */
                    539:        line_length = 0;
                    540:        do {
                    541:                /* get next */
                    542:                ref = axl_stream_get_until_zero (stream, NULL, &chunk_matched, axl_true, 1, "\n", NULL);
                    543: 
                    544:                /* get lenght and update */
                    545:                max = ref ? strlen (ref) : 0;
                    546:                if (max > line_length)
                    547:                        line_length = max;
                    548: 
                    549:        } while (chunk_matched != -2);
                    550: 
                    551:        /* close the stream */
                    552:        axl_stream_free (stream);
                    553:        stream = axl_stream_new (NULL, -1, exarg_get_string ("input"), -1, &err);
                    554:        
                    555:        /* build format for each line */
                    556:        format = axl_strdup_printf ("%%-%ds  \\\n", line_length + 1);
                    557: 
                    558:        file   = axl_knife_file_name (exarg_get_string ("input"));
                    559: 
                    560:        
                    561:        
                    562:        fprintf (fstream, "/**\n");
                    563:        fprintf (fstream, " * C inline representation for DTD %s, created by axl-knife\n", file);
                    564:        fprintf (fstream, " */\n");
                    565: 
                    566:        axl_knife_clean_name (file);
                    567:        axl_stream_to_upper (file);
                    568:        
                    569:        fprintf (fstream, "#ifndef __%s_H__\n", file);
                    570:        fprintf (fstream, "#define __%s_H__\n", file);
                    571:        fprintf (fstream, "#define %s \"\\n\\\n", file);
                    572:        do {
                    573:                /* get next */
                    574:                ref      = axl_stream_get_until_zero (stream, NULL, &chunk_matched, axl_true, 1, "\n", NULL);
                    575:                if (ref != NULL) {
                    576:                        iterator = 0;
                    577:                        while (iterator < strlen (ref)) {
                    578:                                /* update blank */
                    579:                                if (ref[iterator] == '\t')
                    580:                                        ref[iterator] = ' ';
                    581:                                if (ref[iterator] == '\"')
                    582:                                        ref[iterator] = '\'';
                    583: 
                    584:                                /* next iterator */
                    585:                                iterator++;
                    586:                        } /* end while */
                    587:                } /* end if */
                    588: 
                    589:                /* get lenght and update */
                    590:                fprintf (fstream, format, ref);
                    591: 
                    592:        } while (chunk_matched != -2);
                    593: 
                    594: 
                    595: 
                    596:        fprintf (fstream, "\\n\"\n");
                    597: 
                    598:        fprintf (fstream, "#endif\n");
                    599:        
                    600:        axl_stream_free (stream);
                    601:        axl_free (file);
                    602:        axl_free (format);
                    603: 
                    604:        /* check output argument */
                    605:        if (exarg_is_defined ("output"))
                    606:                fclose (fstream);
                    607: 
                    608:        return axl_true;
                    609: }
                    610: 
                    611: /** 
                    612:  * @brief Allows to check if the provided file (basefile) is newer
                    613:  * than the file (compare).
                    614:  * 
                    615:  * @param basefile The base file to check.
                    616:  * @param compare The compare file to check
                    617:  * 
                    618:  * @return axl_true if the modification time is newer than compare,
                    619:  * otherwise axl_false is returned.
                    620:  */
                    621: axl_bool axl_knife_check_if_newer (const char * basefile, const char * compare)
                    622: {
                    623:        struct stat stat1, stat2;
                    624:        
                    625:        /* get stats from both files */
                    626:        if (stat (basefile, &stat1) != 0)
                    627:                return axl_false;
                    628:        if (stat (compare, &stat2) != 0)
                    629:                return axl_false;
                    630: 
                    631:        /* return value comparation */
                    632:        return stat1.st_mtime > stat2.st_mtime;
                    633: }
                    634: 
                    635: 
                    636: int main (int argc, char ** argv)
                    637: {
                    638: 
                    639:        axlDoc   * doc = NULL;
                    640:        axlError * err = NULL;
                    641: 
                    642:        /* init the axl library */
                    643:        if (! axl_init ()) {
                    644:                printf ("Failed to initialize axl library. Terminating axl-knife tool..\n");
                    645:                return -1;
                    646:        } /* end if */
                    647: 
                    648:        /* install headers for help */
                    649:        exarg_add_usage_header  (HELP_HEADER);
                    650:        exarg_add_help_header   (HELP_HEADER);
                    651:        exarg_post_help_header  (POST_HEADER);
                    652:        exarg_post_usage_header (POST_HEADER);
                    653: 
                    654:        /* init exarg library */
                    655:        exarg_install_arg ("version", "v", EXARG_NONE, 
                    656:                           "Shows current axl-knife version.");
                    657: 
                    658:        /* file options */
                    659:        exarg_install_arg ("input", "i", EXARG_STRING, 
                    660:                           "Allows to configure the input document to process.");
                    661:        exarg_install_arg ("output", "o", EXARG_STRING,
                    662:                           "Allows to configure the output file to be used. This is optional. If the knife command produces an output, it is by default sent to the stdout.");
                    663:        exarg_install_arg ("ifnewer", "n", EXARG_NONE,
                    664:                           "In process involving generating output from an input file, this option allows to stop the process and return success code (0) if the output file is found to be newer than input file. ");
                    665:                           
                    666: 
                    667:        exarg_install_arg ("htmlize", "e", EXARG_NONE,
                    668:                           "Takes an input xml document and produces an transformation preparing the document to be included into an html web page");
                    669: 
                    670:        exarg_install_arg ("pre-class", "p", EXARG_STRING,
                    671:                           "In conjunction with --htmlize option, which option allows to configure the CSS class to be placed on top most <pre> node.");
                    672:        
                    673:        /* log options */
                    674:        exarg_install_arg ("enable-log", "l", EXARG_NONE,
                    675:                           "Allows to activate the console log debug.");
                    676: 
                    677:        exarg_install_arg ("enable-log-color", "g", EXARG_NONE,
                    678:                           "Activates the console logs and uses some ansi characters to colorify the log output. If this option is activated, it is implicitly activated the --enable-log");
                    679:        
                    680:        /* dtd-to-c options */
                    681:        exarg_install_arg ("dtd-to-c", NULL, EXARG_NONE,
                    682:                           "Creates a C header definition representing the DTD provided, suitable to be opened by libaxl");
                    683: 
                    684:        exarg_install_arg ("check-xml", "c", EXARG_NONE,
                    685:                           "Allows to check an xml document (if it is properly formated). Combine this option with --input");
                    686: 
                    687:        /* add dependecies */
                    688:        exarg_add_dependency ("htmlize", "input");
                    689:        exarg_add_dependency ("pre-class", "htmlize");
                    690:        exarg_add_dependency ("dtd-to-c", "input");
                    691:        exarg_add_dependency ("check-xml", "input");
                    692:        
                    693:        exarg_add_dependency ("ifnewer", "input");
                    694:        exarg_add_dependency ("ifnewer", "output");
                    695:        
                    696:        /* exclusion */
                    697:        exarg_add_exclusion ("htmlize", "dtd-to-c");
                    698:        
                    699:        /* call to parse arguments */
                    700:        exarg_parse (argc, argv);
                    701: 
                    702:        /* get current process id */
                    703:        axl_knife_pid = getpid ();
                    704: 
                    705:        /* normal operations */
                    706:        if (exarg_is_defined ("version")) {
                    707:                return printf ("%s\n", VERSION);
                    708:        }
                    709: 
                    710:        /* check ifnewer option */
                    711:        if (exarg_is_defined ("ifnewer")) {
                    712:                /* check that both files exists */
                    713:                if (axl_knife_file_test (exarg_get_string ("input"), FILE_EXISTS) &&
                    714:                    axl_knife_file_test (exarg_get_string ("output"), FILE_EXISTS)) {
                    715:                        if (! axl_knife_check_if_newer (exarg_get_string ("input"), exarg_get_string ("output"))) {
                    716:                                goto finish;
                    717:                        } /* end if */
                    718:                } /* end if */
                    719:        } /* end if */
                    720: 
                    721:        /* check parameters defined */
                    722:        if (! exarg_is_defined ("htmlize") &&
                    723:            ! exarg_is_defined ("check-xml") &&
                    724:            ! exarg_is_defined ("dtd-to-c")) {
                    725:                msg ("no action was defined..");
                    726:                goto finish;
                    727:        }
                    728:                
                    729:        /* parse log options */
                    730:        axl_log_enable (exarg_is_defined ("enable-log"));
                    731: 
                    732:        /* check color to activate both logs */
                    733:        if (exarg_is_defined ("enable-log-color")) {
                    734:                axl_log_enable (axl_true);
                    735:                axl_log_color_enable (axl_true);
                    736:        }
                    737: 
                    738:        /* check to load a document */
                    739:        if (exarg_is_defined ("dtd-to-c")) {
                    740:                /* do not open any document, as it is opened by an axl
                    741:                 * stream directly */
                    742:        } else if (exarg_is_defined ("input")) {
                    743:                /* parse document from file */
                    744:                doc = axl_doc_parse_from_file (exarg_get_string ("input"), &err);
                    745:                if (doc == NULL) {
                    746:                        error ("Failed to open document (%s), error found: %s",
                    747:                               exarg_get_string ("input"), axl_error_get (err));
                    748:                        axl_error_free (err);
                    749:                        goto finish;
                    750:                } /* end if */
                    751:                msg ("document loaded properly: %s", exarg_get_string ("input"));
                    752:        } /* end if */
                    753: 
                    754:        
                    755:        /* check process options */
                    756:        if (exarg_is_defined ("htmlize")) {
                    757:                /* call to htmlize the content received */
                    758:                axl_knife_htmlize (doc);
                    759:        } else if (exarg_is_defined ("dtd-to-c")) {
                    760: 
                    761:                /* call to produce a C representation from the DTD
                    762:                 * provided */
                    763:                axl_knife_dtd_to_c ();
                    764:        } else if (exarg_is_defined ("check-xml")) {
                    765:                /* do nothing because if reached this code, document
                    766:                 * was loadedd properly */
                    767:                msg ("document is OK");
                    768:        } /* end if */
                    769: 
                    770: 
                    771:  finish:
                    772:        /* dealloc document opened */
                    773:        axl_doc_free (doc);
                    774: 
                    775:        /* terminating axl */
                    776:        axl_end ();
                    777: 
                    778:        /* terminate exarg */
                    779:        exarg_end ();
                    780: 
                    781:        return 0;
                    782: }
                    783: 
                    784: /** 
                    785:  * @brief Allows to perform a set of test for the provided path.
                    786:  * 
                    787:  * @param path The path that will be checked.
                    788:  *
                    789:  * @param test The set of test to be performed. Separate each test
                    790:  * with "|" to perform several test at the same time.
                    791:  * 
                    792:  * @return axl_true if all test returns axl_true. Otherwise axl_false is returned.
                    793:  */
                    794: axl_bool   axl_knife_file_test (const char * path, FileTest test)
                    795: {
                    796:        axl_bool result = axl_false;
                    797:        struct stat file_info;
                    798: 
                    799:        /* perform common checks */
                    800:        axl_return_val_if_fail (path, axl_false);
                    801: 
                    802:        /* call to get status */
                    803:        result = (stat (path, &file_info) == 0);
                    804:        if (! result) {
                    805:                /* check that it is requesting for not file exists */
                    806:                if (errno == ENOENT && (test & FILE_EXISTS) == FILE_EXISTS)
                    807:                        return axl_false;
                    808: 
                    809:                error ("filed to check test on %s, stat call has failed (result=%d, error=%s)", path, result, strerror (errno));
                    810:                return axl_false;
                    811:        } /* end if */
                    812: 
                    813:        /* check for file exists */
                    814:        if ((test & FILE_EXISTS) == FILE_EXISTS) {
                    815:                /* check result */
                    816:                if (result == axl_false)
                    817:                        return axl_false;
                    818:                
                    819:                /* reached this point the file exists */
                    820:                result = axl_true;
                    821:        }
                    822: 
                    823:        /* check if the file is a link */
                    824:        if ((test & FILE_IS_LINK) == FILE_IS_LINK) {
                    825:                if (! S_ISLNK (file_info.st_mode))
                    826:                        return axl_false;
                    827: 
                    828:                /* reached this point the file is link */
                    829:                result = axl_true;
                    830:        }
                    831: 
                    832:        /* check if the file is a regular */
                    833:        if ((test & FILE_IS_REGULAR) == FILE_IS_REGULAR) {
                    834:                if (! S_ISREG (file_info.st_mode))
                    835:                        return axl_false;
                    836: 
                    837:                /* reached this point the file is link */
                    838:                result = axl_true;
                    839:        }
                    840: 
                    841:        /* check if the file is a directory */
                    842:        if ((test & FILE_IS_DIR) == FILE_IS_DIR) {
                    843:                if (! S_ISDIR (file_info.st_mode)) {
                    844:                        return axl_false;
                    845:                }
                    846: 
                    847:                /* reached this point the file is link */
                    848:                result = axl_true;
                    849:        }
                    850: 
                    851:        /* return current result */
                    852:        return result;
                    853: }
                    854: 
                    855: /** 
                    856:  * \page axl_knife_manual axl-knife: command line tool built on top of Axl.
                    857:  *
                    858:  * \section intro Introduction
                    859:  *
                    860:  * <b>axl-knife</b> is a command line tool that includes xml related
                    861:  * features and the intention is to concentrate into the tool,
                    862:  * features to make Axl Library available to the command line.
                    863:  *
                    864:  * This manual includes information about operations that support the
                    865:  * tool and how to activate them:
                    866:  *
                    867:  *  - \ref axl_knife_dtd_to_c
                    868:  *  - \ref axl_knife_htmlize
                    869:  * 
                    870:  * \section axl_knife_dtd_to_c Building a C inline representation from a DTD file
                    871:  *
                    872:  * This feature allows to create a C header file representing a
                    873:  * particular DTD file, so it can be used and parsed as it were a file
                    874:  * by included into the source.
                    875:  * 
                    876:  * The main advantage from this approach is that the application or
                    877:  * library using Axl Library services do not have to bundle DTD files
                    878:  * into its installers. Those files are already included into the
                    879:  * source. In many cases, those DTD files are than less than 1K.
                    880:  * 
                    881:  * To create a C DTD representation use:
                    882:  * \code
                    883:  * >> axl-knife --input DTD-FILE.dtd --dtd-to-c --output DTD-FILE.dtd.h --ifnewer
                    884:  * \endcode
                    885:  *
                    886:  * The argument <b>--ifnewer</b> allows to signal the tool to not
                    887:  * produce any output if the input file is not found to be newer. 
                    888:  *
                    889:  * Now use \ref axl_dtd_parse to open a reference to the DTD.
                    890:  *
                    891:  * \section axl_knife_htmlize Prepare xml content to be included into xml
                    892:  *
                    893:  * In the case you require to include xml content into a web page, you
                    894:  * can use <b>--htmlize</b> option to prepare the content, adding CSS
                    895:  * style classes to configure how your xml document will look.
                    896:  *
                    897:  * Use the following to produce xml content ready to be included into html:
                    898:  *
                    899:  * \code
                    900:  * >> axl-knife --input xml-file.xml --htmlize
                    901:  * \endcode
                    902:  *
                    903:  */

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>