Annotation of parser3/src/classes/math.C, revision 1.26
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.26 ! paf 8: static const char* IDENT_MATH_C="$Date: 2003/04/04 09:42:04 $";
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);
155: pa_MD5Encode((const unsigned char *)password,
1.26 ! paf 156: (const unsigned char *)normal_salt, 1/*TRUE: mix in magic string*/,
! 157: sample_buf, sample_size,
! 158: 0, 0);
1.20 paf 159: r.write_pass_lang(*new(pool) String(pool, sample_buf));
160: } else {
161: #ifdef HAVE_CRYPT
162: const char *sample_buf=crypt(password, normal_salt);
163: if(!sample_buf // nothing generated
164: || !sample_buf[0] // generated nothing
165: || strncmp(sample_buf, normal_salt, prefix_size)!=0) // salt prefix not preserved
166: throw Exception("parser.runtime",
167: &method_name,
1.21 paf 168: "on this platform does not support '%.*s' salt prefix", prefix_size, normal_salt);
1.20 paf 169:
1.19 paf 170: r.write_pass_lang(*new(pool) String(pool, sample_buf));
1.20 paf 171: #else
1.19 paf 172: throw Exception("parser.runtime",
173: &method_name,
174: "salt must start with '" PA_MD5PW_ID "'");
1.20 paf 175: #endif
176: }
1.19 paf 177: }
178:
1.26 ! paf 179: static void _md5(Request& r, const String& method_name, MethodParams *params) {
! 180: Pool& pool=r.pool();
! 181: const char *string=params->as_string(0, "parameter must be string").cstr();
! 182: const size_t sample_size=120;
! 183: char sample_buf[sample_size];
! 184: const int sample_bytes_count=12;
! 185: unsigned char sample_bytes[sample_bytes_count];
! 186: pa_MD5Encode((const unsigned char *)string,
! 187: (const unsigned char *)"", 0/*FALSE: mix in magic string*/,
! 188: sample_buf, sample_size,
! 189: sample_bytes, sample_bytes_count);
! 190: char *sample_bytes_hex=(char *)pool.malloc(sample_bytes_count*2/*byte->hh*/+1/*for zero-teminator*/);
! 191: unsigned char *src=sample_bytes;
! 192: unsigned char *end=sample_bytes+sample_bytes_count;
! 193: char *dest=sample_bytes_hex;
! 194: while(src<end)
! 195: dest+=snprintf(dest, 3, "%02x", *src++);
! 196:
! 197: r.write_pass_lang(*new(pool) String(pool, sample_bytes_hex));
! 198: }
! 199:
1.1 parser 200: // constructor
201:
1.18 paf 202: MMath::MMath(Pool& apool) : Methoded(apool, "math") {
1.1 parser 203: // ^FUNC(expr)
204: #define ADD1(name) \
205: add_native_method(#name, Method::CT_STATIC, _##name, 1, 1)
206:
207: ADD1(round); ADD1(floor); ADD1(ceiling);
1.3 parser 208: ADD1(trunc); ADD1(frac);
1.1 parser 209: ADD1(abs); ADD1(sign);
210: ADD1(exp); ADD1(log);
211: ADD1(sin); ADD1(asin);
212: ADD1(cos); ADD1(acos);
213: ADD1(tan); ADD1(atan);
1.3 parser 214: ADD1(degrees); ADD1(radians);
1.1 parser 215: ADD1(sqrt);
1.3 parser 216: ADD1(random);
1.1 parser 217:
218: #define ADD2(name) \
219: add_native_method(#name, Method::CT_STATIC, _##name, 2, 2)
220:
221: // ^pow(x;y)
222: ADD2(pow);
223:
1.19 paf 224: // ^crypt[password;salt]
225: ADD2(crypt);
1.26 ! paf 226:
! 227: // ^md5[string]
! 228: ADD1(md5);
1.5 parser 229: }
230:
231: // in MSVC each thread has it's own pseudo-random sequence
232: // in win32 apache each thread can handle multiple requests
233: // so to get proper randoms we remember random generated in one thread
234: void MMath::configure_admin(Request&) {
235: // setting seed
236: srand(
1.14 paf 237: randomizer
1.5 parser 238: #ifdef WIN32
1.14 paf 239: ^ GetCurrentThreadId()
1.5 parser 240: #else
1.14 paf 241: ^ getpid()
1.5 parser 242: #endif
1.20 paf 243: ^(unsigned int)time(NULL)
1.5 parser 244: );
245: if(!randomizer)
246: randomizer=rand();
1.1 parser 247: }
248:
1.2 parser 249: // global variables
250:
251: Methoded *math_base_class;
252: Hash *math_consts;
253:
1.1 parser 254: // creator
255:
256: Methoded *MMath_create(Pool& pool) {
1.2 parser 257: math_consts=new(pool) Hash(pool);
258: math_consts->put(
259: *new(pool) String(pool, "PI"),
1.3 parser 260: new(pool) VDouble(pool, PI));
1.2 parser 261:
262: return math_base_class=new(pool) MMath(pool);
1.1 parser 263: }
E-mail: