File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / libpdel / tmpl / tmpl.h
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Tue Feb 21 23:25:53 2012 UTC (13 years, 1 month ago) by misho
Branches: libpdel, MAIN
CVS tags: v0_5_3, HEAD
libpdel

    1: 
    2: /*
    3:  * Copyright (c) 2001-2002 Packet Design, LLC.
    4:  * All rights reserved.
    5:  * 
    6:  * Subject to the following obligations and disclaimer of warranty,
    7:  * use and redistribution of this software, in source or object code
    8:  * forms, with or without modifications are expressly permitted by
    9:  * Packet Design; provided, however, that:
   10:  * 
   11:  *    (i)  Any and all reproductions of the source or object code
   12:  *         must include the copyright notice above and the following
   13:  *         disclaimer of warranties; and
   14:  *    (ii) No rights are granted, in any manner or form, to use
   15:  *         Packet Design trademarks, including the mark "PACKET DESIGN"
   16:  *         on advertising, endorsements, or otherwise except as such
   17:  *         appears in the above copyright notice or in the software.
   18:  * 
   19:  * THIS SOFTWARE IS BEING PROVIDED BY PACKET DESIGN "AS IS", AND
   20:  * TO THE MAXIMUM EXTENT PERMITTED BY LAW, PACKET DESIGN MAKES NO
   21:  * REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED, REGARDING
   22:  * THIS SOFTWARE, INCLUDING WITHOUT LIMITATION, ANY AND ALL IMPLIED
   23:  * WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE,
   24:  * OR NON-INFRINGEMENT.  PACKET DESIGN DOES NOT WARRANT, GUARANTEE,
   25:  * OR MAKE ANY REPRESENTATIONS REGARDING THE USE OF, OR THE RESULTS
   26:  * OF THE USE OF THIS SOFTWARE IN TERMS OF ITS CORRECTNESS, ACCURACY,
   27:  * RELIABILITY OR OTHERWISE.  IN NO EVENT SHALL PACKET DESIGN BE
   28:  * LIABLE FOR ANY DAMAGES RESULTING FROM OR ARISING OUT OF ANY USE
   29:  * OF THIS SOFTWARE, INCLUDING WITHOUT LIMITATION, ANY DIRECT,
   30:  * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, PUNITIVE, OR CONSEQUENTIAL
   31:  * DAMAGES, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, LOSS OF
   32:  * USE, DATA OR PROFITS, HOWEVER CAUSED AND UNDER ANY THEORY OF
   33:  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
   34:  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
   35:  * THE USE OF THIS SOFTWARE, EVEN IF PACKET DESIGN IS ADVISED OF
   36:  * THE POSSIBILITY OF SUCH DAMAGE.
   37:  *
   38:  * Author: Archie Cobbs <archie@freebsd.org>
   39:  */
   40: 
   41: #ifndef _PDEL_TMPL_TMPL_H_
   42: #define _PDEL_TMPL_TMPL_H_
   43: 
   44: /*
   45:  * This code supports a simple text template language. An input file
   46:  * is scanned for special function tags. A function tag is an at sign
   47:  * ('@') followed by a contiguous sequence of letters, digits, and
   48:  * underscores, followed by parentheses containing zero or more
   49:  * function arguments.
   50:  *
   51:  * Function arguments may be either other function tags (the argument
   52:  * to the outer function is the result of the inner function invocation),
   53:  * or constant literal strings in double quotes (the argument is the
   54:  * value of the string). Note this means that all function arguments
   55:  * begin with either an at sign or a double quote character. Function
   56:  * arguments are enclosed in parentheses, separated by commas, and
   57:  * may have whitespace around them.
   58:  *
   59:  * When processed, function tags result in a function invocation
   60:  * which returns some text which is then substituted into the output
   61:  * stream. Special control flow function tags control input processing
   62:  * and return the empty string.
   63:  *
   64:  * Non-function tag text input is passed through unaltered.
   65:  *
   66:  * Constant literal strings (enclosed in double quotes) respect the
   67:  * normal C backslash escapes.
   68:  *
   69:  * Built-in functions that take no arguments do not require parentheses,
   70:  * but parentheses may be included for separation purposes.
   71:  *
   72:  * User-supplied functions may return NULL and set errno to indicate
   73:  * an error, in which case the corresponding error string is substituted,
   74:  * after being formatted by the caller-supplied error formatter.
   75:  *
   76:  * The predefined functions are:
   77:  *
   78:  *   @while(x) ... @endwhile
   79:  *	The text in between is repeated as long as x is true
   80:  *	(x is true if it is an integer whose value is non-zero).
   81:  *
   82:  *   @loop(numloops) ... @endloop
   83:  *	The text in between is repeated 'numloops' times, which
   84:  *	must be an integer value.
   85:  *
   86:  *   @loopindex(i)
   87:  *	Returns the loop index (counting from zero) of the loop
   88:  *	that is 'i' loops out from the innermost loop, else -1.
   89:  *	If argument is omitted it is assumed to be zero.
   90:  *
   91:  *   @if(x) ... [ @elif ... [ @elif ... ] ] [ @else ... ] @endif
   92:  *	Conditional execution depending on the truth value of x.
   93:  *
   94:  *   @define(name) ... @enddef
   95:  *	Defines a run-time function. The text in between is executed
   96:  *	whenever @name(...) is subsequently invoked. During this execution,
   97:  *	the variables "argc" and "arg0", "arg1", ... are set to the
   98:  *	function argument count and arguments, respectively (the function
   99:  *	name itself is always the value of "arg0".
  100:  *
  101:  *   @break
  102:  *	Break out of the nearest enclosing @loop or @while loop.
  103:  *
  104:  *   @continue
  105:  *	Continue with the next iteration of the nearest enclosing
  106:  *	@loop or @while loop.
  107:  *
  108:  *   @return
  109:  *	Return from within a run-time function.
  110:  *
  111:  *   @equal(x,y)
  112:  *	Returns "1" if x and y are identical, else "0".
  113:  *
  114:  *   @not(x)
  115:  *	Returns "1" unless x is an integer whose value is non-zero.
  116:  *
  117:  *   @and(arg1, arg2, ...)
  118:  *	Returns logical "and" of the truth values of the arguments.
  119:  *
  120:  *   @or(arg1, arg2, ...)
  121:  *	Returns logical "or" of the truth values of the arguments.
  122:  *
  123:  *   @add(arg1, ...)
  124:  *	Returns sum of the numerical values of the arguments.
  125:  *
  126:  *   @mul(arg1, ...)
  127:  *	Returns product of the numerical values of the arguments.
  128:  *
  129:  *   @div(arg1, arg2)
  130:  *	Returns the first argument divided by the second.
  131:  *
  132:  *   @mod(arg1, arg2)
  133:  *	Returns the first argument modulo by the second.
  134:  *
  135:  *   @lt(arg1, arg2)
  136:  *   @gt(arg1, arg2)
  137:  *   @le(arg1, arg2)
  138:  *   @ge(arg1, arg2)
  139:  *	Returns "0" or "1" as arg1 is <, >, <=, or >= arg2.
  140:  *
  141:  *   @sub(arg1, ...)
  142:  *	Returns subtraction of the numerical values of the second
  143:  *	and subsequent arguments from the first.
  144:  *
  145:  *   @error(arg)
  146:  *	Output the argument using the caller's error formatting.
  147:  *
  148:  *   @eval(arg)
  149:  *	The argument itself is processed. Embedded function tags
  150:  *	are processed normally, while the rest is passed unaltered.
  151:  *
  152:  *   @funcname(arg1, arg2, ...)
  153:  *	Invoke the generic user-supplied function handler.
  154:  *
  155:  *   @@
  156:  *	Expands to ``@''
  157:  *
  158:  *   @cat(arg1, arg2, ...)
  159:  *	Returns concatentated arguments.
  160:  *
  161:  *   @set(var, value)
  162:  *	Sets run-time variable 'var' to have value 'value'. Variables
  163:  *	are global and live until tmpl_process() returns. Returns the
  164:  *	empty string.
  165:  *
  166:  *   @get(var)
  167:  *	Retrieves the value of run-time variable 'var'.
  168:  *	Returns the empty string if 'var' is not set.
  169:  *
  170:  *   @urlencode(arg)
  171:  *	Expands and URL-encodes the argument.
  172:  *
  173:  *   @htmlencode(arg)
  174:  *	Expands and HTML-encodes the first argument.
  175:  *
  176:  *   @output(arg)
  177:  *	Outputs the argument directly to the output stream. That is,
  178:  *	if this function is invoked from within a user-defined function,
  179:  *	the argument goes directly to the template output rather than
  180:  *	being concatenated to the return value of the function.
  181:  *
  182:  *   @flush()
  183:  *	Flushes the template output.
  184:  *
  185:  *   @foobar(...)
  186:  *	If "foobar" is not defined as a run-time function, then this
  187:  *	invokes the supplied handler for user-defined functions.
  188:  *	Here "foobar" is any name other than the above function names,
  189:  *	consisting of only letters, digits, and underscores.
  190:  *
  191:  */
  192: 
  193: struct tmpl;
  194: struct tmpl_ctx;
  195: 
  196: #define TMPL_FUNCTION_CHARACTER		'@'
  197: 
  198: /***********************************************************************
  199: 			CALLBACK FUNCTION TYPES
  200: ***********************************************************************/
  201: 
  202: /*
  203:  * Generic function handler type. Should return a malloc'd string (allocated
  204:  * with the same memory type as passed to tmpl_ctx_create()), or else return
  205:  * NULL and either set errno or set *errmsgp to point to a malloc'd string
  206:  * (same memory type) containing an error message. *errmsgp will be NULL
  207:  * initially.
  208:  *
  209:  * The first argument is the function name. Subsequent arguments are the
  210:  * (evaluated) arguments passed to the function. Note that this means "argc"
  211:  * is always >= 1. argv[argc] is NOT guaranteed to be NULL.
  212:  */
  213: typedef char	*tmpl_handler_t(struct tmpl_ctx *ctx, char **errmsgp,
  214: 			int argc, char **argv);
  215: 
  216: /*
  217:  * Error formatter type. This should format the error message string
  218:  * using whatever formatting is desired and return the result in a
  219:  * malloc'd buffer (note: just returning 'errmsg' is wrong; use
  220:  * strdup(3) instead). Return NULL and set errno if there is an error.
  221:  * The "arg" parameter is the same "arg" passed to tmpl_ctx_create().
  222:  */
  223: typedef char	*tmpl_errfmtr_t(struct tmpl_ctx *ctx, const char *errmsg);
  224: 
  225: /***********************************************************************
  226: 			PUBLIC FUNCTIONS
  227: ***********************************************************************/
  228: 
  229: __BEGIN_DECLS
  230: 
  231: /*
  232:  * Create template object by parsing "input". The input stream must
  233:  * be seekable, and is not used again after this function returns.
  234:  *
  235:  * If num_errors != NULL, it will be set to the number of parse
  236:  * errors encountered while parsing the input.
  237:  *
  238:  * Returns NULL and sets errno on failure.
  239:  */
  240: extern struct	tmpl *tmpl_create(FILE *input,
  241: 			int *num_errors, const char *mtype);
  242: 
  243: /*
  244:  * Similar to tmpl_create(), but memory map the file so that its
  245:  * contents don't have to be stored in heap memory. However, the
  246:  * file contents must not change or else subsequent executions will
  247:  * give garbled output.
  248:  */
  249: extern struct	tmpl *tmpl_create_mmap(const char *path,
  250: 			int *num_errors, const char *mtype);
  251: 
  252: /*
  253:  * Destroy and free a template file created by tmpl_create().
  254:  *
  255:  * Upon return, *tmplp is set to NULL. If *tmplp is already NULL,
  256:  * this does nothing.
  257:  */
  258: extern void	tmpl_destroy(struct tmpl **tmplp);
  259: 
  260: /*
  261:  * Create a template execution context.
  262:  *
  263:  * For all functions that are not predefined, the function is looked up
  264:  * in the 'userfuncs' array (which is not copied and so must remain
  265:  * valid for the life of the exectution context). This array must be
  266:  * sorted by name and terminated with an entry having a NULL name.
  267:  * All userfuncs will have 'arg' passed as their first argument.
  268:  *
  269:  * All strings returned by "handler" MUST be dynamically allocated
  270:  * using memory type 'mtype'.
  271:  *
  272:  * The 'errfmtr' is called to specially format any generated error
  273:  * messages; if it is NULL, error messages are output as plain text.
  274:  */
  275: extern struct	tmpl_ctx *tmpl_ctx_create(void *arg, const char *mtype,
  276: 			tmpl_handler_t *handler, tmpl_errfmtr_t *errfmtr);
  277: 
  278: /*
  279:  * Get context argument.
  280:  */
  281: extern void	*tmpl_ctx_get_arg(struct tmpl_ctx *ctx);
  282: 
  283: /*
  284:  * Get context memory type.
  285:  */
  286: extern const	char *tmpl_ctx_get_mtype(struct tmpl_ctx *ctx);
  287: 
  288: /*
  289:  * Execute a template and output the result to 'output'.
  290:  *
  291:  * If a system error is encountered, execution halts and -1
  292:  * is returned with 'errno' set. Otherwise, zero is returned.
  293:  *
  294:  * Flags:
  295:  *
  296:  *   TMPL_SKIP_NL_WHITE		Skip inter-function text consisting only of
  297:  *				one or more newlines followed by whitespace.
  298:  */
  299: extern int	tmpl_execute(struct tmpl *tmpl, struct tmpl_ctx *ctx,
  300: 			FILE *output, int flags);
  301: 
  302: #define TMPL_SKIP_NL_WHITE	0x0001
  303: 
  304: /*
  305:  * Destroy and free a template context created by tmpl_ctx_create().
  306:  *
  307:  * Upon return, *ctxp is set to NULL. If *ctxp is already NULL,
  308:  * this does nothing.
  309:  */
  310: extern void	tmpl_ctx_destroy(struct tmpl_ctx **ctxp);
  311: 
  312: /*
  313:  * Functions to get and set variables associated with a context.
  314:  */
  315: extern const	char *tmpl_ctx_get_var(struct tmpl_ctx *ctx, const char *name);
  316: extern int	tmpl_ctx_set_var(struct tmpl_ctx *ctx,
  317: 			const char *name, const char *value);
  318: 
  319: /*
  320:  * Invoke a template function and write the output to "output".
  321:  *
  322:  * Returns zero if successful, otherwise returns -1 and either
  323:  * sets *errmsgp to NULL and errno to the error code, or else
  324:  * sets *errmsgp to some appropriate error message.
  325:  */
  326: extern int	tmpl_execute_func(struct tmpl_ctx *ctx, FILE *output,
  327: 			char **errmsgp, int argc, char **argv,
  328: 			int flags);
  329: 
  330: /*
  331:  * Reset a template context.
  332:  * 
  333:  * This undoes the result of any variable assignments and run-time
  334:  * function definitions from a previous execution using the context.
  335:  * That is, the context is returned to original state as returned by
  336:  * tmpl_ctx_create().
  337:  */
  338: extern void	tmpl_ctx_reset(struct tmpl_ctx *ctx);
  339: 
  340: /*
  341:  * Built-in handler for handling a fixed list of user functions.
  342:  *
  343:  * The 'userfuncs' array must be sorted lexicographically by name.
  344:  */
  345: 
  346: /* Structure describing a user-supplied function */
  347: struct tmpl_func {
  348: 	const char	*name;		/* function name, null to end list */
  349: 	u_int		min_args;	/* min # args (not counting name) */
  350: 	u_int		max_args;	/* max # args (not counting name) */
  351: 	tmpl_handler_t	*handler;	/* handler for function */
  352: };
  353: 
  354: extern char	*tmpl_list_handler(struct tmpl_ctx *ctx,
  355: 			const struct tmpl_func *userfuncs, u_int uflen,
  356: 			char **errmsgp, int argc, char **argv);
  357: 
  358: __END_DECLS
  359: 
  360: #endif	/* _PDEL_TMPL_TMPL_H_ */

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