Annotation of parser3/src/classes/math.C, revision 1.27

1.1       parser      1: /** @file
                      2:        Parser: @b math parser class.
                      3: 
1.24      paf         4:        Copyright(c) 2001, 2003 ArtLebedev Group(http://www.artlebedev.com)
1.20      paf         5:        Author: Alexandr Petrosian <paf@design.ru>(http://paf.design.ru)
1.22      paf         6: */
1.1       parser      7: 
1.27    ! paf         8: static const char* IDENT_MATH_C="$Date: 2003/04/14 14:59:29 $";
1.1       parser      9: 
                     10: #include "pa_common.h"
                     11: #include "pa_vint.h"
1.2       parser     12: #include "pa_vmath.h"
1.1       parser     13: #include "pa_request.h"
1.19      paf        14: #include "pa_md5.h"
1.1       parser     15: 
1.5       parser     16: #ifdef WIN32
1.19      paf        17: // for threadID
1.5       parser     18: #      include <windows.h>
                     19: #endif
                     20: 
1.20      paf        21: #ifdef HAVE_CRYPT_H
                     22: #include <crypt.h>
                     23: #endif
                     24: 
1.1       parser     25: // defines
                     26: 
1.3       parser     27: #define PI 3.1415926535
1.20      paf        28: #define MAX_SALT 8
1.1       parser     29: 
                     30: // class
                     31: 
                     32: class MMath : public Methoded {
                     33: public:
                     34:        MMath(Pool& pool);
1.5       parser     35:        void configure_admin(Request& r);
                     36: 
1.1       parser     37: public: // Methoded
1.15      paf        38:        bool used_directly() { return false; }
1.1       parser     39: };
                     40: 
                     41: // methods
1.5       parser     42: static unsigned int randomizer=0;
1.20      paf        43: static inline int _random(uint top) {
                     44:        return (int)(((double)((randomizer=rand())% RAND_MAX)) / RAND_MAX * top );
                     45: }
1.1       parser     46: static void _random(Request& r, const String& method_name, MethodParams *params) {
                     47:        Pool& pool=r.pool();
                     48: 
1.3       parser     49:        Value& range=params->as_junction(0, "range must be expression");
1.25      paf        50:        double top=r.process_to_value(range).as_double();
                     51:        if(top<=0)
1.16      paf        52:                throw Exception("parser.runtime",
1.1       parser     53:                        &method_name,
1.25      paf        54:                        "top must be above 0(%g)", top);
1.1       parser     55:        
1.20      paf        56:        r.write_no_lang(*new(pool) VInt(pool, _random(uint(top))));
1.1       parser     57: }
                     58: 
                     59: 
1.20      paf        60: typedef double(*math1_func_ptr)(double);
1.3       parser     61: static double frac(double param) { return param-trunc(param); }
                     62: static double degrees(double param) { return param /PI *180; }
                     63: static double radians(double param) { return param /180 *PI; }
1.1       parser     64: 
                     65: static void math1(Request& r, 
                     66:                                  const String& method_name, MethodParams *params,
                     67:                                  math1_func_ptr func) {
                     68:        Pool& pool=r.pool();
1.3       parser     69:        Value& param=params->as_junction(0, "parameter must be expression");
1.1       parser     70: 
1.18      paf        71:        double result=(*func)(r.process_to_value(param).as_double());
                     72:        r.write_no_lang(*new(pool) VDouble(pool, result));
1.1       parser     73: }
                     74: 
                     75: #define MATH1(name) \
                     76:        static void _##name(Request& r, const String& method_name, MethodParams *params) {\
                     77:                math1(r, method_name, params, &name);\
                     78:        }
                     79: #define MATH1P(name_parser, name_c) \
                     80:        static void _##name_parser(Request& r, const String& method_name, MethodParams *params) {\
                     81:                math1(r, method_name, params, &name_c);\
                     82:        }
                     83: MATH1(round);  MATH1(floor);   MATH1P(ceiling, ceil);
1.3       parser     84: MATH1(trunc);  MATH1(frac);
1.1       parser     85: MATH1P(abs, fabs);     MATH1(sign);
                     86: MATH1(exp);    MATH1(log);     
                     87: MATH1(sin);    MATH1(asin);    
                     88: MATH1(cos);    MATH1(acos);    
                     89: MATH1(tan);    MATH1(atan);
1.3       parser     90: MATH1(degrees);        MATH1(radians);
1.1       parser     91: MATH1(sqrt);
                     92: 
                     93: 
                     94: typedef double (*math2_func_ptr)(double, double);
                     95: static void math2(Request& r, 
                     96:                                  const String& method_name, MethodParams *params,
                     97:                                  math2_func_ptr func) {
                     98:        Pool& pool=r.pool();
1.3       parser     99:        Value& a=params->as_junction(0, "parameter must be expression");
                    100:        Value& b=params->as_junction(1, "parameter must be expression");
1.1       parser    101: 
1.18      paf       102:        double result=(*func)(
1.17      paf       103:                r.process_to_value(a).as_double(),
1.18      paf       104:                r.process_to_value(b).as_double());
                    105:        r.write_no_lang(*new(pool) VDouble(pool, result));
1.1       parser    106: }
                    107: 
                    108: #define MATH2(name) \
                    109:        static void _##name(Request& r, const String& method_name, MethodParams *params) {\
                    110:                math2(r, method_name, params, &name);\
                    111:        }
                    112: MATH2(pow);
                    113: 
1.20      paf       114: inline bool is_salt_body_char(int c) {
                    115:        return isalnum(c) || c == '.' || c=='/';
                    116: }
                    117: static size_t calc_prefix_size(const char *salt) {
                    118:        if(size_t salt_size=strlen(salt)) {
                    119:                if(!is_salt_body_char(salt[0])) { // $...  {...
                    120:                        const char *cur=salt+1; // skip
                    121:                        while(is_salt_body_char(*cur++)) // ...$  ...}
                    122:                                ;
                    123:                        return cur-salt;
                    124:                } else
                    125:                        return 0;
                    126:        } else
                    127:                return 0;
                    128: }
1.19      paf       129: static void _crypt(Request& r, const String& method_name, MethodParams *params) {
                    130:        Pool& pool=r.pool();
                    131:        const char *password=params->as_string(0, "password must be string").cstr();
1.20      paf       132:        const char *maybe_bodyless_salt=params->as_string(1, "salt must be string").cstr();
                    133: 
                    134:        size_t prefix_size=calc_prefix_size(maybe_bodyless_salt);
                    135:        const char *normal_salt;
                    136:        char normalize_buf[MAX_STRING];
                    137:        if(prefix_size==strlen(maybe_bodyless_salt)) { // bodyless?
                    138:                strncpy(normalize_buf, maybe_bodyless_salt, MAX_STRING-MAX_SALT-1);
                    139:                char *cur=normalize_buf+strlen(normalize_buf);
                    140:                // sould add up MAX_SALT random chars
                    141:                static unsigned char itoa64[] =         /* 0 ... 63 => ASCII - 64 */
                    142:                "./0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz";
                    143:                for(int i=0; i<MAX_SALT; i++)
                    144:                        *cur++=itoa64[_random(64)];
                    145:                *cur=0;
                    146:                normal_salt=normalize_buf;
                    147:        } else
                    148:                normal_salt=maybe_bodyless_salt;
1.19      paf       149: 
                    150:     /* FreeBSD style MD5 string 
                    151:      */
1.20      paf       152:     if(strncmp(normal_salt, PA_MD5PW_ID, PA_MD5PW_IDLEN) == 0) {
1.19      paf       153:                const size_t sample_size=120;
                    154:                char *sample_buf=(char *)pool.malloc(sample_size);
1.27    ! paf       155:                PA_MD5Encode((const unsigned char *)password,
1.26      paf       156:                        (const unsigned char *)normal_salt, 1/*TRUE: mix in magic string*/,
1.27    ! paf       157:                        sample_buf, sample_size);
1.20      paf       158:                r.write_pass_lang(*new(pool) String(pool, sample_buf));
                    159:     } else {
                    160: #ifdef HAVE_CRYPT
                    161:                const char *sample_buf=crypt(password, normal_salt);
                    162:                if(!sample_buf  // nothing generated
                    163:                        || !sample_buf[0] // generated nothing
                    164:                        || strncmp(sample_buf, normal_salt, prefix_size)!=0) // salt prefix not preserved
                    165:                        throw Exception("parser.runtime",
                    166:                                &method_name,
1.21      paf       167:                                "on this platform does not support '%.*s' salt prefix", prefix_size, normal_salt);
1.20      paf       168:                
1.19      paf       169:                r.write_pass_lang(*new(pool) String(pool, sample_buf));
1.20      paf       170: #else
1.19      paf       171:                throw Exception("parser.runtime",
                    172:                        &method_name,
                    173:                        "salt must start with '" PA_MD5PW_ID "'");
1.20      paf       174: #endif
                    175:        }
1.19      paf       176: }
                    177: 
1.26      paf       178: static void _md5(Request& r, const String& method_name, MethodParams *params) {
                    179:        Pool& pool=r.pool();
                    180:        const char *string=params->as_string(0, "parameter must be string").cstr();
1.27    ! paf       181: 
        !           182: 
        !           183:        PA_MD5_CTX context;
        !           184:        unsigned char digest[16];
        !           185:        PA_MD5Init(&context);
        !           186:        PA_MD5Update(&context, (const unsigned char*)string, strlen(string));
        !           187:        PA_MD5Final(digest, &context);
        !           188: 
        !           189:        char *digest_bytes_hex=(char *)pool.malloc(sizeof(digest)*2/*byte->hh*/+1/*for zero-teminator*/);
        !           190:        unsigned char *src=digest;
        !           191:        unsigned char *end=digest+sizeof(digest);
        !           192:        char *dest=digest_bytes_hex;
1.26      paf       193:        while(src<end)
                    194:                dest+=snprintf(dest, 3, "%02x", *src++);
                    195: 
1.27    ! paf       196:        r.write_pass_lang(*new(pool) String(pool, digest_bytes_hex));
1.26      paf       197: }
                    198: 
1.1       parser    199: // constructor
                    200: 
1.18      paf       201: MMath::MMath(Pool& apool) : Methoded(apool, "math") {
1.1       parser    202:        // ^FUNC(expr)  
                    203: #define ADD1(name) \
                    204:        add_native_method(#name, Method::CT_STATIC, _##name, 1, 1)
                    205: 
                    206:        ADD1(round);    ADD1(floor);    ADD1(ceiling);
1.3       parser    207:        ADD1(trunc);    ADD1(frac);
1.1       parser    208:        ADD1(abs);      ADD1(sign);
                    209:        ADD1(exp);      ADD1(log);      
                    210:        ADD1(sin);      ADD1(asin);     
                    211:        ADD1(cos);      ADD1(acos);     
                    212:        ADD1(tan);      ADD1(atan);
1.3       parser    213:        ADD1(degrees);  ADD1(radians);
1.1       parser    214:        ADD1(sqrt);
1.3       parser    215:        ADD1(random);
1.1       parser    216: 
                    217: #define ADD2(name) \
                    218:        add_native_method(#name, Method::CT_STATIC, _##name, 2, 2)
                    219: 
                    220:        // ^pow(x;y)
                    221:        ADD2(pow);
                    222: 
1.19      paf       223:        // ^crypt[password;salt]
                    224:        ADD2(crypt);
1.26      paf       225: 
                    226:        // ^md5[string]
                    227:        ADD1(md5);
1.5       parser    228: }
                    229: 
                    230: // in MSVC each thread has it's own pseudo-random sequence
                    231: // in win32 apache each thread can handle multiple requests
                    232: // so to get proper randoms we remember random generated in one thread
                    233: void MMath::configure_admin(Request&) {
                    234:        // setting seed
                    235:        srand(
1.14      paf       236:                randomizer
1.5       parser    237: #ifdef WIN32
1.14      paf       238:                ^ GetCurrentThreadId()
1.5       parser    239: #else
1.14      paf       240:                ^ getpid()
1.5       parser    241: #endif
1.20      paf       242:                ^(unsigned int)time(NULL)
1.5       parser    243:        );
                    244:        if(!randomizer)
                    245:                randomizer=rand();
1.1       parser    246: }
                    247: 
1.2       parser    248: // global variables
                    249: 
                    250: Methoded *math_base_class;
                    251: Hash *math_consts;
                    252: 
1.1       parser    253: // creator
                    254: 
                    255: Methoded *MMath_create(Pool& pool) {
1.2       parser    256:        math_consts=new(pool) Hash(pool);
                    257:        math_consts->put(
                    258:                *new(pool) String(pool, "PI"), 
1.3       parser    259:                new(pool) VDouble(pool, PI));
1.2       parser    260: 
                    261:        return math_base_class=new(pool) MMath(pool);
1.1       parser    262: }

E-mail: