0000: 2f 2a 0a 2a 2a 20 43 6f 70 79 72 69 67 68 74 20 /*.** Copyright
0010: 28 63 29 20 32 30 30 36 20 44 2e 20 52 69 63 68 (c) 2006 D. Rich
0020: 61 72 64 20 48 69 70 70 0a 2a 2a 0a 2a 2a 20 54 ard Hipp.**.** T
0030: 68 69 73 20 70 72 6f 67 72 61 6d 20 69 73 20 66 his program is f
0040: 72 65 65 20 73 6f 66 74 77 61 72 65 3b 20 79 6f ree software; yo
0050: 75 20 63 61 6e 20 72 65 64 69 73 74 72 69 62 75 u can redistribu
0060: 74 65 20 69 74 20 61 6e 64 2f 6f 72 0a 2a 2a 20 te it and/or.**
0070: 6d 6f 64 69 66 79 20 69 74 20 75 6e 64 65 72 20 modify it under
0080: 74 68 65 20 74 65 72 6d 73 20 6f 66 20 74 68 65 the terms of the
0090: 20 47 4e 55 20 47 65 6e 65 72 61 6c 20 50 75 62 GNU General Pub
00a0: 6c 69 63 0a 2a 2a 20 4c 69 63 65 6e 73 65 20 76 lic.** License v
00b0: 65 72 73 69 6f 6e 20 32 20 61 73 20 70 75 62 6c ersion 2 as publ
00c0: 69 73 68 65 64 20 62 79 20 74 68 65 20 46 72 65 ished by the Fre
00d0: 65 20 53 6f 66 74 77 61 72 65 20 46 6f 75 6e 64 e Software Found
00e0: 61 74 69 6f 6e 2e 0a 2a 2a 0a 2a 2a 20 54 68 69 ation..**.** Thi
00f0: 73 20 70 72 6f 67 72 61 6d 20 69 73 20 64 69 73 s program is dis
0100: 74 72 69 62 75 74 65 64 20 69 6e 20 74 68 65 20 tributed in the
0110: 68 6f 70 65 20 74 68 61 74 20 69 74 20 77 69 6c hope that it wil
0120: 6c 20 62 65 20 75 73 65 66 75 6c 2c 0a 2a 2a 20 l be useful,.**
0130: 62 75 74 20 57 49 54 48 4f 55 54 20 41 4e 59 20 but WITHOUT ANY
0140: 57 41 52 52 41 4e 54 59 3b 20 77 69 74 68 6f 75 WARRANTY; withou
0150: 74 20 65 76 65 6e 20 74 68 65 20 69 6d 70 6c 69 t even the impli
0160: 65 64 20 77 61 72 72 61 6e 74 79 20 6f 66 0a 2a ed warranty of.*
0170: 2a 20 4d 45 52 43 48 41 4e 54 41 42 49 4c 49 54 * MERCHANTABILIT
0180: 59 20 6f 72 20 46 49 54 4e 45 53 53 20 46 4f 52 Y or FITNESS FOR
0190: 20 41 20 50 41 52 54 49 43 55 4c 41 52 20 50 55 A PARTICULAR PU
01a0: 52 50 4f 53 45 2e 20 20 53 65 65 20 74 68 65 20 RPOSE. See the
01b0: 47 4e 55 0a 2a 2a 20 47 65 6e 65 72 61 6c 20 50 GNU.** General P
01c0: 75 62 6c 69 63 20 4c 69 63 65 6e 73 65 20 66 6f ublic License fo
01d0: 72 20 6d 6f 72 65 20 64 65 74 61 69 6c 73 2e 0a r more details..
01e0: 2a 2a 20 0a 2a 2a 20 59 6f 75 20 73 68 6f 75 6c ** .** You shoul
01f0: 64 20 68 61 76 65 20 72 65 63 65 69 76 65 64 20 d have received
0200: 61 20 63 6f 70 79 20 6f 66 20 74 68 65 20 47 4e a copy of the GN
0210: 55 20 47 65 6e 65 72 61 6c 20 50 75 62 6c 69 63 U General Public
0220: 0a 2a 2a 20 4c 69 63 65 6e 73 65 20 61 6c 6f 6e .** License alon
0230: 67 20 77 69 74 68 20 74 68 69 73 20 6c 69 62 72 g with this libr
0240: 61 72 79 3b 20 69 66 20 6e 6f 74 2c 20 77 72 69 ary; if not, wri
0250: 74 65 20 74 6f 20 74 68 65 0a 2a 2a 20 46 72 65 te to the.** Fre
0260: 65 20 53 6f 66 74 77 61 72 65 20 46 6f 75 6e 64 e Software Found
0270: 61 74 69 6f 6e 2c 20 49 6e 63 2e 2c 20 35 39 20 ation, Inc., 59
0280: 54 65 6d 70 6c 65 20 50 6c 61 63 65 20 2d 20 53 Temple Place - S
0290: 75 69 74 65 20 33 33 30 2c 0a 2a 2a 20 42 6f 73 uite 330,.** Bos
02a0: 74 6f 6e 2c 20 4d 41 20 20 30 32 31 31 31 2d 31 ton, MA 02111-1
02b0: 33 30 37 2c 20 55 53 41 2e 0a 2a 2a 0a 2a 2a 20 307, USA..**.**
02c0: 41 75 74 68 6f 72 20 63 6f 6e 74 61 63 74 20 69 Author contact i
02d0: 6e 66 6f 72 6d 61 74 69 6f 6e 3a 0a 2a 2a 20 20 nformation:.**
02e0: 20 64 72 68 40 68 77 61 63 69 2e 63 6f 6d 0a 2a drh@hwaci.com.*
02f0: 2a 20 20 20 68 74 74 70 3a 2f 2f 77 77 77 2e 68 * http://www.h
0300: 77 61 63 69 2e 63 6f 6d 2f 64 72 68 2f 0a 2a 2a waci.com/drh/.**
0310: 0a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a .***************
0320: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a ****************
0330: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a ****************
0340: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a ****************
0350: 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a 2a ****************
0360: 0a 2a 2a 0a 2a 2a 20 54 68 69 73 20 66 69 6c 65 .**.** This file
0370: 20 63 6f 6e 74 61 69 6e 73 20 43 20 66 75 6e 63 contains C func
0380: 74 69 6f 6e 73 20 61 6e 64 20 70 72 6f 63 65 64 tions and proced
0390: 75 72 65 73 20 74 68 61 74 20 70 72 6f 76 69 64 ures that provid
03a0: 65 20 75 73 65 66 75 6c 0a 2a 2a 20 73 65 72 76 e useful.** serv
03b0: 69 63 65 73 20 74 6f 20 43 47 49 20 70 72 6f 67 ices to CGI prog
03c0: 72 61 6d 73 2e 20 20 54 68 65 72 65 20 61 72 65 rams. There are
03d0: 20 70 72 6f 63 65 64 75 72 65 73 20 66 6f 72 20 procedures for
03e0: 70 61 72 73 69 6e 67 20 61 6e 64 0a 2a 2a 20 64 parsing and.** d
03f0: 69 73 70 65 6e 73 69 6e 67 20 51 55 45 52 59 5f ispensing QUERY_
0400: 53 54 52 49 4e 47 20 70 61 72 61 6d 65 74 65 72 STRING parameter
0410: 73 20 61 6e 64 20 63 6f 6f 6b 69 65 73 2c 20 74 s and cookies, t
0420: 68 65 20 22 6d 70 72 69 6e 74 66 28 29 22 0a 2a he "mprintf()".*
0430: 2a 20 66 6f 72 6d 61 74 74 69 6e 67 20 66 75 6e * formatting fun
0440: 63 74 69 6f 6e 20 61 6e 64 20 69 74 73 20 63 6f ction and its co
0450: 75 73 69 6e 73 2c 20 61 6e 64 20 72 6f 75 74 69 usins, and routi
0460: 6e 65 73 20 74 6f 20 65 6e 63 6f 64 65 20 61 6e nes to encode an
0470: 64 0a 2a 2a 20 64 65 63 6f 64 65 20 73 74 72 69 d.** decode stri
0480: 6e 67 73 20 69 6e 20 48 54 4d 4c 20 6f 72 20 48 ngs in HTML or H
0490: 54 54 50 2e 0a 2a 2f 0a 23 69 6e 63 6c 75 64 65 TTP..*/.#include
04a0: 20 22 63 6f 6e 66 69 67 2e 68 22 0a 23 69 6e 63 "config.h".#inc
04b0: 6c 75 64 65 20 3c 73 79 73 2f 73 6f 63 6b 65 74 lude <sys/socket
04c0: 2e 68 3e 0a 23 69 6e 63 6c 75 64 65 20 3c 6e 65 .h>.#include <ne
04d0: 74 69 6e 65 74 2f 69 6e 2e 68 3e 0a 23 69 6e 63 tinet/in.h>.#inc
04e0: 6c 75 64 65 20 3c 61 72 70 61 2f 69 6e 65 74 2e lude <arpa/inet.
04f0: 68 3e 0a 23 69 6e 63 6c 75 64 65 20 3c 74 69 6d h>.#include <tim
0500: 65 2e 68 3e 0a 23 69 6e 63 6c 75 64 65 20 3c 73 e.h>.#include <s
0510: 79 73 2f 74 69 6d 65 73 2e 68 3e 0a 23 69 6e 63 ys/times.h>.#inc
0520: 6c 75 64 65 20 3c 73 79 73 2f 74 69 6d 65 2e 68 lude <sys/time.h
0530: 3e 0a 23 69 6e 63 6c 75 64 65 20 3c 73 79 73 2f >.#include <sys/
0540: 77 61 69 74 2e 68 3e 0a 23 69 6e 63 6c 75 64 65 wait.h>.#include
0550: 20 3c 73 74 64 69 6f 2e 68 3e 0a 23 69 6e 63 6c <stdio.h>.#incl
0560: 75 64 65 20 3c 73 74 64 6c 69 62 2e 68 3e 0a 23 ude <stdlib.h>.#
0570: 69 6e 63 6c 75 64 65 20 3c 73 79 73 2f 73 65 6c include <sys/sel
0580: 65 63 74 2e 68 3e 0a 23 69 6e 63 6c 75 64 65 20 ect.h>.#include
0590: 3c 75 6e 69 73 74 64 2e 68 3e 0a 23 69 6e 63 6c <unistd.h>.#incl
05a0: 75 64 65 20 22 63 67 69 2e 68 22 0a 0a 23 69 66 ude "cgi.h"..#if
05b0: 20 49 4e 54 45 52 46 41 43 45 0a 2f 2a 0a 2a 2a INTERFACE./*.**
05c0: 20 53 68 6f 72 74 63 75 74 73 20 66 6f 72 20 63 Shortcuts for c
05d0: 67 69 5f 70 61 72 61 6d 65 74 65 72 2e 20 20 50 gi_parameter. P
05e0: 28 22 78 22 29 20 72 65 74 75 72 6e 73 20 74 68 ("x") returns th
05f0: 65 20 76 61 6c 75 65 20 6f 66 20 71 75 65 72 79 e value of query
0600: 20 70 61 72 61 6d 65 74 65 72 0a 2a 2a 20 6f 72 parameter.** or
0610: 20 63 6f 6f 6b 69 65 20 22 78 22 2c 20 6f 72 20 cookie "x", or
0620: 4e 55 4c 4c 20 69 66 20 74 68 65 72 65 20 69 73 NULL if there is
0630: 20 6e 6f 20 73 75 63 68 20 70 61 72 61 6d 65 74 no such paramet
0640: 65 72 20 6f 72 20 63 6f 6f 6b 69 65 2e 20 20 50 er or cookie. P
0650: 44 28 22 78 22 2c 22 79 22 29 0a 2a 2a 20 64 6f D("x","y").** do
0660: 65 73 20 74 68 65 20 73 61 6d 65 20 65 78 63 65 es the same exce
0670: 70 74 20 22 79 22 20 69 73 20 72 65 74 75 72 6e pt "y" is return
0680: 65 64 20 69 6e 20 70 6c 61 63 65 20 6f 66 20 4e ed in place of N
0690: 55 4c 4c 20 69 66 20 74 68 65 72 65 20 69 73 20 ULL if there is
06a0: 6e 6f 74 20 6d 61 74 63 68 2e 0a 2a 2f 0a 23 64 not match..*/.#d
06b0: 65 66 69 6e 65 20 50 28 78 29 20 20 20 20 20 20 efine P(x)
06c0: 20 20 63 67 69 5f 70 61 72 61 6d 65 74 65 72 28 cgi_parameter(
06d0: 28 78 29 2c 30 29 0a 23 64 65 66 69 6e 65 20 50 (x),0).#define P
06e0: 44 28 78 2c 79 29 20 20 20 20 20 63 67 69 5f 70 D(x,y) cgi_p
06f0: 61 72 61 6d 65 74 65 72 28 28 78 29 2c 28 79 29 arameter((x),(y)
0700: 29 0a 23 64 65 66 69 6e 65 20 51 50 28 78 29 20 ).#define QP(x)
0710: 20 20 20 20 20 20 71 75 6f 74 61 62 6c 65 5f 73 quotable_s
0720: 74 72 69 6e 67 28 63 67 69 5f 70 61 72 61 6d 65 tring(cgi_parame
0730: 74 65 72 28 28 78 29 2c 30 29 29 0a 23 64 65 66 ter((x),0)).#def
0740: 69 6e 65 20 51 50 44 28 78 2c 79 29 20 20 20 20 ine QPD(x,y)
0750: 71 75 6f 74 61 62 6c 65 5f 73 74 72 69 6e 67 28 quotable_string(
0760: 63 67 69 5f 70 61 72 61 6d 65 74 65 72 28 28 78 cgi_parameter((x
0770: 29 2c 28 79 29 29 29 0a 0a 23 65 6e 64 69 66 20 ),(y)))..#endif
0780: 2f 2a 20 49 4e 54 45 52 46 41 43 45 20 2a 2f 0a /* INTERFACE */.
0790: 0a 2f 2a 0a 2a 2a 20 50 72 6f 76 69 64 65 20 61 ./*.** Provide a
07a0: 20 72 65 6c 69 61 62 6c 65 20 69 6d 70 6c 65 6d reliable implem
07b0: 65 6e 74 61 74 69 6f 6e 20 6f 66 20 61 20 63 61 entation of a ca
07c0: 73 65 6c 65 73 73 20 73 74 72 69 6e 67 20 63 6f seless string co
07d0: 6d 70 61 72 69 73 6f 6e 0a 2a 2a 20 66 75 6e 63 mparison.** func
07e0: 74 69 6f 6e 2e 0a 2a 2f 0a 23 64 65 66 69 6e 65 tion..*/.#define
07f0: 20 73 74 72 69 63 6d 70 20 73 71 6c 69 74 65 33 stricmp sqlite3
0800: 53 74 72 49 43 6d 70 0a 65 78 74 65 72 6e 20 69 StrICmp.extern i
0810: 6e 74 20 73 71 6c 69 74 65 33 53 74 72 49 43 6d nt sqlite3StrICm
0820: 70 28 63 6f 6e 73 74 20 63 68 61 72 2a 2c 20 63 p(const char*, c
0830: 6f 6e 73 74 20 63 68 61 72 2a 29 3b 0a 0a 2f 2a onst char*);../*
0840: 0a 2a 2a 20 54 68 65 20 62 6f 64 79 20 6f 66 20 .** The body of
0850: 74 68 65 20 48 54 54 50 20 72 65 70 6c 79 20 74 the HTTP reply t
0860: 65 78 74 20 69 73 20 73 74 6f 72 65 64 20 68 65 ext is stored he
0870: 72 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 42 6c re..*/.static Bl
0880: 6f 62 20 63 67 69 43 6f 6e 74 65 6e 74 20 3d 20 ob cgiContent =
0890: 42 4c 4f 42 5f 49 4e 49 54 49 41 4c 49 5a 45 52 BLOB_INITIALIZER
08a0: 3b 0a 0a 2f 2a 0a 2a 2a 20 41 70 70 65 6e 64 20 ;../*.** Append
08b0: 72 65 70 6c 79 20 63 6f 6e 74 65 6e 74 20 74 6f reply content to
08c0: 20 77 68 61 74 20 61 6c 72 65 61 64 79 20 65 78 what already ex
08d0: 69 73 74 73 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 ists..*/.void cg
08e0: 69 5f 61 70 70 65 6e 64 5f 63 6f 6e 74 65 6e 74 i_append_content
08f0: 28 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 44 61 (const char *zDa
0900: 74 61 2c 20 69 6e 74 20 6e 41 6d 74 29 7b 0a 20 ta, int nAmt){.
0910: 20 62 6c 6f 62 5f 61 70 70 65 6e 64 28 26 63 67 blob_append(&cg
0920: 69 43 6f 6e 74 65 6e 74 2c 20 7a 44 61 74 61 2c iContent, zData,
0930: 20 6e 41 6d 74 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a nAmt);.}../*.**
0940: 20 52 65 73 65 74 20 74 68 65 20 48 54 54 50 20 Reset the HTTP
0950: 72 65 70 6c 79 20 74 65 78 74 20 74 6f 20 62 65 reply text to be
0960: 20 61 6e 20 65 6d 70 74 79 20 73 74 72 69 6e 67 an empty string
0970: 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 72 65 ..*/.void cgi_re
0980: 73 65 74 5f 63 6f 6e 74 65 6e 74 28 76 6f 69 64 set_content(void
0990: 29 7b 0a 20 20 62 6c 6f 62 5f 72 65 73 65 74 28 ){. blob_reset(
09a0: 26 63 67 69 43 6f 6e 74 65 6e 74 29 3b 0a 7d 0a &cgiContent);.}.
09b0: 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 61 20 ./*.** Return a
09c0: 70 6f 69 6e 74 65 72 20 74 6f 20 74 68 65 20 48 pointer to the H
09d0: 54 54 50 20 72 65 70 6c 79 20 74 65 78 74 2e 0a TTP reply text..
09e0: 2a 2f 0a 63 68 61 72 20 2a 63 67 69 5f 65 78 74 */.char *cgi_ext
09f0: 72 61 63 74 5f 63 6f 6e 74 65 6e 74 28 69 6e 74 ract_content(int
0a00: 20 2a 70 6e 41 6d 74 29 7b 0a 20 20 72 65 74 75 *pnAmt){. retu
0a10: 72 6e 20 62 6c 6f 62 5f 62 75 66 66 65 72 28 26 rn blob_buffer(&
0a20: 63 67 69 43 6f 6e 74 65 6e 74 29 3b 0a 7d 0a 0a cgiContent);.}..
0a30: 2f 2a 0a 2a 2a 20 41 64 64 69 74 69 6f 6e 61 6c /*.** Additional
0a40: 20 69 6e 66 6f 72 6d 61 74 69 6f 6e 20 75 73 65 information use
0a50: 64 20 74 6f 20 66 6f 72 6d 20 74 68 65 20 48 54 d to form the HT
0a60: 54 50 20 72 65 70 6c 79 0a 2a 2f 0a 73 74 61 74 TP reply.*/.stat
0a70: 69 63 20 63 68 61 72 20 2a 7a 43 6f 6e 74 65 6e ic char *zConten
0a80: 74 54 79 70 65 20 3d 20 22 74 65 78 74 2f 68 74 tType = "text/ht
0a90: 6d 6c 22 3b 20 20 20 20 20 2f 2a 20 43 6f 6e 74 ml"; /* Cont
0aa0: 65 6e 74 20 74 79 70 65 20 6f 66 20 74 68 65 20 ent type of the
0ab0: 72 65 70 6c 79 20 2a 2f 0a 73 74 61 74 69 63 20 reply */.static
0ac0: 63 68 61 72 20 2a 7a 52 65 70 6c 79 53 74 61 74 char *zReplyStat
0ad0: 75 73 20 3d 20 22 4f 4b 22 3b 20 20 20 20 20 20 us = "OK";
0ae0: 20 20 20 20 20 20 2f 2a 20 52 65 70 6c 79 20 73 /* Reply s
0af0: 74 61 74 75 73 20 64 65 73 63 72 69 70 74 69 6f tatus descriptio
0b00: 6e 20 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 n */.static int
0b10: 69 52 65 70 6c 79 53 74 61 74 75 73 20 3d 20 32 iReplyStatus = 2
0b20: 30 30 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 00;
0b30: 20 20 2f 2a 20 52 65 70 6c 79 20 73 74 61 74 75 /* Reply statu
0b40: 73 20 63 6f 64 65 20 2a 2f 0a 73 74 61 74 69 63 s code */.static
0b50: 20 42 6c 6f 62 20 65 78 74 72 61 48 65 61 64 65 Blob extraHeade
0b60: 72 20 3d 20 42 4c 4f 42 5f 49 4e 49 54 49 41 4c r = BLOB_INITIAL
0b70: 49 5a 45 52 3b 20 20 2f 2a 20 45 78 74 72 61 20 IZER; /* Extra
0b80: 68 65 61 64 65 72 20 74 65 78 74 20 2a 2f 0a 73 header text */.s
0b90: 74 61 74 69 63 20 69 6e 74 20 66 75 6c 6c 48 74 tatic int fullHt
0ba0: 74 70 52 65 70 6c 79 20 3d 20 30 3b 20 20 20 20 tpReply = 0;
0bb0: 20 20 2f 2a 20 54 72 75 65 20 66 6f 72 20 61 20 /* True for a
0bc0: 66 75 6c 6c 2d 62 6c 6f 77 6e 20 48 54 54 50 20 full-blown HTTP
0bd0: 68 65 61 64 65 72 20 2a 2f 0a 0a 2f 2a 0a 2a 2a header */../*.**
0be0: 20 53 65 74 20 74 68 65 20 72 65 70 6c 79 20 63 Set the reply c
0bf0: 6f 6e 74 65 6e 74 20 74 79 70 65 0a 2a 2f 0a 76 ontent type.*/.v
0c00: 6f 69 64 20 63 67 69 5f 73 65 74 5f 63 6f 6e 74 oid cgi_set_cont
0c10: 65 6e 74 5f 74 79 70 65 28 63 6f 6e 73 74 20 63 ent_type(const c
0c20: 68 61 72 20 2a 7a 54 79 70 65 29 7b 0a 20 20 7a har *zType){. z
0c30: 43 6f 6e 74 65 6e 74 54 79 70 65 20 3d 20 6d 70 ContentType = mp
0c40: 72 69 6e 74 66 28 22 25 73 22 2c 20 7a 54 79 70 rintf("%s", zTyp
0c50: 65 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 65 74 e);.}../*.** Set
0c60: 20 74 68 65 20 72 65 70 6c 79 20 73 74 61 74 75 the reply statu
0c70: 73 20 63 6f 64 65 0a 2a 2f 0a 76 6f 69 64 20 63 s code.*/.void c
0c80: 67 69 5f 73 65 74 5f 73 74 61 74 75 73 28 69 6e gi_set_status(in
0c90: 74 20 69 53 74 61 74 2c 20 63 6f 6e 73 74 20 63 t iStat, const c
0ca0: 68 61 72 20 2a 7a 53 74 61 74 29 7b 0a 20 20 7a har *zStat){. z
0cb0: 52 65 70 6c 79 53 74 61 74 75 73 20 3d 20 6d 70 ReplyStatus = mp
0cc0: 72 69 6e 74 66 28 22 25 73 22 2c 20 7a 53 74 61 rintf("%s", zSta
0cd0: 74 29 3b 0a 20 20 69 52 65 70 6c 79 53 74 61 74 t);. iReplyStat
0ce0: 75 73 20 3d 20 69 53 74 61 74 3b 0a 7d 0a 0a 2f us = iStat;.}../
0cf0: 2a 0a 2a 2a 20 41 70 70 65 6e 64 20 74 65 78 74 *.** Append text
0d00: 20 74 6f 20 74 68 65 20 68 65 61 64 65 72 20 6f to the header o
0d10: 66 20 61 6e 20 48 54 54 50 20 72 65 70 6c 79 0a f an HTTP reply.
0d20: 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 61 70 70 65 */.void cgi_appe
0d30: 6e 64 5f 68 65 61 64 65 72 28 63 6f 6e 73 74 20 nd_header(const
0d40: 63 68 61 72 20 2a 7a 4c 69 6e 65 29 7b 0a 20 20 char *zLine){.
0d50: 62 6c 6f 62 5f 61 70 70 65 6e 64 28 26 65 78 74 blob_append(&ext
0d60: 72 61 48 65 61 64 65 72 2c 20 7a 4c 69 6e 65 2c raHeader, zLine,
0d70: 20 2d 31 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 -1);.}../*.** S
0d80: 65 74 20 61 20 63 6f 6f 6b 69 65 2e 0a 2a 2a 0a et a cookie..**.
0d90: 2a 2a 20 5a 65 72 6f 20 6c 69 66 65 74 69 6d 65 ** Zero lifetime
0da0: 20 69 6d 70 6c 69 65 73 20 61 20 73 65 73 73 69 implies a sessi
0db0: 6f 6e 20 63 6f 6f 6b 69 65 2e 0a 2a 2f 0a 76 6f on cookie..*/.vo
0dc0: 69 64 20 63 67 69 5f 73 65 74 5f 63 6f 6f 6b 69 id cgi_set_cooki
0dd0: 65 28 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 e(. const char
0de0: 2a 7a 4e 61 6d 65 2c 20 20 20 20 2f 2a 20 4e 61 *zName, /* Na
0df0: 6d 65 20 6f 66 20 74 68 65 20 63 6f 6f 6b 69 65 me of the cookie
0e00: 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 */. const char
0e10: 20 2a 7a 56 61 6c 75 65 2c 20 20 20 2f 2a 20 56 *zValue, /* V
0e20: 61 6c 75 65 20 6f 66 20 74 68 65 20 63 6f 6f 6b alue of the cook
0e30: 69 65 2e 20 20 41 75 74 6f 6d 61 74 69 63 61 6c ie. Automatical
0e40: 6c 79 20 65 73 63 61 70 65 64 20 2a 2f 0a 20 20 ly escaped */.
0e50: 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 50 61 74 const char *zPat
0e60: 68 2c 20 20 20 20 2f 2a 20 50 61 74 68 20 63 6f h, /* Path co
0e70: 6f 6b 69 65 20 61 70 70 6c 69 65 73 20 74 6f 2e okie applies to.
0e80: 20 20 4e 55 4c 4c 20 6d 65 61 6e 73 20 22 2f 22 NULL means "/"
0e90: 20 2a 2f 0a 20 20 69 6e 74 20 6c 69 66 65 74 69 */. int lifeti
0ea0: 6d 65 20 20 20 20 20 20 20 20 20 20 2f 2a 20 45 me /* E
0eb0: 78 70 69 72 61 74 69 6f 6e 20 6f 66 20 74 68 65 xpiration of the
0ec0: 20 63 6f 6f 6b 69 65 20 69 6e 20 73 65 63 6f 6e cookie in secon
0ed0: 64 73 20 66 72 6f 6d 20 6e 6f 77 20 2a 2f 0a 29 ds from now */.)
0ee0: 7b 0a 20 20 69 66 28 20 7a 50 61 74 68 3d 3d 30 {. if( zPath==0
0ef0: 20 29 20 7a 50 61 74 68 20 3d 20 22 2f 22 3b 0a ) zPath = "/";.
0f00: 20 20 69 66 28 20 6c 69 66 65 74 69 6d 65 3e 30 if( lifetime>0
0f10: 20 29 7b 0a 20 20 20 20 6c 69 66 65 74 69 6d 65 ){. lifetime
0f20: 20 2b 3d 20 28 69 6e 74 29 74 69 6d 65 28 30 29 += (int)time(0)
0f30: 3b 0a 20 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e ;. blob_appen
0f40: 64 66 28 26 65 78 74 72 61 48 65 61 64 65 72 2c df(&extraHeader,
0f50: 0a 20 20 20 20 20 20 20 22 53 65 74 2d 43 6f 6f . "Set-Coo
0f60: 6b 69 65 3a 20 25 73 3d 25 74 3b 20 50 61 74 68 kie: %s=%t; Path
0f70: 3d 25 73 3b 20 65 78 70 69 72 65 73 3d 25 73 3b =%s; expires=%s;
0f80: 20 56 65 72 73 69 6f 6e 3d 31 5c 72 5c 6e 22 2c Version=1\r\n",
0f90: 0a 20 20 20 20 20 20 20 20 7a 4e 61 6d 65 2c 20 . zName,
0fa0: 7a 56 61 6c 75 65 2c 20 7a 50 61 74 68 2c 20 63 zValue, zPath, c
0fb0: 67 69 5f 72 66 63 38 32 32 5f 64 61 74 65 73 74 gi_rfc822_datest
0fc0: 61 6d 70 28 6c 69 66 65 74 69 6d 65 29 29 3b 0a amp(lifetime));.
0fd0: 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 62 6c 6f }else{. blo
0fe0: 62 5f 61 70 70 65 6e 64 66 28 26 65 78 74 72 61 b_appendf(&extra
0ff0: 48 65 61 64 65 72 2c 0a 20 20 20 20 20 20 20 22 Header,. "
1000: 53 65 74 2d 43 6f 6f 6b 69 65 3a 20 25 73 3d 25 Set-Cookie: %s=%
1010: 74 3b 20 50 61 74 68 3d 25 73 3b 20 56 65 72 73 t; Path=%s; Vers
1020: 69 6f 6e 3d 31 5c 72 5c 6e 22 2c 0a 20 20 20 20 ion=1\r\n",.
1030: 20 20 20 7a 4e 61 6d 65 2c 20 7a 56 61 6c 75 65 zName, zValue
1040: 2c 20 7a 50 61 74 68 29 3b 0a 20 20 7d 0a 7d 0a , zPath);. }.}.
1050: 0a 23 69 66 20 30 0a 2f 2a 0a 2a 2a 20 41 64 64 .#if 0./*.** Add
1060: 20 61 6e 20 45 54 61 67 20 68 65 61 64 65 72 20 an ETag header
1070: 6c 69 6e 65 0a 2a 2f 0a 73 74 61 74 69 63 20 63 line.*/.static c
1080: 68 61 72 20 2a 63 67 69 5f 61 64 64 5f 65 74 61 har *cgi_add_eta
1090: 67 28 63 68 61 72 20 2a 7a 54 78 74 2c 20 69 6e g(char *zTxt, in
10a0: 74 20 6e 4c 65 6e 29 7b 0a 20 20 4d 44 35 43 6f t nLen){. MD5Co
10b0: 6e 74 65 78 74 20 63 74 78 3b 0a 20 20 75 6e 73 ntext ctx;. uns
10c0: 69 67 6e 65 64 20 63 68 61 72 20 64 69 67 65 73 igned char diges
10d0: 74 5b 31 36 5d 3b 0a 20 20 69 6e 74 20 69 2c 20 t[16];. int i,
10e0: 6a 3b 0a 20 20 63 68 61 72 20 7a 45 54 61 67 5b j;. char zETag[
10f0: 36 34 5d 3b 0a 0a 20 20 4d 44 35 49 6e 69 74 28 64];.. MD5Init(
1100: 26 63 74 78 29 3b 0a 20 20 4d 44 35 55 70 64 61 &ctx);. MD5Upda
1110: 74 65 28 26 63 74 78 2c 7a 54 78 74 2c 6e 4c 65 te(&ctx,zTxt,nLe
1120: 6e 29 3b 0a 20 20 4d 44 35 46 69 6e 61 6c 28 64 n);. MD5Final(d
1130: 69 67 65 73 74 2c 26 63 74 78 29 3b 0a 20 20 66 igest,&ctx);. f
1140: 6f 72 28 6a 3d 69 3d 30 3b 20 69 3c 31 36 3b 20 or(j=i=0; i<16;
1150: 69 2b 2b 2c 6a 2b 3d 32 29 7b 0a 20 20 20 20 62 i++,j+=2){. b
1160: 70 72 69 6e 74 66 28 26 7a 45 54 61 67 5b 6a 5d printf(&zETag[j]
1170: 2c 73 69 7a 65 6f 66 28 7a 45 54 61 67 29 2d 6a ,sizeof(zETag)-j
1180: 2c 22 25 30 32 78 22 2c 28 69 6e 74 29 64 69 67 ,"%02x",(int)dig
1190: 65 73 74 5b 69 5d 29 3b 0a 20 20 7d 0a 20 20 62 est[i]);. }. b
11a0: 6c 6f 62 5f 61 70 70 65 6e 64 66 28 26 65 78 74 lob_appendf(&ext
11b0: 72 61 48 65 61 64 65 72 2c 20 22 45 54 61 67 3a raHeader, "ETag:
11c0: 20 25 73 5c 72 5c 6e 22 2c 20 7a 45 54 61 67 29 %s\r\n", zETag)
11d0: 3b 0a 20 20 72 65 74 75 72 6e 20 73 74 72 64 75 ;. return strdu
11e0: 70 28 7a 45 54 61 67 29 3b 0a 7d 0a 0a 2f 2a 0a p(zETag);.}../*.
11f0: 2a 2a 20 44 6f 20 73 6f 6d 65 20 63 61 63 68 65 ** Do some cache
1200: 20 63 6f 6e 74 72 6f 6c 20 73 74 75 66 66 2e 20 control stuff.
1210: 46 69 72 73 74 2c 20 77 65 20 67 65 6e 65 72 61 First, we genera
1220: 74 65 20 61 6e 20 45 54 61 67 20 61 6e 64 20 69 te an ETag and i
1230: 6e 63 6c 75 64 65 20 69 74 20 69 6e 0a 2a 2a 20 nclude it in.**
1240: 74 68 65 20 72 65 73 70 6f 6e 73 65 20 68 65 61 the response hea
1250: 64 65 72 73 2e 20 53 65 63 6f 6e 64 2c 20 77 65 ders. Second, we
1260: 20 64 6f 20 77 68 61 74 65 76 65 72 20 69 73 20 do whatever is
1270: 6e 65 63 65 73 73 61 72 79 20 74 6f 20 64 65 74 necessary to det
1280: 65 72 6d 69 6e 65 20 69 66 0a 2a 2a 20 74 68 65 ermine if.** the
1290: 20 72 65 71 75 65 73 74 20 77 61 73 20 61 73 6b request was ask
12a0: 69 6e 67 20 61 62 6f 75 74 20 63 61 63 68 69 6e ing about cachin
12b0: 67 20 61 6e 64 20 77 68 65 74 68 65 72 20 77 65 g and whether we
12c0: 20 6e 65 65 64 20 74 6f 20 73 65 6e 64 20 62 61 need to send ba
12d0: 63 6b 20 74 68 65 0a 2a 2a 20 72 65 73 70 6f 6e ck the.** respon
12e0: 73 65 20 62 6f 64 79 2e 20 49 66 20 77 65 20 73 se body. If we s
12f0: 68 6f 75 6c 64 6e 27 74 20 73 65 6e 64 20 61 20 houldn't send a
1300: 62 6f 64 79 2c 20 72 65 74 75 72 6e 20 6e 6f 6e body, return non
1310: 2d 7a 65 72 6f 2e 0a 2a 2a 0a 2a 2a 20 43 75 72 -zero..**.** Cur
1320: 72 65 6e 74 6c 79 2c 20 77 65 20 6a 75 73 74 20 rently, we just
1330: 63 68 65 63 6b 20 74 68 65 20 45 54 61 67 20 61 check the ETag a
1340: 67 61 69 6e 73 74 20 61 6e 79 20 49 66 2d 4e 6f gainst any If-No
1350: 6e 65 2d 4d 61 74 63 68 20 68 65 61 64 65 72 2e ne-Match header.
1360: 0a 2a 2a 0a 2a 2a 20 46 49 58 4d 45 3a 20 49 6e .**.** FIXME: In
1370: 20 73 6f 6d 65 20 63 61 73 65 73 20 28 61 74 74 some cases (att
1380: 61 63 68 6d 65 6e 74 73 2c 20 66 69 6c 65 20 63 achments, file c
1390: 6f 6e 74 65 6e 74 73 29 20 77 65 20 63 6f 75 6c ontents) we coul
13a0: 64 20 63 68 65 63 6b 0a 2a 2a 20 49 66 2d 4d 6f d check.** If-Mo
13b0: 64 69 66 69 65 64 2d 53 69 6e 63 65 20 68 65 61 dified-Since hea
13c0: 64 65 72 73 20 61 6e 64 20 61 6c 77 61 79 73 20 ders and always
13d0: 69 6e 63 6c 75 64 65 20 4c 61 73 74 2d 4d 6f 64 include Last-Mod
13e0: 69 66 69 65 64 20 69 6e 20 72 65 73 70 6f 6e 73 ified in respons
13f0: 65 73 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e es..*/.static in
1400: 74 20 63 68 65 63 6b 5f 63 61 63 68 65 5f 63 6f t check_cache_co
1410: 6e 74 72 6f 6c 28 76 6f 69 64 29 7b 0a 20 20 2f ntrol(void){. /
1420: 2a 20 46 49 58 4d 45 3a 20 74 68 65 72 65 27 73 * FIXME: there's
1430: 20 73 6f 6d 65 20 67 6f 74 63 68 61 73 20 77 74 some gotchas wt
1440: 68 20 63 6f 6f 6b 69 65 73 20 61 6e 64 20 73 6f h cookies and so
1450: 6d 65 20 68 65 61 64 65 72 73 2e 20 2a 2f 0a 20 me headers. */.
1460: 20 63 68 61 72 20 2a 7a 45 54 61 67 20 3d 20 63 char *zETag = c
1470: 67 69 5f 61 64 64 5f 65 74 61 67 28 62 6c 6f 62 gi_add_etag(blob
1480: 5f 62 75 66 66 65 72 28 26 63 67 69 43 6f 6e 74 _buffer(&cgiCont
1490: 65 6e 74 29 2c 62 6c 6f 62 5f 73 69 7a 65 28 26 ent),blob_size(&
14a0: 63 67 69 43 6f 6e 74 65 6e 74 29 29 3b 0a 20 20 cgiContent));.
14b0: 63 68 61 72 20 2a 7a 4d 61 74 63 68 20 3d 20 50 char *zMatch = P
14c0: 28 22 48 54 54 50 5f 49 46 5f 4e 4f 4e 45 5f 4d ("HTTP_IF_NONE_M
14d0: 41 54 43 48 22 29 3b 0a 0a 20 20 69 66 28 20 7a ATCH");.. if( z
14e0: 45 54 61 67 21 3d 30 20 26 26 20 7a 4d 61 74 63 ETag!=0 && zMatc
14f0: 68 21 3d 30 20 29 20 7b 0a 20 20 20 20 63 68 61 h!=0 ) {. cha
1500: 72 20 2a 7a 42 75 66 20 3d 20 73 74 72 64 75 70 r *zBuf = strdup
1510: 28 7a 4d 61 74 63 68 29 3b 0a 20 20 20 20 69 66 (zMatch);. if
1520: 28 20 7a 42 75 66 21 3d 30 20 29 7b 0a 20 20 20 ( zBuf!=0 ){.
1530: 20 20 20 63 68 61 72 20 2a 7a 54 6f 6b 20 3d 20 char *zTok =
1540: 30 3b 0a 20 20 20 20 20 20 63 68 61 72 20 2a 7a 0;. char *z
1550: 50 6f 73 3b 0a 20 20 20 20 20 20 66 6f 72 28 20 Pos;. for(
1560: 7a 54 6f 6b 20 3d 20 73 74 72 74 6f 6b 5f 72 28 zTok = strtok_r(
1570: 7a 42 75 66 2c 20 22 2c 5c 22 22 2c 26 7a 50 6f zBuf, ",\"",&zPo
1580: 73 29 3b 0a 20 20 20 20 20 20 20 20 20 20 20 7a s);. z
1590: 54 6f 6b 20 26 26 20 73 74 72 63 61 73 65 63 6d Tok && strcasecm
15a0: 70 28 7a 54 6f 6b 2c 7a 45 54 61 67 29 3b 0a 20 p(zTok,zETag);.
15b0: 20 20 20 20 20 20 20 20 20 20 7a 54 6f 6b 20 3d zTok =
15c0: 20 20 73 74 72 74 6f 6b 5f 72 28 30 2c 20 22 2c strtok_r(0, ",
15d0: 5c 22 22 2c 26 7a 50 6f 73 29 29 7b 7d 0a 20 20 \"",&zPos)){}.
15e0: 20 20 20 20 66 72 65 65 28 7a 42 75 66 29 3b 0a free(zBuf);.
15f0: 20 20 20 20 20 20 69 66 28 7a 54 6f 6b 29 20 72 if(zTok) r
1600: 65 74 75 72 6e 20 31 3b 0a 20 20 20 20 7d 0a 20 eturn 1;. }.
1610: 20 7d 0a 20 20 0a 20 20 72 65 74 75 72 6e 20 30 }. . return 0
1620: 3b 0a 7d 0a 23 65 6e 64 69 66 0a 0a 2f 2a 0a 2a ;.}.#endif../*.*
1630: 2a 20 44 6f 20 61 20 6e 6f 72 6d 61 6c 20 48 54 * Do a normal HT
1640: 54 50 20 72 65 70 6c 79 0a 2a 2f 0a 76 6f 69 64 TP reply.*/.void
1650: 20 63 67 69 5f 72 65 70 6c 79 28 76 6f 69 64 29 cgi_reply(void)
1660: 7b 0a 20 20 69 66 28 20 69 52 65 70 6c 79 53 74 {. if( iReplySt
1670: 61 74 75 73 3c 3d 30 20 29 7b 0a 20 20 20 20 69 atus<=0 ){. i
1680: 52 65 70 6c 79 53 74 61 74 75 73 20 3d 20 32 30 ReplyStatus = 20
1690: 30 3b 0a 20 20 20 20 7a 52 65 70 6c 79 53 74 61 0;. zReplySta
16a0: 74 75 73 20 3d 20 22 4f 4b 22 3b 0a 20 20 7d 0a tus = "OK";. }.
16b0: 0a 23 69 66 20 30 0a 20 20 69 66 28 20 69 52 65 .#if 0. if( iRe
16c0: 70 6c 79 53 74 61 74 75 73 3d 3d 32 30 30 20 26 plyStatus==200 &
16d0: 26 20 63 68 65 63 6b 5f 63 61 63 68 65 5f 63 6f & check_cache_co
16e0: 6e 74 72 6f 6c 28 29 20 29 20 7b 0a 20 20 20 20 ntrol() ) {.
16f0: 2f 2a 20 63 68 61 6e 67 65 20 74 68 65 20 73 74 /* change the st
1700: 61 74 75 73 20 74 6f 20 22 75 6e 63 68 61 6e 67 atus to "unchang
1710: 65 64 22 20 61 6e 64 20 77 65 20 63 61 6e 20 73 ed" and we can s
1720: 6b 69 70 20 73 65 6e 64 69 6e 67 20 74 68 65 0a kip sending the.
1730: 20 20 20 20 2a 2a 20 61 63 74 75 61 6c 20 72 65 ** actual re
1740: 73 70 6f 6e 73 65 20 62 6f 64 79 2e 20 4f 62 76 sponse body. Obv
1750: 69 6f 75 73 6c 79 20 77 65 20 6f 6e 6c 79 20 64 iously we only d
1760: 6f 20 74 68 69 73 20 77 68 65 6e 20 77 65 20 5f o this when we _
1770: 68 61 76 65 5f 20 61 0a 20 20 20 20 2a 2a 20 62 have_ a. ** b
1780: 6f 64 79 20 28 63 6f 64 65 20 32 30 30 29 2e 0a ody (code 200)..
1790: 20 20 20 20 2a 2f 0a 20 20 20 20 69 52 65 70 6c */. iRepl
17a0: 79 53 74 61 74 75 73 20 3d 20 33 30 34 3b 0a 20 yStatus = 304;.
17b0: 20 20 20 7a 52 65 70 6c 79 53 74 61 74 75 73 20 zReplyStatus
17c0: 3d 20 22 4e 6f 74 20 4d 6f 64 69 66 69 65 64 22 = "Not Modified"
17d0: 3b 0a 20 20 7d 0a 23 65 6e 64 69 66 0a 0a 20 20 ;. }.#endif..
17e0: 69 66 28 20 66 75 6c 6c 48 74 74 70 52 65 70 6c if( fullHttpRepl
17f0: 79 20 29 7b 0a 20 20 20 20 70 72 69 6e 74 66 28 y ){. printf(
1800: 22 48 54 54 50 2f 31 2e 30 20 25 64 20 25 73 5c "HTTP/1.0 %d %s\
1810: 72 5c 6e 22 2c 20 69 52 65 70 6c 79 53 74 61 74 r\n", iReplyStat
1820: 75 73 2c 20 7a 52 65 70 6c 79 53 74 61 74 75 73 us, zReplyStatus
1830: 29 3b 0a 20 20 20 20 70 72 69 6e 74 66 28 22 44 );. printf("D
1840: 61 74 65 3a 20 25 73 5c 72 5c 6e 22 2c 20 63 67 ate: %s\r\n", cg
1850: 69 5f 72 66 63 38 32 32 5f 64 61 74 65 73 74 61 i_rfc822_datesta
1860: 6d 70 28 74 69 6d 65 28 30 29 29 29 3b 0a 20 20 mp(time(0)));.
1870: 20 20 70 72 69 6e 74 66 28 22 43 6f 6e 6e 65 63 printf("Connec
1880: 74 69 6f 6e 3a 20 63 6c 6f 73 65 5c 72 5c 6e 22 tion: close\r\n"
1890: 29 3b 0a 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 );. }else{.
18a0: 70 72 69 6e 74 66 28 22 53 74 61 74 75 73 3a 20 printf("Status:
18b0: 25 64 20 25 73 5c 72 5c 6e 22 2c 20 69 52 65 70 %d %s\r\n", iRep
18c0: 6c 79 53 74 61 74 75 73 2c 20 7a 52 65 70 6c 79 lyStatus, zReply
18d0: 53 74 61 74 75 73 29 3b 0a 20 20 7d 0a 0a 20 20 Status);. }..
18e0: 69 66 28 20 62 6c 6f 62 5f 73 69 7a 65 28 26 65 if( blob_size(&e
18f0: 78 74 72 61 48 65 61 64 65 72 29 3e 30 20 29 7b xtraHeader)>0 ){
1900: 0a 20 20 20 20 70 72 69 6e 74 66 28 22 25 73 22 . printf("%s"
1910: 2c 20 62 6c 6f 62 5f 62 75 66 66 65 72 28 26 65 , blob_buffer(&e
1920: 78 74 72 61 48 65 61 64 65 72 29 29 3b 0a 20 20 xtraHeader));.
1930: 7d 0a 0a 20 20 69 66 28 20 67 2e 69 73 43 6f 6e }.. if( g.isCon
1940: 73 74 20 29 7b 0a 20 20 20 20 2f 2a 20 63 6f 6e st ){. /* con
1950: 73 74 61 6e 74 20 6d 65 61 6e 73 20 74 68 61 74 stant means that
1960: 20 74 68 65 20 69 6e 70 75 74 20 55 52 4c 20 77 the input URL w
1970: 69 6c 6c 20 5f 6e 65 76 65 72 5f 20 67 65 6e 65 ill _never_ gene
1980: 72 61 74 65 20 61 6e 79 74 68 69 6e 67 0a 20 20 rate anything.
1990: 20 20 2a 2a 20 65 6c 73 65 2e 20 49 6e 20 74 68 ** else. In th
19a0: 65 20 63 61 73 65 20 6f 66 20 61 74 74 61 63 68 e case of attach
19b0: 6d 65 6e 74 73 2c 20 74 68 65 20 63 6f 6e 74 65 ments, the conte
19c0: 6e 74 73 20 77 6f 6e 27 74 20 63 68 61 6e 67 65 nts won't change
19d0: 20 62 65 63 61 75 73 65 0a 20 20 20 20 2a 2a 20 because. **
19e0: 61 6e 20 61 74 74 65 6d 70 74 20 74 6f 20 63 68 an attempt to ch
19f0: 61 6e 67 65 20 74 68 65 6d 20 67 65 6e 65 72 61 ange them genera
1a00: 74 65 73 20 61 20 6e 65 77 20 61 74 74 61 63 68 tes a new attach
1a10: 6d 65 6e 74 20 6e 75 6d 62 65 72 2e 20 49 6e 20 ment number. In
1a20: 74 68 65 0a 20 20 20 20 2a 2a 20 63 61 73 65 20 the. ** case
1a30: 6f 66 20 6d 6f 73 74 20 2f 67 65 74 66 69 6c 65 of most /getfile
1a40: 20 63 61 6c 6c 73 20 66 6f 72 20 73 70 65 63 69 calls for speci
1a50: 66 69 63 20 76 65 72 73 69 6f 6e 73 2c 20 74 68 fic versions, th
1a60: 65 20 6f 6e 6c 79 20 77 61 79 20 74 68 65 0a 20 e only way the.
1a70: 20 20 20 2a 2a 20 63 6f 6e 74 65 6e 74 20 63 68 ** content ch
1a80: 61 6e 67 65 73 20 69 73 20 69 66 20 73 6f 6d 65 anges is if some
1a90: 6f 6e 65 20 62 72 65 61 6b 73 20 74 68 65 20 53 one breaks the S
1aa0: 43 4d 2e 20 41 6e 64 20 69 66 20 74 68 61 74 20 CM. And if that
1ab0: 68 61 70 70 65 6e 73 2c 20 61 0a 20 20 20 20 2a happens, a. *
1ac0: 2a 20 73 74 61 6c 65 20 63 61 63 68 65 20 69 73 * stale cache is
1ad0: 20 74 68 65 20 6c 65 61 73 74 20 6f 66 20 74 68 the least of th
1ae0: 65 20 70 72 6f 62 6c 65 6d 2e 20 53 6f 20 77 65 e problem. So we
1af0: 20 70 72 6f 76 69 64 65 20 61 6e 20 45 78 70 69 provide an Expi
1b00: 72 65 73 0a 20 20 20 20 2a 2a 20 68 65 61 64 65 res. ** heade
1b10: 72 20 73 65 74 20 74 6f 20 61 20 72 65 61 73 6f r set to a reaso
1b20: 6e 61 62 6c 65 20 70 65 72 69 6f 64 20 28 64 65 nable period (de
1b30: 66 61 75 6c 74 3a 20 6f 6e 65 20 77 65 65 6b 29 fault: one week)
1b40: 2e 0a 20 20 20 20 2a 2f 0a 20 20 20 20 2f 2a 74 .. */. /*t
1b50: 69 6d 65 5f 74 20 65 78 70 69 72 65 73 20 3d 20 ime_t expires =
1b60: 74 69 6d 65 28 30 29 20 2b 20 61 74 6f 69 28 64 time(0) + atoi(d
1b70: 62 5f 63 6f 6e 66 69 67 28 22 63 6f 6e 73 74 61 b_config("consta
1b80: 6e 74 5f 65 78 70 69 72 65 73 22 2c 22 36 30 34 nt_expires","604
1b90: 38 30 30 22 29 29 3b 2a 2f 0a 20 20 20 20 74 69 800"));*/. ti
1ba0: 6d 65 5f 74 20 65 78 70 69 72 65 73 20 3d 20 74 me_t expires = t
1bb0: 69 6d 65 28 30 29 20 2b 20 36 30 34 38 30 30 3b ime(0) + 604800;
1bc0: 0a 20 20 20 20 70 72 69 6e 74 66 28 20 22 45 78 . printf( "Ex
1bd0: 70 69 72 65 73 3a 20 25 73 5c 72 5c 6e 22 2c 20 pires: %s\r\n",
1be0: 63 67 69 5f 72 66 63 38 32 32 5f 64 61 74 65 73 cgi_rfc822_dates
1bf0: 74 61 6d 70 28 65 78 70 69 72 65 73 29 29 3b 0a tamp(expires));.
1c00: 20 20 7d 0a 0a 20 20 2f 2a 20 43 6f 6e 74 65 6e }.. /* Conten
1c10: 74 20 69 6e 74 65 6e 64 65 64 20 66 6f 72 20 6c t intended for l
1c20: 6f 67 67 65 64 20 69 6e 20 75 73 65 72 73 20 73 ogged in users s
1c30: 68 6f 75 6c 64 20 6f 6e 6c 79 20 62 65 20 63 61 hould only be ca
1c40: 63 68 65 64 20 69 6e 0a 20 20 2a 2a 20 74 68 65 ched in. ** the
1c50: 20 62 72 6f 77 73 65 72 2c 20 6e 6f 74 20 73 6f browser, not so
1c60: 6d 65 20 73 68 61 72 65 64 20 6c 6f 63 61 74 69 me shared locati
1c70: 6f 6e 2e 0a 20 20 2a 2f 0a 20 20 70 72 69 6e 74 on.. */. print
1c80: 66 28 22 43 61 63 68 65 2d 63 6f 6e 74 72 6f 6c f("Cache-control
1c90: 3a 20 70 72 69 76 61 74 65 5c 72 5c 6e 22 29 3b : private\r\n");
1ca0: 0a 0a 23 69 66 20 46 4f 53 53 49 4c 5f 49 31 38 ..#if FOSSIL_I18
1cb0: 4e 0a 20 20 70 72 69 6e 74 66 28 20 22 43 6f 6e N. printf( "Con
1cc0: 74 65 6e 74 2d 54 79 70 65 3a 20 25 73 3b 20 63 tent-Type: %s; c
1cd0: 68 61 72 73 65 74 3d 25 73 5c 72 5c 6e 22 2c 20 harset=%s\r\n",
1ce0: 7a 43 6f 6e 74 65 6e 74 54 79 70 65 2c 20 6e 6c zContentType, nl
1cf0: 5f 6c 61 6e 67 69 6e 66 6f 28 43 4f 44 45 53 45 _langinfo(CODESE
1d00: 54 29 29 3b 0a 23 65 6c 73 65 0a 20 20 70 72 69 T));.#else. pri
1d10: 6e 74 66 28 20 22 43 6f 6e 74 65 6e 74 2d 54 79 ntf( "Content-Ty
1d20: 70 65 3a 20 25 73 3b 20 63 68 61 72 73 65 74 3d pe: %s; charset=
1d30: 49 53 4f 2d 38 38 35 39 2d 31 5c 72 5c 6e 22 2c ISO-8859-1\r\n",
1d40: 20 7a 43 6f 6e 74 65 6e 74 54 79 70 65 29 3b 0a zContentType);.
1d50: 23 65 6e 64 69 66 0a 20 20 69 66 28 20 73 74 72 #endif. if( str
1d60: 63 6d 70 28 7a 43 6f 6e 74 65 6e 74 54 79 70 65 cmp(zContentType
1d70: 2c 22 61 70 70 6c 69 63 61 74 69 6f 6e 2f 78 2d ,"application/x-
1d80: 66 6f 73 73 69 6c 22 29 3d 3d 30 20 29 7b 0a 20 fossil")==0 ){.
1d90: 20 20 20 62 6c 6f 62 5f 63 6f 6d 70 72 65 73 73 blob_compress
1da0: 28 26 63 67 69 43 6f 6e 74 65 6e 74 2c 20 26 63 (&cgiContent, &c
1db0: 67 69 43 6f 6e 74 65 6e 74 29 3b 0a 20 20 7d 0a giContent);. }.
1dc0: 0a 20 20 69 66 28 20 69 52 65 70 6c 79 53 74 61 . if( iReplySta
1dd0: 74 75 73 20 21 3d 20 33 30 34 20 29 20 7b 0a 20 tus != 304 ) {.
1de0: 20 20 20 70 72 69 6e 74 66 28 20 22 43 6f 6e 74 printf( "Cont
1df0: 65 6e 74 2d 4c 65 6e 67 74 68 3a 20 25 64 5c 72 ent-Length: %d\r
1e00: 5c 6e 22 2c 20 62 6c 6f 62 5f 73 69 7a 65 28 26 \n", blob_size(&
1e10: 63 67 69 43 6f 6e 74 65 6e 74 29 20 29 3b 0a 20 cgiContent) );.
1e20: 20 7d 0a 20 20 70 72 69 6e 74 66 28 22 5c 72 5c }. printf("\r\
1e30: 6e 22 29 3b 0a 20 20 69 66 28 20 62 6c 6f 62 5f n");. if( blob_
1e40: 73 69 7a 65 28 26 63 67 69 43 6f 6e 74 65 6e 74 size(&cgiContent
1e50: 29 3e 30 20 26 26 20 69 52 65 70 6c 79 53 74 61 )>0 && iReplySta
1e60: 74 75 73 20 21 3d 20 33 30 34 20 29 7b 0a 20 20 tus != 304 ){.
1e70: 20 20 66 77 72 69 74 65 28 62 6c 6f 62 5f 62 75 fwrite(blob_bu
1e80: 66 66 65 72 28 26 63 67 69 43 6f 6e 74 65 6e 74 ffer(&cgiContent
1e90: 29 2c 20 31 2c 20 62 6c 6f 62 5f 73 69 7a 65 28 ), 1, blob_size(
1ea0: 26 63 67 69 43 6f 6e 74 65 6e 74 29 2c 20 73 74 &cgiContent), st
1eb0: 64 6f 75 74 29 3b 0a 20 20 7d 0a 20 20 43 47 49 dout);. }. CGI
1ec0: 44 45 42 55 47 28 28 22 44 4f 4e 45 5c 6e 22 29 DEBUG(("DONE\n")
1ed0: 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 44 6f 20 61 );.}../*.** Do a
1ee0: 20 72 65 64 69 72 65 63 74 20 72 65 71 75 65 73 redirect reques
1ef0: 74 20 74 6f 20 74 68 65 20 55 52 4c 20 67 69 76 t to the URL giv
1f00: 65 6e 20 69 6e 20 74 68 65 20 61 72 67 75 6d 65 en in the argume
1f10: 6e 74 2e 0a 2a 2a 0a 2a 2a 20 54 68 65 20 55 52 nt..**.** The UR
1f20: 4c 20 6d 75 73 74 20 62 65 20 72 65 6c 61 74 69 L must be relati
1f30: 76 65 20 74 6f 20 74 68 65 20 62 61 73 65 20 6f ve to the base o
1f40: 66 20 74 68 65 20 66 6f 73 73 69 6c 20 73 65 72 f the fossil ser
1f50: 76 65 72 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 ver..*/.void cgi
1f60: 5f 72 65 64 69 72 65 63 74 28 63 6f 6e 73 74 20 _redirect(const
1f70: 63 68 61 72 20 2a 7a 55 52 4c 29 7b 0a 20 20 63 char *zURL){. c
1f80: 68 61 72 20 2a 7a 4c 6f 63 61 74 69 6f 6e 3b 0a har *zLocation;.
1f90: 20 20 43 47 49 44 45 42 55 47 28 28 22 72 65 64 CGIDEBUG(("red
1fa0: 69 72 65 63 74 20 74 6f 20 25 73 5c 6e 22 2c 20 irect to %s\n",
1fb0: 7a 55 52 4c 29 29 3b 0a 20 20 69 66 28 20 73 74 zURL));. if( st
1fc0: 72 6e 63 6d 70 28 7a 55 52 4c 2c 22 68 74 74 70 rncmp(zURL,"http
1fd0: 3a 22 2c 35 29 3d 3d 30 20 7c 7c 20 73 74 72 6e :",5)==0 || strn
1fe0: 63 6d 70 28 7a 55 52 4c 2c 22 68 74 74 70 73 3a cmp(zURL,"https:
1ff0: 22 2c 36 29 3d 3d 30 20 7c 7c 20 2a 7a 55 52 4c ",6)==0 || *zURL
2000: 3d 3d 27 2f 27 20 29 7b 0a 20 20 20 20 63 67 69 =='/' ){. cgi
2010: 5f 70 61 6e 69 63 28 22 69 6e 76 61 6c 69 64 20 _panic("invalid
2020: 72 65 64 69 72 65 63 74 20 55 52 4c 3a 20 25 73 redirect URL: %s
2030: 22 2c 20 7a 55 52 4c 29 3b 0a 20 20 7d 0a 20 20 ", zURL);. }.
2040: 7a 4c 6f 63 61 74 69 6f 6e 20 3d 20 6d 70 72 69 zLocation = mpri
2050: 6e 74 66 28 22 4c 6f 63 61 74 69 6f 6e 3a 20 25 ntf("Location: %
2060: 73 2f 25 73 5c 72 5c 6e 22 2c 20 67 2e 7a 42 61 s/%s\r\n", g.zBa
2070: 73 65 55 52 4c 2c 20 7a 55 52 4c 29 3b 0a 20 20 seURL, zURL);.
2080: 63 67 69 5f 61 70 70 65 6e 64 5f 68 65 61 64 65 cgi_append_heade
2090: 72 28 7a 4c 6f 63 61 74 69 6f 6e 29 3b 0a 20 20 r(zLocation);.
20a0: 63 67 69 5f 72 65 73 65 74 5f 63 6f 6e 74 65 6e cgi_reset_conten
20b0: 74 28 29 3b 0a 20 20 63 67 69 5f 70 72 69 6e 74 t();. cgi_print
20c0: 66 28 22 3c 68 74 6d 6c 3e 5c 6e 3c 70 3e 52 65 f("<html>\n<p>Re
20d0: 64 69 72 65 63 74 20 74 6f 20 25 68 3c 2f 70 3e direct to %h</p>
20e0: 5c 6e 3c 2f 68 74 6d 6c 3e 5c 6e 22 2c 20 7a 55 \n</html>\n", zU
20f0: 52 4c 29 3b 0a 20 20 63 67 69 5f 73 65 74 5f 73 RL);. cgi_set_s
2100: 74 61 74 75 73 28 33 30 32 2c 20 22 4d 6f 76 65 tatus(302, "Move
2110: 64 20 54 65 6d 70 6f 72 61 72 69 6c 79 22 29 3b d Temporarily");
2120: 0a 20 20 66 72 65 65 28 7a 4c 6f 63 61 74 69 6f . free(zLocatio
2130: 6e 29 3b 0a 20 20 63 67 69 5f 72 65 70 6c 79 28 n);. cgi_reply(
2140: 29 3b 0a 20 20 65 78 69 74 28 30 29 3b 0a 7d 0a );. exit(0);.}.
2150: 0a 2f 2a 0a 2a 2a 20 49 6e 66 6f 72 6d 61 74 69 ./*.** Informati
2160: 6f 6e 20 61 62 6f 75 74 20 61 6c 6c 20 71 75 65 on about all que
2170: 72 79 20 70 61 72 61 6d 65 74 65 72 73 20 61 6e ry parameters an
2180: 64 20 63 6f 6f 6b 69 65 73 20 61 72 65 20 73 74 d cookies are st
2190: 6f 72 65 64 0a 2a 2a 20 69 6e 20 74 68 65 73 65 ored.** in these
21a0: 20 76 61 72 69 61 62 6c 65 73 2e 0a 2a 2f 0a 73 variables..*/.s
21b0: 74 61 74 69 63 20 69 6e 74 20 6e 41 6c 6c 6f 63 tatic int nAlloc
21c0: 51 50 20 3d 20 30 3b 20 2f 2a 20 53 70 61 63 65 QP = 0; /* Space
21d0: 20 61 6c 6c 6f 63 61 74 65 64 20 66 6f 72 20 61 allocated for a
21e0: 50 61 72 61 6d 51 50 5b 5d 20 2a 2f 0a 73 74 61 ParamQP[] */.sta
21f0: 74 69 63 20 69 6e 74 20 6e 55 73 65 64 51 50 20 tic int nUsedQP
2200: 3d 20 30 3b 20 20 2f 2a 20 53 70 61 63 65 20 61 = 0; /* Space a
2210: 63 74 75 61 6c 6c 79 20 75 73 65 64 20 69 6e 20 ctually used in
2220: 61 50 61 72 61 6d 51 50 5b 5d 20 2a 2f 0a 73 74 aParamQP[] */.st
2230: 61 74 69 63 20 69 6e 74 20 73 6f 72 74 51 50 20 atic int sortQP
2240: 3d 20 30 3b 20 20 20 2f 2a 20 54 72 75 65 20 69 = 0; /* True i
2250: 66 20 61 50 61 72 61 6d 51 50 5b 5d 20 6e 65 65 f aParamQP[] nee
2260: 64 73 20 73 6f 72 74 69 6e 67 20 2a 2f 0a 73 74 ds sorting */.st
2270: 61 74 69 63 20 69 6e 74 20 73 65 71 51 50 20 3d atic int seqQP =
2280: 20 30 3b 20 20 20 20 2f 2a 20 53 65 71 75 65 6e 0; /* Sequen
2290: 63 65 20 6e 75 6d 62 65 72 73 20 2a 2f 0a 73 74 ce numbers */.st
22a0: 61 74 69 63 20 73 74 72 75 63 74 20 51 50 61 72 atic struct QPar
22b0: 61 6d 20 7b 20 20 20 2f 2a 20 4f 6e 65 20 65 6e am { /* One en
22c0: 74 72 79 20 66 6f 72 20 65 61 63 68 20 71 75 65 try for each que
22d0: 72 79 20 70 61 72 61 6d 65 74 65 72 20 6f 72 20 ry parameter or
22e0: 63 6f 6f 6b 69 65 20 2a 2f 0a 20 20 63 6f 6e 73 cookie */. cons
22f0: 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65 3b 20 20 t char *zName;
2300: 20 20 20 20 20 20 2f 2a 20 50 61 72 61 6d 65 74 /* Paramet
2310: 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 6e 61 6d er or cookie nam
2320: 65 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 e */. const cha
2330: 72 20 2a 7a 56 61 6c 75 65 3b 20 20 20 20 20 20 r *zValue;
2340: 20 2f 2a 20 56 61 6c 75 65 20 6f 66 20 74 68 65 /* Value of the
2350: 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 query parameter
2360: 20 6f 72 20 63 6f 6f 6b 69 65 20 2a 2f 0a 20 20 or cookie */.
2370: 69 6e 74 20 73 65 71 3b 20 20 20 20 20 20 20 20 int seq;
2380: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4f 72 64 /* Ord
2390: 65 72 20 6f 66 20 69 6e 73 65 72 74 69 6f 6e 20 er of insertion
23a0: 2a 2f 0a 7d 20 2a 61 50 61 72 61 6d 51 50 3b 20 */.} *aParamQP;
23b0: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 41 /* A
23c0: 6e 20 61 72 72 61 79 20 6f 66 20 61 6c 6c 20 70 n array of all p
23d0: 61 72 61 6d 65 74 65 72 73 20 61 6e 64 20 63 6f arameters and co
23e0: 6f 6b 69 65 73 20 2a 2f 0a 0a 2f 2a 0a 2a 2a 20 okies */../*.**
23f0: 41 64 64 20 61 6e 6f 74 68 65 72 20 71 75 65 72 Add another quer
2400: 79 20 70 61 72 61 6d 65 74 65 72 20 6f 72 20 63 y parameter or c
2410: 6f 6f 6b 69 65 20 74 6f 20 74 68 65 20 70 61 72 ookie to the par
2420: 61 6d 65 74 65 72 20 73 65 74 2e 0a 2a 2a 20 7a ameter set..** z
2430: 4e 61 6d 65 20 69 73 20 74 68 65 20 6e 61 6d 65 Name is the name
2440: 20 6f 66 20 74 68 65 20 71 75 65 72 79 20 70 61 of the query pa
2450: 72 61 6d 65 74 65 72 20 6f 72 20 63 6f 6f 6b 69 rameter or cooki
2460: 65 20 61 6e 64 20 7a 56 61 6c 75 65 0a 2a 2a 20 e and zValue.**
2470: 69 73 20 69 74 73 20 66 75 6c 6c 79 20 64 65 63 is its fully dec
2480: 6f 64 65 64 20 76 61 6c 75 65 2e 0a 2a 2a 0a 2a oded value..**.*
2490: 2a 20 7a 4e 61 6d 65 20 61 6e 64 20 7a 56 61 6c * zName and zVal
24a0: 75 65 20 61 72 65 20 6e 6f 74 20 63 6f 70 69 65 ue are not copie
24b0: 64 20 61 6e 64 20 6d 75 73 74 20 6e 6f 74 20 63 d and must not c
24c0: 68 61 6e 67 65 20 6f 72 20 62 65 0a 2a 2a 20 64 hange or be.** d
24d0: 65 61 6c 6c 6f 63 61 74 65 64 20 61 66 74 65 72 eallocated after
24e0: 20 74 68 69 73 20 72 6f 75 74 69 6e 65 20 72 65 this routine re
24f0: 74 75 72 6e 73 2e 0a 2a 2f 0a 73 74 61 74 69 63 turns..*/.static
2500: 20 76 6f 69 64 20 63 67 69 5f 73 65 74 5f 70 61 void cgi_set_pa
2510: 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79 28 63 rameter_nocopy(c
2520: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65 onst char *zName
2530: 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 56 , const char *zV
2540: 61 6c 75 65 29 7b 0a 20 20 69 66 28 20 6e 41 6c alue){. if( nAl
2550: 6c 6f 63 51 50 3c 3d 6e 55 73 65 64 51 50 20 29 locQP<=nUsedQP )
2560: 7b 0a 20 20 20 20 6e 41 6c 6c 6f 63 51 50 20 3d {. nAllocQP =
2570: 20 6e 41 6c 6c 6f 63 51 50 2a 32 20 2b 20 31 30 nAllocQP*2 + 10
2580: 3b 0a 20 20 20 20 61 50 61 72 61 6d 51 50 20 3d ;. aParamQP =
2590: 20 72 65 61 6c 6c 6f 63 28 20 61 50 61 72 61 6d realloc( aParam
25a0: 51 50 2c 20 6e 41 6c 6c 6f 63 51 50 2a 73 69 7a QP, nAllocQP*siz
25b0: 65 6f 66 28 61 50 61 72 61 6d 51 50 5b 30 5d 29 eof(aParamQP[0])
25c0: 20 29 3b 0a 20 20 20 20 69 66 28 20 61 50 61 72 );. if( aPar
25d0: 61 6d 51 50 3d 3d 30 20 29 20 65 78 69 74 28 31 amQP==0 ) exit(1
25e0: 29 3b 0a 20 20 7d 0a 20 20 61 50 61 72 61 6d 51 );. }. aParamQ
25f0: 50 5b 6e 55 73 65 64 51 50 5d 2e 7a 4e 61 6d 65 P[nUsedQP].zName
2600: 20 3d 20 7a 4e 61 6d 65 3b 0a 20 20 61 50 61 72 = zName;. aPar
2610: 61 6d 51 50 5b 6e 55 73 65 64 51 50 5d 2e 7a 56 amQP[nUsedQP].zV
2620: 61 6c 75 65 20 3d 20 7a 56 61 6c 75 65 3b 0a 20 alue = zValue;.
2630: 20 61 50 61 72 61 6d 51 50 5b 6e 55 73 65 64 51 aParamQP[nUsedQ
2640: 50 5d 2e 73 65 71 20 3d 20 73 65 71 51 50 2b 2b P].seq = seqQP++
2650: 3b 0a 20 20 6e 55 73 65 64 51 50 2b 2b 3b 0a 20 ;. nUsedQP++;.
2660: 20 73 6f 72 74 51 50 20 3d 20 31 3b 0a 7d 0a 0a sortQP = 1;.}..
2670: 2f 2a 0a 2a 2a 20 41 64 64 20 61 6e 6f 74 68 65 /*.** Add anothe
2680: 72 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 r query paramete
2690: 72 20 6f 72 20 63 6f 6f 6b 69 65 20 74 6f 20 74 r or cookie to t
26a0: 68 65 20 70 61 72 61 6d 65 74 65 72 20 73 65 74 he parameter set
26b0: 2e 0a 2a 2a 20 7a 4e 61 6d 65 20 69 73 20 74 68 ..** zName is th
26c0: 65 20 6e 61 6d 65 20 6f 66 20 74 68 65 20 71 75 e name of the qu
26d0: 65 72 79 20 70 61 72 61 6d 65 74 65 72 20 6f 72 ery parameter or
26e0: 20 63 6f 6f 6b 69 65 20 61 6e 64 20 7a 56 61 6c cookie and zVal
26f0: 75 65 0a 2a 2a 20 69 73 20 69 74 73 20 66 75 6c ue.** is its ful
2700: 6c 79 20 64 65 63 6f 64 65 64 20 76 61 6c 75 65 ly decoded value
2710: 2e 0a 2a 2a 0a 2a 2a 20 43 6f 70 69 65 73 20 61 ..**.** Copies a
2720: 72 65 20 6d 61 64 65 20 6f 66 20 62 6f 74 68 20 re made of both
2730: 74 68 65 20 7a 4e 61 6d 65 20 61 6e 64 20 7a 56 the zName and zV
2740: 61 6c 75 65 20 70 61 72 61 6d 65 74 65 72 73 2e alue parameters.
2750: 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 73 65 74 .*/.void cgi_set
2760: 5f 70 61 72 61 6d 65 74 65 72 28 63 6f 6e 73 74 _parameter(const
2770: 20 63 68 61 72 20 2a 7a 4e 61 6d 65 2c 20 63 6f char *zName, co
2780: 6e 73 74 20 63 68 61 72 20 2a 7a 56 61 6c 75 65 nst char *zValue
2790: 29 7b 0a 20 20 63 67 69 5f 73 65 74 5f 70 61 72 ){. cgi_set_par
27a0: 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79 28 6d 70 ameter_nocopy(mp
27b0: 72 69 6e 74 66 28 22 25 73 22 2c 7a 4e 61 6d 65 rintf("%s",zName
27c0: 29 2c 20 6d 70 72 69 6e 74 66 28 22 25 73 22 2c ), mprintf("%s",
27d0: 7a 56 61 6c 75 65 29 29 3b 0a 7d 0a 0a 2f 2a 0a zValue));.}../*.
27e0: 2a 2a 20 41 64 64 20 61 20 71 75 65 72 79 20 70 ** Add a query p
27f0: 61 72 61 6d 65 74 65 72 2e 20 20 54 68 65 20 7a arameter. The z
2800: 4e 61 6d 65 20 70 6f 72 74 69 6f 6e 20 69 73 20 Name portion is
2810: 66 69 78 65 64 20 62 75 74 20 61 20 63 6f 70 79 fixed but a copy
2820: 0a 2a 2a 20 6d 75 73 74 20 62 65 20 6d 61 64 65 .** must be made
2830: 20 6f 66 20 7a 56 61 6c 75 65 2e 0a 2a 2f 0a 76 of zValue..*/.v
2840: 6f 69 64 20 63 67 69 5f 73 65 74 65 6e 76 28 63 oid cgi_setenv(c
2850: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65 onst char *zName
2860: 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 56 , const char *zV
2870: 61 6c 75 65 29 7b 0a 20 20 63 67 69 5f 73 65 74 alue){. cgi_set
2880: 5f 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 _parameter_nocop
2890: 79 28 7a 4e 61 6d 65 2c 20 6d 70 72 69 6e 74 66 y(zName, mprintf
28a0: 28 22 25 73 22 2c 7a 56 61 6c 75 65 29 29 3b 0a ("%s",zValue));.
28b0: 7d 0a 20 0a 0a 2f 2a 0a 2a 2a 20 41 64 64 20 61 }. ../*.** Add a
28c0: 20 6c 69 73 74 20 6f 66 20 71 75 65 72 79 20 70 list of query p
28d0: 61 72 61 6d 65 74 65 72 73 20 6f 72 20 63 6f 6f arameters or coo
28e0: 6b 69 65 73 20 74 6f 20 74 68 65 20 70 61 72 61 kies to the para
28f0: 6d 65 74 65 72 20 73 65 74 2e 0a 2a 2a 0a 2a 2a meter set..**.**
2900: 20 45 61 63 68 20 70 61 72 61 6d 65 74 65 72 20 Each parameter
2910: 69 73 20 6f 66 20 74 68 65 20 66 6f 72 6d 20 4e is of the form N
2920: 41 4d 45 3d 56 41 4c 55 45 2e 20 20 42 6f 74 68 AME=VALUE. Both
2930: 20 74 68 65 20 4e 41 4d 45 20 61 6e 64 20 74 68 the NAME and th
2940: 65 0a 2a 2a 20 56 41 4c 55 45 20 6d 61 79 20 62 e.** VALUE may b
2950: 65 20 75 72 6c 2d 65 6e 63 6f 64 65 64 20 28 22 e url-encoded ("
2960: 2b 22 20 66 6f 72 20 73 70 61 63 65 2c 20 22 25 +" for space, "%
2970: 48 48 22 20 66 6f 72 20 6f 74 68 65 72 20 73 70 HH" for other sp
2980: 65 63 69 61 6c 0a 2a 2a 20 63 68 61 72 61 63 74 ecial.** charact
2990: 65 72 73 29 2e 20 20 42 75 74 20 74 68 69 73 20 ers). But this
29a0: 72 6f 75 74 69 6e 65 20 61 73 73 75 6d 65 73 20 routine assumes
29b0: 74 68 61 74 20 4e 41 4d 45 20 63 6f 6e 74 61 69 that NAME contai
29c0: 6e 73 20 6e 6f 0a 2a 2a 20 73 70 65 63 69 61 6c ns no.** special
29d0: 20 63 68 61 72 61 63 74 65 72 20 61 6e 64 20 74 character and t
29e0: 68 65 72 65 66 6f 72 65 20 64 6f 65 73 20 6e 6f herefore does no
29f0: 74 20 64 65 63 6f 64 65 20 69 74 2e 0a 2a 2a 0a t decode it..**.
2a00: 2a 2a 20 49 66 20 4e 41 4d 45 20 62 65 67 69 6e ** If NAME begin
2a10: 73 20 77 69 74 68 20 61 6e 6f 74 68 65 72 20 6f s with another o
2a20: 74 68 65 72 20 74 68 61 6e 20 61 20 6c 6f 77 65 ther than a lowe
2a30: 72 2d 63 61 73 65 20 6c 65 74 74 65 72 20 74 68 r-case letter th
2a40: 65 6e 0a 2a 2a 20 74 68 65 20 65 6e 74 69 72 65 en.** the entire
2a50: 20 4e 41 4d 45 3d 56 41 4c 55 45 20 74 65 72 6d NAME=VALUE term
2a60: 20 69 73 20 69 67 6e 6f 72 65 64 2e 20 20 48 65 is ignored. He
2a70: 6e 63 65 3a 0a 2a 2a 0a 2a 2a 20 20 20 20 20 20 nce:.**.**
2a80: 2a 20 20 63 6f 6f 6b 69 65 73 20 61 6e 64 20 71 * cookies and q
2a90: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 73 20 uery parameters
2aa0: 74 68 61 74 20 68 61 76 65 20 75 70 70 65 72 63 that have upperc
2ab0: 61 73 65 20 6e 61 6d 65 73 0a 2a 2a 20 20 20 20 ase names.**
2ac0: 20 20 20 20 20 61 72 65 20 69 67 6e 6f 72 65 64 are ignored
2ad0: 2e 0a 2a 2a 0a 2a 2a 20 20 20 20 20 20 2a 20 20 ..**.** *
2ae0: 69 74 20 69 73 20 69 6d 70 6f 73 73 69 62 6c 65 it is impossible
2af0: 20 66 6f 72 20 61 20 63 6f 6f 6b 69 65 20 6f 72 for a cookie or
2b00: 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 query parameter
2b10: 20 74 6f 0a 2a 2a 20 20 20 20 20 20 20 20 20 6f to.** o
2b20: 76 65 72 72 69 64 65 20 74 68 65 20 76 61 6c 75 verride the valu
2b30: 65 20 6f 66 20 61 6e 20 65 6e 76 69 72 6f 6e 6d e of an environm
2b40: 65 6e 74 20 76 61 72 69 61 62 6c 65 20 73 69 6e ent variable sin
2b50: 63 65 0a 2a 2a 20 20 20 20 20 20 20 20 20 65 6e ce.** en
2b60: 76 69 72 6f 6e 6d 65 6e 74 20 76 61 72 69 61 62 vironment variab
2b70: 6c 65 73 20 61 6c 77 61 79 73 20 68 61 76 65 20 les always have
2b80: 75 70 70 65 72 63 61 73 65 20 6e 61 6d 65 73 2e uppercase names.
2b90: 0a 2a 2a 0a 2a 2a 20 50 61 72 61 6d 65 74 65 72 .**.** Parameter
2ba0: 73 20 61 72 65 20 73 65 70 61 72 61 74 65 64 20 s are separated
2bb0: 62 79 20 74 68 65 20 22 74 65 72 6d 69 6e 61 74 by the "terminat
2bc0: 6f 72 22 20 63 68 61 72 61 63 74 65 72 2e 20 20 or" character.
2bd0: 57 68 69 74 65 73 70 61 63 65 0a 2a 2a 20 62 65 Whitespace.** be
2be0: 66 6f 72 65 20 74 68 65 20 4e 41 4d 45 20 69 73 fore the NAME is
2bf0: 20 69 67 6e 6f 72 65 64 2e 0a 2a 2a 0a 2a 2a 20 ignored..**.**
2c00: 54 68 65 20 69 6e 70 75 74 20 73 74 72 69 6e 67 The input string
2c10: 20 22 7a 22 20 69 73 20 6d 6f 64 69 66 69 65 64 "z" is modified
2c20: 20 62 75 74 20 6e 6f 20 63 6f 70 69 65 73 20 69 but no copies i
2c30: 73 20 6d 61 64 65 2e 20 20 22 7a 22 0a 2a 2a 20 s made. "z".**
2c40: 73 68 6f 75 6c 64 20 6e 6f 74 20 62 65 20 64 65 should not be de
2c50: 61 6c 6c 6f 63 61 74 65 64 20 6f 72 20 63 68 61 allocated or cha
2c60: 6e 67 65 64 20 61 67 61 69 6e 20 61 66 74 65 72 nged again after
2c70: 20 74 68 69 73 20 72 6f 75 74 69 6e 65 0a 2a 2a this routine.**
2c80: 20 72 65 74 75 72 6e 73 20 6f 72 20 69 74 20 77 returns or it w
2c90: 69 6c 6c 20 63 6f 72 72 75 70 74 20 74 68 65 20 ill corrupt the
2ca0: 70 61 72 61 6d 65 74 65 72 20 74 61 62 6c 65 2e parameter table.
2cb0: 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20 .*/.static void
2cc0: 61 64 64 5f 70 61 72 61 6d 5f 6c 69 73 74 28 63 add_param_list(c
2cd0: 68 61 72 20 2a 7a 2c 20 69 6e 74 20 74 65 72 6d har *z, int term
2ce0: 69 6e 61 74 6f 72 29 7b 0a 20 20 77 68 69 6c 65 inator){. while
2cf0: 28 20 2a 7a 20 29 7b 0a 20 20 20 20 63 68 61 72 ( *z ){. char
2d00: 20 2a 7a 4e 61 6d 65 3b 0a 20 20 20 20 63 68 61 *zName;. cha
2d10: 72 20 2a 7a 56 61 6c 75 65 3b 0a 20 20 20 20 77 r *zValue;. w
2d20: 68 69 6c 65 28 20 69 73 73 70 61 63 65 28 2a 7a hile( isspace(*z
2d30: 29 20 29 7b 20 7a 2b 2b 3b 20 7d 0a 20 20 20 20 ) ){ z++; }.
2d40: 7a 4e 61 6d 65 20 3d 20 7a 3b 0a 20 20 20 20 77 zName = z;. w
2d50: 68 69 6c 65 28 20 2a 7a 20 26 26 20 2a 7a 21 3d hile( *z && *z!=
2d60: 27 3d 27 20 26 26 20 2a 7a 21 3d 74 65 72 6d 69 '=' && *z!=termi
2d70: 6e 61 74 6f 72 20 29 7b 20 7a 2b 2b 3b 20 7d 0a nator ){ z++; }.
2d80: 20 20 20 20 69 66 28 20 2a 7a 3d 3d 27 3d 27 20 if( *z=='='
2d90: 29 7b 0a 20 20 20 20 20 20 2a 7a 20 3d 20 30 3b ){. *z = 0;
2da0: 0a 20 20 20 20 20 20 7a 2b 2b 3b 0a 20 20 20 20 . z++;.
2db0: 20 20 7a 56 61 6c 75 65 20 3d 20 7a 3b 0a 20 20 zValue = z;.
2dc0: 20 20 20 20 77 68 69 6c 65 28 20 2a 7a 20 26 26 while( *z &&
2dd0: 20 2a 7a 21 3d 74 65 72 6d 69 6e 61 74 6f 72 20 *z!=terminator
2de0: 29 7b 20 7a 2b 2b 3b 20 7d 0a 20 20 20 20 20 20 ){ z++; }.
2df0: 69 66 28 20 2a 7a 20 29 7b 0a 20 20 20 20 20 20 if( *z ){.
2e00: 20 20 2a 7a 20 3d 20 30 3b 0a 20 20 20 20 20 20 *z = 0;.
2e10: 20 20 7a 2b 2b 3b 0a 20 20 20 20 20 20 7d 0a 20 z++;. }.
2e20: 20 20 20 20 20 64 65 68 74 74 70 69 7a 65 28 7a dehttpize(z
2e30: 56 61 6c 75 65 29 3b 0a 20 20 20 20 7d 65 6c 73 Value);. }els
2e40: 65 7b 0a 20 20 20 20 20 20 69 66 28 20 2a 7a 20 e{. if( *z
2e50: 29 7b 20 2a 7a 2b 2b 20 3d 20 30 3b 20 7d 0a 20 ){ *z++ = 0; }.
2e60: 20 20 20 20 20 7a 56 61 6c 75 65 20 3d 20 22 22 zValue = ""
2e70: 3b 0a 20 20 20 20 7d 0a 20 20 20 20 69 66 28 20 ;. }. if(
2e80: 69 73 6c 6f 77 65 72 28 7a 4e 61 6d 65 5b 30 5d islower(zName[0]
2e90: 29 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 ) ){. cgi_s
2ea0: 65 74 5f 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 et_parameter_noc
2eb0: 6f 70 79 28 7a 4e 61 6d 65 2c 20 7a 56 61 6c 75 opy(zName, zValu
2ec0: 65 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 7d 0a e);. }. }.}.
2ed0: 0a 2f 2a 0a 2a 2a 20 2a 70 7a 20 69 73 20 61 20 ./*.** *pz is a
2ee0: 73 74 72 69 6e 67 20 74 68 61 74 20 63 6f 6e 73 string that cons
2ef0: 69 73 74 73 20 6f 66 20 6d 75 6c 74 69 70 6c 65 ists of multiple
2f00: 20 6c 69 6e 65 73 20 6f 66 20 74 65 78 74 2e 20 lines of text.
2f10: 20 54 68 69 73 0a 2a 2a 20 72 6f 75 74 69 6e 65 This.** routine
2f20: 20 66 69 6e 64 73 20 74 68 65 20 65 6e 64 20 6f finds the end o
2f30: 66 20 74 68 65 20 63 75 72 72 65 6e 74 20 6c 69 f the current li
2f40: 6e 65 20 6f 66 20 74 65 78 74 20 61 6e 64 20 63 ne of text and c
2f50: 6f 6e 76 65 72 74 73 0a 2a 2a 20 74 68 65 20 22 onverts.** the "
2f60: 5c 6e 22 20 6f 72 20 22 5c 72 5c 6e 22 20 74 68 \n" or "\r\n" th
2f70: 61 74 20 65 6e 64 73 20 74 68 61 74 20 6c 69 6e at ends that lin
2f80: 65 20 69 6e 74 6f 20 61 20 22 5c 30 30 30 22 2e e into a "\000".
2f90: 20 20 49 74 20 74 68 65 6e 0a 2a 2a 20 61 64 76 It then.** adv
2fa0: 61 6e 63 65 73 20 2a 70 7a 20 74 6f 20 74 68 65 ances *pz to the
2fb0: 20 62 65 67 69 6e 6e 69 6e 67 20 6f 66 20 74 68 beginning of th
2fc0: 65 20 6e 65 78 74 20 6c 69 6e 65 20 61 6e 64 20 e next line and
2fd0: 72 65 74 75 72 6e 73 20 74 68 65 0a 2a 2a 20 70 returns the.** p
2fe0: 72 65 76 69 6f 75 73 20 76 61 6c 75 65 20 6f 66 revious value of
2ff0: 20 2a 70 7a 20 28 77 68 69 63 68 20 69 73 20 74 *pz (which is t
3000: 68 65 20 73 74 61 72 74 20 6f 66 20 74 68 65 20 he start of the
3010: 63 75 72 72 65 6e 74 20 6c 69 6e 65 2e 29 0a 2a current line.).*
3020: 2f 0a 73 74 61 74 69 63 20 63 68 61 72 20 2a 67 /.static char *g
3030: 65 74 5f 6c 69 6e 65 5f 66 72 6f 6d 5f 73 74 72 et_line_from_str
3040: 69 6e 67 28 63 68 61 72 20 2a 2a 70 7a 2c 20 69 ing(char **pz, i
3050: 6e 74 20 2a 70 4c 65 6e 29 7b 0a 20 20 63 68 61 nt *pLen){. cha
3060: 72 20 2a 7a 20 3d 20 2a 70 7a 3b 0a 20 20 69 6e r *z = *pz;. in
3070: 74 20 69 3b 0a 20 20 69 66 28 20 7a 5b 30 5d 3d t i;. if( z[0]=
3080: 3d 30 20 29 20 72 65 74 75 72 6e 20 30 3b 0a 20 =0 ) return 0;.
3090: 20 66 6f 72 28 69 3d 30 3b 20 7a 5b 69 5d 3b 20 for(i=0; z[i];
30a0: 69 2b 2b 29 7b 0a 20 20 20 20 69 66 28 20 7a 5b i++){. if( z[
30b0: 69 5d 3d 3d 27 5c 6e 27 20 29 7b 0a 20 20 20 20 i]=='\n' ){.
30c0: 20 20 69 66 28 20 69 3e 30 20 26 26 20 7a 5b 69 if( i>0 && z[i
30d0: 2d 31 5d 3d 3d 27 5c 72 27 20 29 7b 0a 20 20 20 -1]=='\r' ){.
30e0: 20 20 20 20 20 7a 5b 69 2d 31 5d 20 3d 20 30 3b z[i-1] = 0;
30f0: 0a 20 20 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 . }else{.
3100: 20 20 20 20 20 20 7a 5b 69 5d 20 3d 20 30 3b 0a z[i] = 0;.
3110: 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 69 2b }. i+
3120: 2b 3b 0a 20 20 20 20 20 20 62 72 65 61 6b 3b 0a +;. break;.
3130: 20 20 20 20 7d 0a 20 20 7d 0a 20 20 2a 70 7a 20 }. }. *pz
3140: 3d 20 26 7a 5b 69 5d 3b 0a 20 20 2a 70 4c 65 6e = &z[i];. *pLen
3150: 20 2d 3d 20 69 3b 0a 20 20 72 65 74 75 72 6e 20 -= i;. return
3160: 7a 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 65 20 z;.}../*.** The
3170: 69 6e 70 75 74 20 2a 70 7a 20 70 6f 69 6e 74 73 input *pz points
3180: 20 74 6f 20 63 6f 6e 74 65 6e 74 20 74 68 61 74 to content that
3190: 20 69 73 20 74 65 72 6d 69 6e 61 74 65 64 20 62 is terminated b
31a0: 79 20 61 20 22 5c 72 5c 6e 22 0a 2a 2a 20 66 6f y a "\r\n".** fo
31b0: 6c 6c 6f 77 65 64 20 62 79 20 74 68 65 20 62 6f llowed by the bo
31c0: 75 6e 64 72 79 20 6d 61 72 6b 65 72 20 7a 42 6f undry marker zBo
31d0: 75 6e 64 72 79 2e 20 20 41 6e 20 65 78 74 72 61 undry. An extra
31e0: 20 22 2d 2d 22 20 6d 61 79 20 6f 72 0a 2a 2a 20 "--" may or.**
31f0: 6d 61 79 20 6e 6f 74 20 62 65 20 61 70 70 65 6e may not be appen
3200: 64 65 64 20 74 6f 20 74 68 65 20 62 6f 75 6e 64 ded to the bound
3210: 72 79 20 6d 61 72 6b 65 72 2e 20 20 54 68 65 72 ry marker. Ther
3220: 65 20 61 72 65 20 2a 70 4c 65 6e 20 63 68 61 72 e are *pLen char
3230: 61 63 74 65 72 73 0a 2a 2a 20 69 6e 20 2a 70 7a acters.** in *pz
3240: 2e 0a 2a 2a 0a 2a 2a 20 54 68 69 73 20 72 6f 75 ..**.** This rou
3250: 74 69 6e 65 20 61 64 64 73 20 61 20 22 5c 30 30 tine adds a "\00
3260: 30 22 20 74 6f 20 74 68 65 20 65 6e 64 20 6f 66 0" to the end of
3270: 20 74 68 65 20 63 6f 6e 74 65 6e 74 20 28 6f 76 the content (ov
3280: 65 72 77 72 69 74 69 6e 67 0a 2a 2a 20 74 68 65 erwriting.** the
3290: 20 22 5c 72 5c 6e 22 29 20 61 6e 64 20 72 65 74 "\r\n") and ret
32a0: 75 72 6e 73 20 61 20 70 6f 69 6e 74 65 72 20 74 urns a pointer t
32b0: 6f 20 74 68 65 20 63 6f 6e 74 65 6e 74 2e 20 20 o the content.
32c0: 54 68 65 20 2a 70 7a 20 69 6e 70 75 74 0a 2a 2a The *pz input.**
32d0: 20 69 73 20 61 64 6a 75 73 74 65 64 20 74 6f 20 is adjusted to
32e0: 70 6f 69 6e 74 20 74 6f 20 74 68 65 20 66 69 72 point to the fir
32f0: 73 74 20 6c 69 6e 65 20 66 6f 6c 6c 6f 77 69 6e st line followin
3300: 67 20 74 68 65 20 62 6f 75 6e 64 72 79 2e 0a 2a g the boundry..*
3310: 2a 20 54 68 65 20 6c 65 6e 67 74 68 20 6f 66 20 * The length of
3320: 74 68 65 20 63 6f 6e 74 65 6e 74 20 69 73 20 73 the content is s
3330: 74 6f 72 65 64 20 69 6e 20 2a 70 6e 43 6f 6e 74 tored in *pnCont
3340: 65 6e 74 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 63 ent..*/.static c
3350: 68 61 72 20 2a 67 65 74 5f 62 6f 75 6e 64 65 64 har *get_bounded
3360: 5f 63 6f 6e 74 65 6e 74 28 0a 20 20 63 68 61 72 _content(. char
3370: 20 2a 2a 70 7a 2c 20 20 20 20 20 20 20 20 20 2f **pz, /
3380: 2a 20 43 6f 6e 74 65 6e 74 20 74 61 6b 65 6e 20 * Content taken
3390: 66 72 6f 6d 20 68 65 72 65 20 2a 2f 0a 20 20 69 from here */. i
33a0: 6e 74 20 2a 70 4c 65 6e 2c 20 20 20 20 20 20 20 nt *pLen,
33b0: 20 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 62 /* Number of b
33c0: 79 74 65 73 20 6f 66 20 64 61 74 61 20 69 6e 20 ytes of data in
33d0: 28 2a 70 7a 29 5b 5d 20 2a 2f 0a 20 20 63 68 61 (*pz)[] */. cha
33e0: 72 20 2a 7a 42 6f 75 6e 64 72 79 2c 20 20 20 20 r *zBoundry,
33f0: 2f 2a 20 42 6f 75 6e 64 72 79 20 74 65 78 74 20 /* Boundry text
3400: 6d 61 72 6b 69 6e 67 20 74 68 65 20 65 6e 64 20 marking the end
3410: 6f 66 20 63 6f 6e 74 65 6e 74 20 2a 2f 0a 20 20 of content */.
3420: 69 6e 74 20 2a 70 6e 43 6f 6e 74 65 6e 74 20 20 int *pnContent
3430: 20 20 20 2f 2a 20 57 72 69 74 65 20 74 68 65 20 /* Write the
3440: 73 69 7a 65 20 6f 66 20 74 68 65 20 63 6f 6e 74 size of the cont
3450: 65 6e 74 20 68 65 72 65 20 2a 2f 0a 29 7b 0a 20 ent here */.){.
3460: 20 63 68 61 72 20 2a 7a 20 3d 20 2a 70 7a 3b 0a char *z = *pz;.
3470: 20 20 69 6e 74 20 6c 65 6e 20 3d 20 2a 70 4c 65 int len = *pLe
3480: 6e 3b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 69 6e n;. int i;. in
3490: 74 20 6e 42 6f 75 6e 64 72 79 20 3d 20 73 74 72 t nBoundry = str
34a0: 6c 65 6e 28 7a 42 6f 75 6e 64 72 79 29 3b 0a 20 len(zBoundry);.
34b0: 20 2a 70 6e 43 6f 6e 74 65 6e 74 20 3d 20 6c 65 *pnContent = le
34c0: 6e 3b 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69 3c n;. for(i=0; i<
34d0: 6c 65 6e 3b 20 69 2b 2b 29 7b 0a 20 20 20 20 69 len; i++){. i
34e0: 66 28 20 7a 5b 69 5d 3d 3d 27 5c 6e 27 20 26 26 f( z[i]=='\n' &&
34f0: 20 73 74 72 6e 63 6d 70 28 7a 42 6f 75 6e 64 72 strncmp(zBoundr
3500: 79 2c 20 26 7a 5b 69 2b 31 5d 2c 20 6e 42 6f 75 y, &z[i+1], nBou
3510: 6e 64 72 79 29 3d 3d 30 20 29 7b 0a 20 20 20 20 ndry)==0 ){.
3520: 20 20 69 66 28 20 69 3e 30 20 26 26 20 7a 5b 69 if( i>0 && z[i
3530: 2d 31 5d 3d 3d 27 5c 72 27 20 29 20 69 2d 2d 3b -1]=='\r' ) i--;
3540: 0a 20 20 20 20 20 20 7a 5b 69 5d 20 3d 20 30 3b . z[i] = 0;
3550: 0a 20 20 20 20 20 20 2a 70 6e 43 6f 6e 74 65 6e . *pnConten
3560: 74 20 3d 20 69 3b 0a 20 20 20 20 20 20 69 20 2b t = i;. i +
3570: 3d 20 6e 42 6f 75 6e 64 72 79 3b 0a 20 20 20 20 = nBoundry;.
3580: 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 7d 0a 20 break;. }.
3590: 20 7d 0a 20 20 2a 70 7a 20 3d 20 26 7a 5b 69 5d }. *pz = &z[i]
35a0: 3b 0a 20 20 67 65 74 5f 6c 69 6e 65 5f 66 72 6f ;. get_line_fro
35b0: 6d 5f 73 74 72 69 6e 67 28 70 7a 2c 20 70 4c 65 m_string(pz, pLe
35c0: 6e 29 3b 0a 20 20 72 65 74 75 72 6e 20 7a 3b 20 n);. return z;
35d0: 20 20 20 20 20 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 .}../*.** T
35e0: 6f 6b 65 6e 69 7a 65 20 61 20 6c 69 6e 65 20 6f okenize a line o
35f0: 66 20 74 65 78 74 20 69 6e 74 6f 20 61 73 20 6d f text into as m
3600: 61 6e 79 20 61 73 20 6e 41 72 67 20 74 6f 6b 65 any as nArg toke
3610: 6e 73 2e 20 20 4d 61 6b 65 0a 2a 2a 20 61 7a 41 ns. Make.** azA
3620: 72 67 5b 5d 20 70 6f 69 6e 74 20 74 6f 20 74 68 rg[] point to th
3630: 65 20 73 74 61 72 74 20 6f 66 20 65 61 63 68 20 e start of each
3640: 74 6f 6b 65 6e 2e 0a 2a 2a 0a 2a 2a 20 54 6f 6b token..**.** Tok
3650: 65 6e 73 20 63 6f 6e 73 69 73 74 20 6f 66 20 73 ens consist of s
3660: 70 61 63 65 20 6f 72 20 73 65 6d 69 2d 63 6f 6c pace or semi-col
3670: 6f 6e 20 64 65 6c 69 6d 69 74 65 64 20 77 6f 72 on delimited wor
3680: 64 73 20 6f 72 0a 2a 2a 20 73 74 72 69 6e 67 73 ds or.** strings
3690: 20 69 6e 73 69 64 65 20 64 6f 75 62 6c 65 2d 71 inside double-q
36a0: 75 6f 74 65 73 2e 20 20 45 78 61 6d 70 6c 65 3a uotes. Example:
36b0: 0a 2a 2a 0a 2a 2a 20 20 20 20 63 6f 6e 74 65 6e .**.** conten
36c0: 74 2d 64 69 73 70 6f 73 69 74 69 6f 6e 3a 20 66 t-disposition: f
36d0: 6f 72 6d 2d 64 61 74 61 3b 20 6e 61 6d 65 3d 22 orm-data; name="
36e0: 66 6e 22 3b 20 66 69 6c 65 6e 61 6d 65 3d 22 69 fn"; filename="i
36f0: 6e 64 65 78 2e 68 74 6d 6c 22 0a 2a 2a 0a 2a 2a ndex.html".**.**
3700: 20 54 68 65 20 6c 69 6e 65 20 61 62 6f 76 65 20 The line above
3710: 69 73 20 74 6f 6b 65 6e 69 7a 65 64 20 61 73 20 is tokenized as
3720: 66 6f 6c 6c 6f 77 73 3a 0a 2a 2a 0a 2a 2a 20 20 follows:.**.**
3730: 20 20 61 7a 41 72 67 5b 30 5d 20 3d 20 22 63 6f azArg[0] = "co
3740: 6e 74 65 6e 74 2d 64 69 73 70 6f 73 69 74 69 6f ntent-dispositio
3750: 6e 3a 22 0a 2a 2a 20 20 20 20 61 7a 41 72 67 5b n:".** azArg[
3760: 31 5d 20 3d 20 22 66 6f 72 6d 2d 64 61 74 61 22 1] = "form-data"
3770: 0a 2a 2a 20 20 20 20 61 7a 41 72 67 5b 32 5d 20 .** azArg[2]
3780: 3d 20 22 6e 61 6d 65 3d 22 0a 2a 2a 20 20 20 20 = "name=".**
3790: 61 7a 41 72 67 5b 33 5d 20 3d 20 22 66 6e 22 0a azArg[3] = "fn".
37a0: 2a 2a 20 20 20 20 61 7a 41 72 67 5b 34 5d 20 3d ** azArg[4] =
37b0: 20 22 66 69 6c 65 6e 61 6d 65 3d 22 0a 2a 2a 20 "filename=".**
37c0: 20 20 20 61 7a 41 72 67 5b 35 5d 20 3d 20 22 69 azArg[5] = "i
37d0: 6e 64 65 78 2e 68 74 6d 6c 22 0a 2a 2a 20 20 20 ndex.html".**
37e0: 20 61 7a 41 72 67 5b 36 5d 20 3d 20 30 3b 0a 2a azArg[6] = 0;.*
37f0: 2a 0a 2a 2a 20 27 5c 30 30 30 27 20 63 68 61 72 *.** '\000' char
3800: 61 63 74 65 72 73 20 61 72 65 20 69 6e 73 65 72 acters are inser
3810: 74 65 64 20 69 6e 20 7a 5b 5d 20 61 74 20 74 68 ted in z[] at th
3820: 65 20 65 6e 64 20 6f 66 20 65 61 63 68 20 74 6f e end of each to
3830: 6b 65 6e 2e 0a 2a 2a 20 54 68 69 73 20 72 6f 75 ken..** This rou
3840: 74 69 6e 65 20 72 65 74 75 72 6e 73 20 74 68 65 tine returns the
3850: 20 74 6f 74 61 6c 20 6e 75 6d 62 65 72 20 6f 66 total number of
3860: 20 74 6f 6b 65 6e 73 20 6f 6e 20 74 68 65 20 6c tokens on the l
3870: 69 6e 65 2c 20 36 0a 2a 2a 20 69 6e 20 74 68 65 ine, 6.** in the
3880: 20 65 78 61 6d 70 6c 65 20 61 62 6f 76 65 2e 0a example above..
3890: 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 74 6f */.static int to
38a0: 6b 65 6e 69 7a 65 5f 6c 69 6e 65 28 63 68 61 72 kenize_line(char
38b0: 20 2a 7a 2c 20 69 6e 74 20 6d 78 41 72 67 2c 20 *z, int mxArg,
38c0: 63 68 61 72 20 2a 2a 61 7a 41 72 67 29 7b 0a 20 char **azArg){.
38d0: 20 69 6e 74 20 69 20 3d 20 30 3b 0a 20 20 77 68 int i = 0;. wh
38e0: 69 6c 65 28 20 2a 7a 20 29 7b 0a 20 20 20 20 77 ile( *z ){. w
38f0: 68 69 6c 65 28 20 69 73 73 70 61 63 65 28 2a 7a hile( isspace(*z
3900: 29 20 7c 7c 20 2a 7a 3d 3d 27 3b 27 20 29 7b 20 ) || *z==';' ){
3910: 7a 2b 2b 3b 20 7d 0a 20 20 20 20 69 66 28 20 2a z++; }. if( *
3920: 7a 3d 3d 27 22 27 20 26 26 20 7a 5b 31 5d 20 29 z=='"' && z[1] )
3930: 7b 0a 20 20 20 20 20 20 2a 7a 20 3d 20 30 3b 0a {. *z = 0;.
3940: 20 20 20 20 20 20 7a 2b 2b 3b 0a 20 20 20 20 20 z++;.
3950: 20 69 66 28 20 69 3c 6d 78 41 72 67 2d 31 20 29 if( i<mxArg-1 )
3960: 7b 20 61 7a 41 72 67 5b 69 2b 2b 5d 20 3d 20 7a { azArg[i++] = z
3970: 3b 20 7d 0a 20 20 20 20 20 20 77 68 69 6c 65 28 ; }. while(
3980: 20 2a 7a 20 26 26 20 2a 7a 21 3d 27 22 27 20 29 *z && *z!='"' )
3990: 7b 20 7a 2b 2b 3b 20 7d 0a 20 20 20 20 20 20 69 { z++; }. i
39a0: 66 28 20 2a 7a 3d 3d 30 20 29 20 62 72 65 61 6b f( *z==0 ) break
39b0: 3b 0a 20 20 20 20 20 20 2a 7a 20 3d 20 30 3b 0a ;. *z = 0;.
39c0: 20 20 20 20 20 20 7a 2b 2b 3b 0a 20 20 20 20 7d z++;. }
39d0: 65 6c 73 65 7b 0a 20 20 20 20 20 20 69 66 28 20 else{. if(
39e0: 69 3c 6d 78 41 72 67 2d 31 20 29 7b 20 61 7a 41 i<mxArg-1 ){ azA
39f0: 72 67 5b 69 2b 2b 5d 20 3d 20 7a 3b 20 7d 0a 20 rg[i++] = z; }.
3a00: 20 20 20 20 20 77 68 69 6c 65 28 20 2a 7a 20 26 while( *z &
3a10: 26 20 21 69 73 73 70 61 63 65 28 2a 7a 29 20 26 & !isspace(*z) &
3a20: 26 20 2a 7a 21 3d 27 3b 27 20 26 26 20 2a 7a 21 & *z!=';' && *z!
3a30: 3d 27 22 27 20 29 7b 20 7a 2b 2b 3b 20 7d 0a 20 ='"' ){ z++; }.
3a40: 20 20 20 20 20 69 66 28 20 2a 7a 20 26 26 20 2a if( *z && *
3a50: 7a 21 3d 27 22 27 20 29 7b 0a 20 20 20 20 20 20 z!='"' ){.
3a60: 20 20 2a 7a 20 3d 20 30 3b 0a 20 20 20 20 20 20 *z = 0;.
3a70: 20 20 7a 2b 2b 3b 0a 20 20 20 20 20 20 7d 0a 20 z++;. }.
3a80: 20 20 20 7d 0a 20 20 7d 0a 20 20 61 7a 41 72 67 }. }. azArg
3a90: 5b 69 5d 20 3d 20 30 3b 0a 20 20 72 65 74 75 72 [i] = 0;. retur
3aa0: 6e 20 69 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 63 n i;.}../*.** Sc
3ab0: 61 6e 20 74 68 65 20 6d 75 6c 74 69 70 61 72 74 an the multipart
3ac0: 2d 66 6f 72 6d 20 63 6f 6e 74 65 6e 74 20 61 6e -form content an
3ad0: 64 20 6d 61 6b 65 20 61 70 70 72 6f 70 72 69 61 d make appropria
3ae0: 74 65 20 65 6e 74 72 69 65 73 0a 2a 2a 20 69 6e te entries.** in
3af0: 74 6f 20 74 68 65 20 70 61 72 61 6d 65 74 65 72 to the parameter
3b00: 20 74 61 62 6c 65 2e 0a 2a 2a 0a 2a 2a 20 54 68 table..**.** Th
3b10: 65 20 63 6f 6e 74 65 6e 74 20 73 74 72 69 6e 67 e content string
3b20: 20 22 7a 22 20 69 73 20 6d 6f 64 69 66 69 65 64 "z" is modified
3b30: 20 62 79 20 74 68 69 73 20 72 6f 75 74 69 6e 65 by this routine
3b40: 20 62 75 74 20 69 74 20 69 73 0a 2a 2a 20 6e 6f but it is.** no
3b50: 74 20 63 6f 70 69 65 64 2e 20 20 54 68 65 20 63 t copied. The c
3b60: 61 6c 6c 69 6e 67 20 66 75 6e 63 74 69 6f 6e 20 alling function
3b70: 6d 75 73 74 20 6e 6f 74 20 64 65 61 6c 6c 6f 63 must not dealloc
3b80: 61 74 65 20 6f 72 20 6d 6f 64 69 66 79 0a 2a 2a ate or modify.**
3b90: 20 22 7a 22 20 61 66 74 65 72 20 74 68 69 73 20 "z" after this
3ba0: 72 6f 75 74 69 6e 65 20 66 69 6e 69 73 68 65 73 routine finishes
3bb0: 20 6f 72 20 69 74 20 63 6f 75 6c 64 20 63 6f 72 or it could cor
3bc0: 72 75 70 74 20 74 68 65 20 70 61 72 61 6d 65 74 rupt the paramet
3bd0: 65 72 0a 2a 2a 20 74 61 62 6c 65 2e 0a 2a 2f 0a er.** table..*/.
3be0: 73 74 61 74 69 63 20 76 6f 69 64 20 70 72 6f 63 static void proc
3bf0: 65 73 73 5f 6d 75 6c 74 69 70 61 72 74 5f 66 6f ess_multipart_fo
3c00: 72 6d 5f 64 61 74 61 28 63 68 61 72 20 2a 7a 2c rm_data(char *z,
3c10: 20 69 6e 74 20 6c 65 6e 29 7b 0a 20 20 63 68 61 int len){. cha
3c20: 72 20 2a 7a 4c 69 6e 65 3b 0a 20 20 69 6e 74 20 r *zLine;. int
3c30: 6e 41 72 67 2c 20 69 3b 0a 20 20 63 68 61 72 20 nArg, i;. char
3c40: 2a 7a 42 6f 75 6e 64 72 79 3b 0a 20 20 63 68 61 *zBoundry;. cha
3c50: 72 20 2a 7a 56 61 6c 75 65 3b 0a 20 20 63 68 61 r *zValue;. cha
3c60: 72 20 2a 7a 4e 61 6d 65 20 3d 20 30 3b 0a 20 20 r *zName = 0;.
3c70: 69 6e 74 20 73 68 6f 77 42 79 74 65 73 20 3d 20 int showBytes =
3c80: 30 3b 0a 20 20 63 68 61 72 20 2a 61 7a 41 72 67 0;. char *azArg
3c90: 5b 35 30 5d 3b 0a 0a 20 20 7a 42 6f 75 6e 64 72 [50];.. zBoundr
3ca0: 79 20 3d 20 67 65 74 5f 6c 69 6e 65 5f 66 72 6f y = get_line_fro
3cb0: 6d 5f 73 74 72 69 6e 67 28 26 7a 2c 20 26 6c 65 m_string(&z, &le
3cc0: 6e 29 3b 0a 20 20 69 66 28 20 7a 42 6f 75 6e 64 n);. if( zBound
3cd0: 72 79 3d 3d 30 20 29 20 72 65 74 75 72 6e 3b 0a ry==0 ) return;.
3ce0: 20 20 77 68 69 6c 65 28 20 28 7a 4c 69 6e 65 20 while( (zLine
3cf0: 3d 20 67 65 74 5f 6c 69 6e 65 5f 66 72 6f 6d 5f = get_line_from_
3d00: 73 74 72 69 6e 67 28 26 7a 2c 20 26 6c 65 6e 29 string(&z, &len)
3d10: 29 21 3d 30 20 29 7b 0a 20 20 20 20 69 66 28 20 )!=0 ){. if(
3d20: 7a 4c 69 6e 65 5b 30 5d 3d 3d 30 20 29 7b 0a 20 zLine[0]==0 ){.
3d30: 20 20 20 20 20 69 6e 74 20 6e 43 6f 6e 74 65 6e int nConten
3d40: 74 20 3d 20 30 3b 0a 20 20 20 20 20 20 7a 56 61 t = 0;. zVa
3d50: 6c 75 65 20 3d 20 67 65 74 5f 62 6f 75 6e 64 65 lue = get_bounde
3d60: 64 5f 63 6f 6e 74 65 6e 74 28 26 7a 2c 20 26 6c d_content(&z, &l
3d70: 65 6e 2c 20 7a 42 6f 75 6e 64 72 79 2c 20 26 6e en, zBoundry, &n
3d80: 43 6f 6e 74 65 6e 74 29 3b 0a 20 20 20 20 20 20 Content);.
3d90: 69 66 28 20 7a 4e 61 6d 65 20 26 26 20 7a 56 61 if( zName && zVa
3da0: 6c 75 65 20 26 26 20 69 73 6c 6f 77 65 72 28 7a lue && islower(z
3db0: 4e 61 6d 65 5b 30 5d 29 20 29 7b 0a 20 20 20 20 Name[0]) ){.
3dc0: 20 20 20 20 63 67 69 5f 73 65 74 5f 70 61 72 61 cgi_set_para
3dd0: 6d 65 74 65 72 5f 6e 6f 63 6f 70 79 28 7a 4e 61 meter_nocopy(zNa
3de0: 6d 65 2c 20 7a 56 61 6c 75 65 29 3b 0a 20 20 20 me, zValue);.
3df0: 20 20 20 20 20 69 66 28 20 73 68 6f 77 42 79 74 if( showByt
3e00: 65 73 20 29 7b 0a 20 20 20 20 20 20 20 20 20 20 es ){.
3e10: 63 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 74 65 cgi_set_paramete
3e20: 72 5f 6e 6f 63 6f 70 79 28 6d 70 72 69 6e 74 66 r_nocopy(mprintf
3e30: 28 22 25 73 3a 62 79 74 65 73 22 2c 20 7a 4e 61 ("%s:bytes", zNa
3e40: 6d 65 29 2c 0a 20 20 20 20 20 20 20 20 20 20 20 me),.
3e50: 20 20 20 20 6d 70 72 69 6e 74 66 28 22 25 64 22 mprintf("%d"
3e60: 2c 6e 43 6f 6e 74 65 6e 74 29 29 3b 0a 20 20 20 ,nContent));.
3e70: 20 20 20 20 20 7d 0a 20 20 20 20 20 20 7d 0a 20 }. }.
3e80: 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 30 3b 0a zName = 0;.
3e90: 20 20 20 20 20 20 73 68 6f 77 42 79 74 65 73 20 showBytes
3ea0: 3d 20 30 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a = 0;. }else{.
3eb0: 20 20 20 20 20 20 6e 41 72 67 20 3d 20 74 6f 6b nArg = tok
3ec0: 65 6e 69 7a 65 5f 6c 69 6e 65 28 7a 4c 69 6e 65 enize_line(zLine
3ed0: 2c 20 73 69 7a 65 6f 66 28 61 7a 41 72 67 29 2f , sizeof(azArg)/
3ee0: 73 69 7a 65 6f 66 28 61 7a 41 72 67 5b 30 5d 29 sizeof(azArg[0])
3ef0: 2c 20 61 7a 41 72 67 29 3b 0a 20 20 20 20 20 20 , azArg);.
3f00: 66 6f 72 28 69 3d 30 3b 20 69 3c 6e 41 72 67 3b for(i=0; i<nArg;
3f10: 20 69 2b 2b 29 7b 0a 20 20 20 20 20 20 20 20 69 i++){. i
3f20: 6e 74 20 63 20 3d 20 74 6f 6c 6f 77 65 72 28 61 nt c = tolower(a
3f30: 7a 41 72 67 5b 69 5d 5b 30 5d 29 3b 0a 20 20 20 zArg[i][0]);.
3f40: 20 20 20 20 20 69 66 28 20 63 3d 3d 27 63 27 20 if( c=='c'
3f50: 26 26 20 73 74 72 69 63 6d 70 28 61 7a 41 72 67 && stricmp(azArg
3f60: 5b 69 5d 2c 22 63 6f 6e 74 65 6e 74 2d 64 69 73 [i],"content-dis
3f70: 70 6f 73 69 74 69 6f 6e 3a 22 29 3d 3d 30 20 29 position:")==0 )
3f80: 7b 0a 20 20 20 20 20 20 20 20 20 20 69 2b 2b 3b {. i++;
3f90: 0a 20 20 20 20 20 20 20 20 7d 65 6c 73 65 20 69 . }else i
3fa0: 66 28 20 63 3d 3d 27 6e 27 20 26 26 20 73 74 72 f( c=='n' && str
3fb0: 69 63 6d 70 28 61 7a 41 72 67 5b 69 5d 2c 22 6e icmp(azArg[i],"n
3fc0: 61 6d 65 3d 22 29 3d 3d 30 20 29 7b 0a 20 20 20 ame=")==0 ){.
3fd0: 20 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 61 zName = a
3fe0: 7a 41 72 67 5b 2b 2b 69 5d 3b 0a 20 20 20 20 20 zArg[++i];.
3ff0: 20 20 20 7d 65 6c 73 65 20 69 66 28 20 63 3d 3d }else if( c==
4000: 27 66 27 20 26 26 20 73 74 72 69 63 6d 70 28 61 'f' && stricmp(a
4010: 7a 41 72 67 5b 69 5d 2c 22 66 69 6c 65 6e 61 6d zArg[i],"filenam
4020: 65 3d 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 e=")==0 ){.
4030: 20 20 20 20 20 63 68 61 72 20 2a 7a 20 3d 20 61 char *z = a
4040: 7a 41 72 67 5b 2b 2b 69 5d 3b 0a 20 20 20 20 20 zArg[++i];.
4050: 20 20 20 20 20 69 66 28 20 7a 4e 61 6d 65 20 26 if( zName &
4060: 26 20 7a 20 26 26 20 69 73 6c 6f 77 65 72 28 7a & z && islower(z
4070: 4e 61 6d 65 5b 30 5d 29 20 29 7b 0a 20 20 20 20 Name[0]) ){.
4080: 20 20 20 20 20 20 20 20 63 67 69 5f 73 65 74 5f cgi_set_
4090: 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79 parameter_nocopy
40a0: 28 6d 70 72 69 6e 74 66 28 22 25 73 3a 66 69 6c (mprintf("%s:fil
40b0: 65 6e 61 6d 65 22 2c 7a 4e 61 6d 65 29 2c 20 7a ename",zName), z
40c0: 29 3b 0a 20 20 20 20 20 20 20 20 20 20 7d 0a 20 );. }.
40d0: 20 20 20 20 20 20 20 20 20 73 68 6f 77 42 79 74 showByt
40e0: 65 73 20 3d 20 31 3b 0a 20 20 20 20 20 20 20 20 es = 1;.
40f0: 7d 65 6c 73 65 20 69 66 28 20 63 3d 3d 27 63 27 }else if( c=='c'
4100: 20 26 26 20 73 74 72 69 63 6d 70 28 61 7a 41 72 && stricmp(azAr
4110: 67 5b 69 5d 2c 22 63 6f 6e 74 65 6e 74 2d 74 79 g[i],"content-ty
4120: 70 65 3a 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 pe:")==0 ){.
4130: 20 20 20 20 20 20 63 68 61 72 20 2a 7a 20 3d 20 char *z =
4140: 61 7a 41 72 67 5b 2b 2b 69 5d 3b 0a 20 20 20 20 azArg[++i];.
4150: 20 20 20 20 20 20 69 66 28 20 7a 4e 61 6d 65 20 if( zName
4160: 26 26 20 7a 20 26 26 20 69 73 6c 6f 77 65 72 28 && z && islower(
4170: 7a 4e 61 6d 65 5b 30 5d 29 20 29 7b 0a 20 20 20 zName[0]) ){.
4180: 20 20 20 20 20 20 20 20 20 63 67 69 5f 73 65 74 cgi_set
4190: 5f 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 _parameter_nocop
41a0: 79 28 6d 70 72 69 6e 74 66 28 22 25 73 3a 6d 69 y(mprintf("%s:mi
41b0: 6d 65 74 79 70 65 22 2c 7a 4e 61 6d 65 29 2c 20 metype",zName),
41c0: 7a 29 3b 0a 20 20 20 20 20 20 20 20 20 20 7d 0a z);. }.
41d0: 20 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 }.
41e0: 7d 0a 20 20 20 20 7d 0a 20 20 7d 20 20 20 20 20 }. }. }
41f0: 20 20 20 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 49 6e 69 .}../*.** Ini
4200: 74 69 61 6c 69 7a 65 20 74 68 65 20 71 75 65 72 tialize the quer
4210: 79 20 70 61 72 61 6d 65 74 65 72 20 64 61 74 61 y parameter data
4220: 62 61 73 65 2e 20 20 49 6e 66 6f 72 6d 61 74 69 base. Informati
4230: 6f 6e 20 69 73 20 70 75 6c 6c 65 64 20 66 72 6f on is pulled fro
4240: 6d 0a 2a 2a 20 74 68 65 20 51 55 45 52 59 5f 53 m.** the QUERY_S
4250: 54 52 49 4e 47 20 65 6e 76 69 72 6f 6e 6d 65 6e TRING environmen
4260: 74 20 76 61 72 69 61 62 6c 65 20 28 69 66 20 69 t variable (if i
4270: 74 20 65 78 69 73 74 73 29 2c 20 66 72 6f 6d 20 t exists), from
4280: 73 74 61 6e 64 61 72 64 0a 2a 2a 20 69 6e 70 75 standard.** inpu
4290: 74 20 69 66 20 74 68 65 72 65 20 69 73 20 50 4f t if there is PO
42a0: 53 54 20 64 61 74 61 2c 20 61 6e 64 20 66 72 6f ST data, and fro
42b0: 6d 20 48 54 54 50 5f 43 4f 4f 4b 49 45 2e 0a 2a m HTTP_COOKIE..*
42c0: 2f 0a 76 6f 69 64 20 63 67 69 5f 69 6e 69 74 28 /.void cgi_init(
42d0: 76 6f 69 64 29 7b 0a 20 20 63 68 61 72 20 2a 7a void){. char *z
42e0: 3b 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a ;. const char *
42f0: 7a 54 79 70 65 3b 0a 20 20 69 6e 74 20 6c 65 6e zType;. int len
4300: 3b 0a 20 20 7a 20 3d 20 28 63 68 61 72 2a 29 50 ;. z = (char*)P
4310: 28 22 51 55 45 52 59 5f 53 54 52 49 4e 47 22 29 ("QUERY_STRING")
4320: 3b 0a 20 20 69 66 28 20 7a 20 29 7b 0a 20 20 20 ;. if( z ){.
4330: 20 7a 20 3d 20 6d 70 72 69 6e 74 66 28 22 25 73 z = mprintf("%s
4340: 22 2c 7a 29 3b 0a 20 20 20 20 61 64 64 5f 70 61 ",z);. add_pa
4350: 72 61 6d 5f 6c 69 73 74 28 7a 2c 20 27 26 27 29 ram_list(z, '&')
4360: 3b 0a 20 20 7d 0a 0a 20 20 6c 65 6e 20 3d 20 61 ;. }.. len = a
4370: 74 6f 69 28 50 44 28 22 43 4f 4e 54 45 4e 54 5f toi(PD("CONTENT_
4380: 4c 45 4e 47 54 48 22 2c 20 22 30 22 29 29 3b 0a LENGTH", "0"));.
4390: 20 20 67 2e 7a 43 6f 6e 74 65 6e 74 54 79 70 65 g.zContentType
43a0: 20 3d 20 7a 54 79 70 65 20 3d 20 50 28 22 43 4f = zType = P("CO
43b0: 4e 54 45 4e 54 5f 54 59 50 45 22 29 3b 0a 20 20 NTENT_TYPE");.
43c0: 69 66 28 20 6c 65 6e 3e 30 20 26 26 20 7a 54 79 if( len>0 && zTy
43d0: 70 65 20 29 7b 0a 20 20 20 20 62 6c 6f 62 5f 7a pe ){. blob_z
43e0: 65 72 6f 28 26 67 2e 63 67 69 49 6e 29 3b 0a 20 ero(&g.cgiIn);.
43f0: 20 20 20 69 66 28 20 73 74 72 63 6d 70 28 7a 54 if( strcmp(zT
4400: 79 70 65 2c 22 61 70 70 6c 69 63 61 74 69 6f 6e ype,"application
4410: 2f 78 2d 77 77 77 2d 66 6f 72 6d 2d 75 72 6c 65 /x-www-form-urle
4420: 6e 63 6f 64 65 64 22 29 3d 3d 30 20 0a 20 20 20 ncoded")==0 .
4430: 20 20 20 20 20 20 7c 7c 20 73 74 72 6e 63 6d 70 || strncmp
4440: 28 7a 54 79 70 65 2c 22 6d 75 6c 74 69 70 61 72 (zType,"multipar
4450: 74 2f 66 6f 72 6d 2d 64 61 74 61 22 2c 31 39 29 t/form-data",19)
4460: 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 7a 20 3d ==0 ){. z =
4470: 20 6d 61 6c 6c 6f 63 28 20 6c 65 6e 2b 31 20 29 malloc( len+1 )
4480: 3b 0a 20 20 20 20 20 20 69 66 28 20 7a 3d 3d 30 ;. if( z==0
4490: 20 29 20 65 78 69 74 28 31 29 3b 0a 20 20 20 20 ) exit(1);.
44a0: 20 20 6c 65 6e 20 3d 20 66 72 65 61 64 28 7a 2c len = fread(z,
44b0: 20 31 2c 20 6c 65 6e 2c 20 73 74 64 69 6e 29 3b 1, len, stdin);
44c0: 0a 20 20 20 20 20 20 7a 5b 6c 65 6e 5d 20 3d 20 . z[len] =
44d0: 30 3b 0a 20 20 20 20 20 20 69 66 28 20 7a 54 79 0;. if( zTy
44e0: 70 65 5b 30 5d 3d 3d 27 61 27 20 29 7b 0a 20 20 pe[0]=='a' ){.
44f0: 20 20 20 20 20 20 61 64 64 5f 70 61 72 61 6d 5f add_param_
4500: 6c 69 73 74 28 7a 2c 20 27 26 27 29 3b 0a 20 20 list(z, '&');.
4510: 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20 }else{.
4520: 20 20 20 70 72 6f 63 65 73 73 5f 6d 75 6c 74 69 process_multi
4530: 70 61 72 74 5f 66 6f 72 6d 5f 64 61 74 61 28 7a part_form_data(z
4540: 2c 20 6c 65 6e 29 3b 0a 20 20 20 20 20 20 7d 0a , len);. }.
4550: 20 20 20 20 7d 65 6c 73 65 20 69 66 28 20 73 74 }else if( st
4560: 72 63 6d 70 28 7a 54 79 70 65 2c 20 22 61 70 70 rcmp(zType, "app
4570: 6c 69 63 61 74 69 6f 6e 2f 78 2d 66 6f 73 73 69 lication/x-fossi
4580: 6c 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 l")==0 ){.
4590: 62 6c 6f 62 5f 72 65 61 64 5f 66 72 6f 6d 5f 63 blob_read_from_c
45a0: 68 61 6e 6e 65 6c 28 26 67 2e 63 67 69 49 6e 2c hannel(&g.cgiIn,
45b0: 20 73 74 64 69 6e 2c 20 6c 65 6e 29 3b 0a 20 20 stdin, len);.
45c0: 20 20 20 20 62 6c 6f 62 5f 75 6e 63 6f 6d 70 72 blob_uncompr
45d0: 65 73 73 28 26 67 2e 63 67 69 49 6e 2c 20 26 67 ess(&g.cgiIn, &g
45e0: 2e 63 67 69 49 6e 29 3b 0a 20 20 20 20 7d 65 6c .cgiIn);. }el
45f0: 73 65 20 69 66 28 20 73 74 72 63 6d 70 28 7a 54 se if( strcmp(zT
4600: 79 70 65 2c 20 22 61 70 70 6c 69 63 61 74 69 6f ype, "applicatio
4610: 6e 2f 78 2d 66 6f 73 73 69 6c 2d 64 65 62 75 67 n/x-fossil-debug
4620: 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 62 ")==0 ){. b
4630: 6c 6f 62 5f 72 65 61 64 5f 66 72 6f 6d 5f 63 68 lob_read_from_ch
4640: 61 6e 6e 65 6c 28 26 67 2e 63 67 69 49 6e 2c 20 annel(&g.cgiIn,
4650: 73 74 64 69 6e 2c 20 6c 65 6e 29 3b 0a 20 20 20 stdin, len);.
4660: 20 7d 0a 20 20 7d 0a 0a 20 20 7a 20 3d 20 28 63 }. }.. z = (c
4670: 68 61 72 2a 29 50 28 22 48 54 54 50 5f 43 4f 4f har*)P("HTTP_COO
4680: 4b 49 45 22 29 3b 0a 20 20 69 66 28 20 7a 20 29 KIE");. if( z )
4690: 7b 0a 20 20 20 20 7a 20 3d 20 6d 70 72 69 6e 74 {. z = mprint
46a0: 66 28 22 25 73 22 2c 7a 29 3b 0a 20 20 20 20 61 f("%s",z);. a
46b0: 64 64 5f 70 61 72 61 6d 5f 6c 69 73 74 28 7a 2c dd_param_list(z,
46c0: 20 27 3b 27 29 3b 0a 20 20 7d 0a 7d 0a 0a 2f 2a ';');. }.}../*
46d0: 0a 2a 2a 20 54 68 69 73 20 69 73 20 74 68 65 20 .** This is the
46e0: 63 6f 6d 70 61 72 69 73 6f 6e 20 66 75 6e 63 74 comparison funct
46f0: 69 6f 6e 20 75 73 65 64 20 74 6f 20 73 6f 72 74 ion used to sort
4700: 20 74 68 65 20 61 50 61 72 61 6d 51 50 5b 5d 20 the aParamQP[]
4710: 61 72 72 61 79 20 6f 66 0a 2a 2a 20 71 75 65 72 array of.** quer
4720: 79 20 70 61 72 61 6d 65 74 65 72 73 20 61 6e 64 y parameters and
4730: 20 63 6f 6f 6b 69 65 73 2e 0a 2a 2f 0a 73 74 61 cookies..*/.sta
4740: 74 69 63 20 69 6e 74 20 71 70 61 72 61 6d 5f 63 tic int qparam_c
4750: 6f 6d 70 61 72 65 28 63 6f 6e 73 74 20 76 6f 69 ompare(const voi
4760: 64 20 2a 61 2c 20 63 6f 6e 73 74 20 76 6f 69 64 d *a, const void
4770: 20 2a 62 29 7b 0a 20 20 73 74 72 75 63 74 20 51 *b){. struct Q
4780: 50 61 72 61 6d 20 2a 70 41 20 3d 20 28 73 74 72 Param *pA = (str
4790: 75 63 74 20 51 50 61 72 61 6d 2a 29 61 3b 0a 20 uct QParam*)a;.
47a0: 20 73 74 72 75 63 74 20 51 50 61 72 61 6d 20 2a struct QParam *
47b0: 70 42 20 3d 20 28 73 74 72 75 63 74 20 51 50 61 pB = (struct QPa
47c0: 72 61 6d 2a 29 62 3b 0a 20 20 69 6e 74 20 63 3b ram*)b;. int c;
47d0: 0a 20 20 63 20 3d 20 73 74 72 63 6d 70 28 70 41 . c = strcmp(pA
47e0: 2d 3e 7a 4e 61 6d 65 2c 20 70 42 2d 3e 7a 4e 61 ->zName, pB->zNa
47f0: 6d 65 29 3b 0a 20 20 69 66 28 20 63 3d 3d 30 20 me);. if( c==0
4800: 29 7b 0a 20 20 20 20 63 20 3d 20 70 41 2d 3e 73 ){. c = pA->s
4810: 65 71 20 2d 20 70 42 2d 3e 73 65 71 3b 0a 20 20 eq - pB->seq;.
4820: 7d 0a 20 20 72 65 74 75 72 6e 20 63 3b 0a 7d 0a }. return c;.}.
4830: 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 74 68 ./*.** Return th
4840: 65 20 76 61 6c 75 65 20 6f 66 20 61 20 71 75 65 e value of a que
4850: 72 79 20 70 61 72 61 6d 65 74 65 72 20 6f 72 20 ry parameter or
4860: 63 6f 6f 6b 69 65 20 77 68 6f 73 65 20 6e 61 6d cookie whose nam
4870: 65 20 69 73 20 7a 4e 61 6d 65 2e 0a 2a 2a 20 49 e is zName..** I
4880: 66 20 74 68 65 72 65 20 69 73 20 6e 6f 20 71 75 f there is no qu
4890: 65 72 79 20 70 61 72 61 6d 65 74 65 72 20 6f 72 ery parameter or
48a0: 20 63 6f 6f 6b 69 65 20 6e 61 6d 65 64 20 7a 4e cookie named zN
48b0: 61 6d 65 20 61 6e 64 20 74 68 65 20 66 69 72 73 ame and the firs
48c0: 74 0a 2a 2a 20 63 68 61 72 61 63 74 65 72 20 6f t.** character o
48d0: 66 20 7a 4e 61 6d 65 20 69 73 20 75 70 70 65 72 f zName is upper
48e0: 63 61 73 65 2c 20 74 68 65 6e 20 63 68 65 63 6b case, then check
48f0: 20 74 6f 20 73 65 65 20 69 66 20 74 68 65 72 65 to see if there
4900: 20 69 73 20 61 6e 0a 2a 2a 20 65 6e 76 69 72 6f is an.** enviro
4910: 6e 6d 65 6e 74 20 76 61 72 69 61 62 6c 65 20 62 nment variable b
4920: 79 20 74 68 61 74 20 6e 61 6d 65 20 61 6e 64 20 y that name and
4930: 72 65 74 75 72 6e 20 69 74 20 69 66 20 74 68 65 return it if the
4940: 72 65 20 69 73 2e 20 20 41 73 0a 2a 2a 20 61 20 re is. As.** a
4950: 6c 61 73 74 20 72 65 73 6f 72 74 20 77 68 65 6e last resort when
4960: 20 6e 6f 74 68 69 6e 67 20 65 6c 73 65 20 6d 61 nothing else ma
4970: 74 63 68 65 73 2c 20 72 65 74 75 72 6e 20 7a 44 tches, return zD
4980: 65 66 61 75 6c 74 2e 0a 2a 2f 0a 63 6f 6e 73 74 efault..*/.const
4990: 20 63 68 61 72 20 2a 63 67 69 5f 70 61 72 61 6d char *cgi_param
49a0: 65 74 65 72 28 63 6f 6e 73 74 20 63 68 61 72 20 eter(const char
49b0: 2a 7a 4e 61 6d 65 2c 20 63 6f 6e 73 74 20 63 68 *zName, const ch
49c0: 61 72 20 2a 7a 44 65 66 61 75 6c 74 29 7b 0a 20 ar *zDefault){.
49d0: 20 69 6e 74 20 6c 6f 2c 20 68 69 2c 20 6d 69 64 int lo, hi, mid
49e0: 2c 20 63 3b 0a 0a 20 20 2f 2a 20 54 68 65 20 73 , c;.. /* The s
49f0: 6f 72 74 51 50 20 66 6c 61 67 20 69 73 20 73 65 ortQP flag is se
4a00: 74 20 77 68 65 6e 65 76 65 72 20 61 20 6e 65 77 t whenever a new
4a10: 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 query parameter
4a20: 20 69 73 20 69 6e 73 65 72 74 65 64 2e 0a 20 20 is inserted..
4a30: 2a 2a 20 49 74 20 69 6e 64 69 63 61 74 65 73 20 ** It indicates
4a40: 74 68 61 74 20 77 65 20 6e 65 65 64 20 74 6f 20 that we need to
4a50: 72 65 73 6f 72 74 20 74 68 65 20 71 75 65 72 79 resort the query
4a60: 20 70 61 72 61 6d 65 74 65 72 73 2e 0a 20 20 2a parameters.. *
4a70: 2f 0a 20 20 69 66 28 20 73 6f 72 74 51 50 20 29 /. if( sortQP )
4a80: 7b 0a 20 20 20 20 69 6e 74 20 69 2c 20 6a 3b 0a {. int i, j;.
4a90: 20 20 20 20 71 73 6f 72 74 28 61 50 61 72 61 6d qsort(aParam
4aa0: 51 50 2c 20 6e 55 73 65 64 51 50 2c 20 73 69 7a QP, nUsedQP, siz
4ab0: 65 6f 66 28 61 50 61 72 61 6d 51 50 5b 30 5d 29 eof(aParamQP[0])
4ac0: 2c 20 71 70 61 72 61 6d 5f 63 6f 6d 70 61 72 65 , qparam_compare
4ad0: 29 3b 0a 20 20 20 20 73 6f 72 74 51 50 20 3d 20 );. sortQP =
4ae0: 30 3b 0a 20 20 20 20 2f 2a 20 41 66 74 65 72 20 0;. /* After
4af0: 73 6f 72 74 69 6e 67 2c 20 72 65 6d 6f 76 65 20 sorting, remove
4b00: 64 75 70 6c 69 63 61 74 65 20 70 61 72 61 6d 65 duplicate parame
4b10: 74 65 72 73 2e 20 20 54 68 65 20 73 65 63 6f 6e ters. The secon
4b20: 64 61 72 79 20 73 6f 72 74 0a 20 20 20 20 2a 2a dary sort. **
4b30: 20 6b 65 79 20 69 73 20 61 50 61 72 61 6d 51 50 key is aParamQP
4b40: 5b 5d 2e 73 65 71 20 61 6e 64 20 77 65 20 6b 65 [].seq and we ke
4b50: 65 70 20 74 68 65 20 66 69 72 73 74 20 65 6e 74 ep the first ent
4b60: 72 79 2e 20 20 54 68 61 74 20 6d 65 61 6e 73 0a ry. That means.
4b70: 20 20 20 20 2a 2a 20 77 69 74 68 20 64 75 70 6c ** with dupl
4b80: 69 63 61 74 65 20 63 61 6c 6c 73 20 74 6f 20 63 icate calls to c
4b90: 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 74 65 72 gi_set_parameter
4ba0: 28 29 20 74 68 65 20 73 65 63 6f 6e 64 20 61 6e () the second an
4bb0: 64 0a 20 20 20 20 2a 2a 20 73 75 62 73 65 71 75 d. ** subsequ
4bc0: 65 6e 74 20 63 61 6c 6c 73 20 61 72 65 20 65 66 ent calls are ef
4bd0: 66 65 63 74 69 76 65 6c 79 20 6e 6f 2d 6f 70 73 fectively no-ops
4be0: 2e 20 2a 2f 0a 20 20 20 20 66 6f 72 28 69 3d 6a . */. for(i=j
4bf0: 3d 31 3b 20 69 3c 6e 55 73 65 64 51 50 3b 20 69 =1; i<nUsedQP; i
4c00: 2b 2b 29 7b 0a 20 20 20 20 20 20 69 66 28 20 73 ++){. if( s
4c10: 74 72 63 6d 70 28 61 50 61 72 61 6d 51 50 5b 69 trcmp(aParamQP[i
4c20: 5d 2e 7a 4e 61 6d 65 2c 61 50 61 72 61 6d 51 50 ].zName,aParamQP
4c30: 5b 69 2d 31 5d 2e 7a 4e 61 6d 65 29 3d 3d 30 20 [i-1].zName)==0
4c40: 29 7b 0a 20 20 20 20 20 20 20 20 63 6f 6e 74 69 ){. conti
4c50: 6e 75 65 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 nue;. }.
4c60: 20 20 20 69 66 28 20 6a 3c 69 20 29 7b 0a 20 20 if( j<i ){.
4c70: 20 20 20 20 20 20 6d 65 6d 63 70 79 28 26 61 50 memcpy(&aP
4c80: 61 72 61 6d 51 50 5b 6a 5d 2c 20 26 61 50 61 72 aramQP[j], &aPar
4c90: 61 6d 51 50 5b 69 5d 2c 20 73 69 7a 65 6f 66 28 amQP[i], sizeof(
4ca0: 61 50 61 72 61 6d 51 50 5b 6a 5d 29 29 3b 0a 20 aParamQP[j]));.
4cb0: 20 20 20 20 20 7d 0a 20 20 20 20 20 20 6a 2b 2b }. j++
4cc0: 3b 0a 20 20 20 20 7d 0a 20 20 20 20 6e 55 73 65 ;. }. nUse
4cd0: 64 51 50 20 3d 20 6a 3b 0a 20 20 7d 0a 0a 20 20 dQP = j;. }..
4ce0: 2f 2a 20 44 6f 20 61 20 62 69 6e 61 72 79 20 73 /* Do a binary s
4cf0: 65 61 72 63 68 20 66 6f 72 20 61 20 6d 61 74 63 earch for a matc
4d00: 68 69 6e 67 20 71 75 65 72 79 20 70 61 72 61 6d hing query param
4d10: 65 74 65 72 20 2a 2f 0a 20 20 6c 6f 20 3d 20 30 eter */. lo = 0
4d20: 3b 0a 20 20 68 69 20 3d 20 6e 55 73 65 64 51 50 ;. hi = nUsedQP
4d30: 2d 31 3b 0a 20 20 77 68 69 6c 65 28 20 6c 6f 3c -1;. while( lo<
4d40: 3d 68 69 20 29 7b 0a 20 20 20 20 6d 69 64 20 3d =hi ){. mid =
4d50: 20 28 6c 6f 2b 68 69 29 2f 32 3b 0a 20 20 20 20 (lo+hi)/2;.
4d60: 63 20 3d 20 73 74 72 63 6d 70 28 61 50 61 72 61 c = strcmp(aPara
4d70: 6d 51 50 5b 6d 69 64 5d 2e 7a 4e 61 6d 65 2c 20 mQP[mid].zName,
4d80: 7a 4e 61 6d 65 29 3b 0a 20 20 20 20 69 66 28 20 zName);. if(
4d90: 63 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 43 47 c==0 ){. CG
4da0: 49 44 45 42 55 47 28 28 22 6d 65 6d 2d 6d 61 74 IDEBUG(("mem-mat
4db0: 63 68 20 5b 25 73 5d 20 3d 20 5b 25 73 5d 5c 6e ch [%s] = [%s]\n
4dc0: 22 2c 20 7a 4e 61 6d 65 2c 20 61 50 61 72 61 6d ", zName, aParam
4dd0: 51 50 5b 6d 69 64 5d 2e 7a 56 61 6c 75 65 29 29 QP[mid].zValue))
4de0: 3b 0a 20 20 20 20 20 20 72 65 74 75 72 6e 20 61 ;. return a
4df0: 50 61 72 61 6d 51 50 5b 6d 69 64 5d 2e 7a 56 61 ParamQP[mid].zVa
4e00: 6c 75 65 3b 0a 20 20 20 20 7d 65 6c 73 65 20 69 lue;. }else i
4e10: 66 28 20 63 3e 30 20 29 7b 0a 20 20 20 20 20 20 f( c>0 ){.
4e20: 68 69 20 3d 20 6d 69 64 2d 31 3b 0a 20 20 20 20 hi = mid-1;.
4e30: 7d 65 6c 73 65 7b 0a 20 20 20 20 20 20 6c 6f 20 }else{. lo
4e40: 3d 20 6d 69 64 2b 31 3b 0a 20 20 20 20 7d 0a 20 = mid+1;. }.
4e50: 20 7d 0a 0a 20 20 2f 2a 20 49 66 20 6e 6f 20 6d }.. /* If no m
4e60: 61 74 63 68 20 69 73 20 66 6f 75 6e 64 20 61 6e atch is found an
4e70: 64 20 74 68 65 20 6e 61 6d 65 20 62 65 67 69 6e d the name begin
4e80: 73 20 77 69 74 68 20 61 6e 20 75 70 70 65 72 2d s with an upper-
4e90: 63 61 73 65 0a 20 20 2a 2a 20 6c 65 74 74 65 72 case. ** letter
4ea0: 2c 20 74 68 65 6e 20 63 68 65 63 6b 20 74 6f 20 , then check to
4eb0: 73 65 65 20 69 66 20 74 68 65 72 65 20 69 73 20 see if there is
4ec0: 61 6e 20 65 6e 76 69 72 6f 6e 6d 65 6e 74 20 76 an environment v
4ed0: 61 72 69 61 62 6c 65 0a 20 20 2a 2a 20 77 69 74 ariable. ** wit
4ee0: 68 20 74 68 65 20 67 69 76 65 6e 20 6e 61 6d 65 h the given name
4ef0: 2e 0a 20 20 2a 2f 0a 20 20 69 66 28 20 69 73 75 .. */. if( isu
4f00: 70 70 65 72 28 7a 4e 61 6d 65 5b 30 5d 29 20 29 pper(zName[0]) )
4f10: 7b 0a 20 20 20 20 63 6f 6e 73 74 20 63 68 61 72 {. const char
4f20: 20 2a 7a 56 61 6c 75 65 20 3d 20 67 65 74 65 6e *zValue = geten
4f30: 76 28 7a 4e 61 6d 65 29 3b 0a 20 20 20 20 69 66 v(zName);. if
4f40: 28 20 7a 56 61 6c 75 65 20 29 7b 0a 20 20 20 20 ( zValue ){.
4f50: 20 20 63 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 cgi_set_parame
4f60: 74 65 72 5f 6e 6f 63 6f 70 79 28 7a 4e 61 6d 65 ter_nocopy(zName
4f70: 2c 20 7a 56 61 6c 75 65 29 3b 0a 20 20 20 20 20 , zValue);.
4f80: 20 43 47 49 44 45 42 55 47 28 28 22 65 6e 76 2d CGIDEBUG(("env-
4f90: 6d 61 74 63 68 20 5b 25 73 5d 20 3d 20 5b 25 73 match [%s] = [%s
4fa0: 5d 5c 6e 22 2c 20 7a 4e 61 6d 65 2c 20 7a 56 61 ]\n", zName, zVa
4fb0: 6c 75 65 29 29 3b 0a 20 20 20 20 20 20 72 65 74 lue));. ret
4fc0: 75 72 6e 20 7a 56 61 6c 75 65 3b 0a 20 20 20 20 urn zValue;.
4fd0: 7d 0a 20 20 7d 0a 20 20 43 47 49 44 45 42 55 47 }. }. CGIDEBUG
4fe0: 28 28 22 6e 6f 2d 6d 61 74 63 68 20 5b 25 73 5d (("no-match [%s]
4ff0: 5c 6e 22 2c 20 7a 4e 61 6d 65 29 29 3b 0a 20 20 \n", zName));.
5000: 72 65 74 75 72 6e 20 7a 44 65 66 61 75 6c 74 3b return zDefault;
5010: 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 50 72 69 6e 74 20 .}../*.** Print
5020: 43 47 49 20 64 65 62 75 67 67 69 6e 67 20 6d 65 CGI debugging me
5030: 73 73 61 67 65 73 2e 0a 2a 2f 0a 76 6f 69 64 20 ssages..*/.void
5040: 63 67 69 5f 64 65 62 75 67 28 63 6f 6e 73 74 20 cgi_debug(const
5050: 63 68 61 72 20 2a 7a 46 6f 72 6d 61 74 2c 20 2e char *zFormat, .
5060: 2e 2e 29 7b 0a 20 20 76 61 5f 6c 69 73 74 20 61 ..){. va_list a
5070: 70 3b 0a 20 20 69 66 28 20 67 2e 66 44 65 62 75 p;. if( g.fDebu
5080: 67 20 29 7b 0a 20 20 20 20 76 61 5f 73 74 61 72 g ){. va_star
5090: 74 28 61 70 2c 20 7a 46 6f 72 6d 61 74 29 3b 0a t(ap, zFormat);.
50a0: 20 20 20 20 76 66 70 72 69 6e 74 66 28 67 2e 66 vfprintf(g.f
50b0: 44 65 62 75 67 2c 20 7a 46 6f 72 6d 61 74 2c 20 Debug, zFormat,
50c0: 61 70 29 3b 0a 20 20 20 20 76 61 5f 65 6e 64 28 ap);. va_end(
50d0: 61 70 29 3b 0a 20 20 20 20 66 66 6c 75 73 68 28 ap);. fflush(
50e0: 67 2e 66 44 65 62 75 67 29 3b 0a 20 20 7d 0a 7d g.fDebug);. }.}
50f0: 0a 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 74 ../*.** Return t
5100: 72 75 65 20 69 66 20 61 6e 79 20 6f 66 20 74 68 rue if any of th
5110: 65 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 e query paramete
5120: 72 73 20 69 6e 20 74 68 65 20 61 72 67 75 6d 65 rs in the argume
5130: 6e 74 0a 2a 2a 20 6c 69 73 74 20 61 72 65 20 64 nt.** list are d
5140: 65 66 69 6e 65 64 2e 0a 2a 2f 0a 69 6e 74 20 63 efined..*/.int c
5150: 67 69 5f 61 6e 79 28 63 6f 6e 73 74 20 63 68 61 gi_any(const cha
5160: 72 20 2a 7a 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 r *z, ...){. va
5170: 5f 6c 69 73 74 20 61 70 3b 0a 20 20 63 68 61 72 _list ap;. char
5180: 20 2a 7a 32 3b 0a 20 20 69 66 28 20 63 67 69 5f *z2;. if( cgi_
5190: 70 61 72 61 6d 65 74 65 72 28 7a 2c 30 29 21 3d parameter(z,0)!=
51a0: 30 20 29 20 72 65 74 75 72 6e 20 31 3b 0a 20 20 0 ) return 1;.
51b0: 76 61 5f 73 74 61 72 74 28 61 70 2c 20 7a 29 3b va_start(ap, z);
51c0: 0a 20 20 77 68 69 6c 65 28 20 28 7a 32 20 3d 20 . while( (z2 =
51d0: 76 61 5f 61 72 67 28 61 70 2c 20 63 68 61 72 2a va_arg(ap, char*
51e0: 29 29 21 3d 30 20 29 7b 0a 20 20 20 20 69 66 28 ))!=0 ){. if(
51f0: 20 63 67 69 5f 70 61 72 61 6d 65 74 65 72 28 7a cgi_parameter(z
5200: 32 2c 30 29 21 3d 30 20 29 20 72 65 74 75 72 6e 2,0)!=0 ) return
5210: 20 31 3b 0a 20 20 7d 0a 20 20 76 61 5f 65 6e 64 1;. }. va_end
5220: 28 61 70 29 3b 0a 20 20 72 65 74 75 72 6e 20 30 (ap);. return 0
5230: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 ;.}../*.** Retur
5240: 6e 20 74 72 75 65 20 69 66 20 61 6c 6c 20 6f 66 n true if all of
5250: 20 74 68 65 20 71 75 65 72 79 20 70 61 72 61 6d the query param
5260: 65 74 65 72 73 20 69 6e 20 74 68 65 20 61 72 67 eters in the arg
5270: 75 6d 65 6e 74 20 6c 69 73 74 0a 2a 2a 20 61 72 ument list.** ar
5280: 65 20 64 65 66 69 6e 65 64 2e 0a 2a 2f 0a 69 6e e defined..*/.in
5290: 74 20 63 67 69 5f 61 6c 6c 28 63 6f 6e 73 74 20 t cgi_all(const
52a0: 63 68 61 72 20 2a 7a 2c 20 2e 2e 2e 29 7b 0a 20 char *z, ...){.
52b0: 20 76 61 5f 6c 69 73 74 20 61 70 3b 0a 20 20 63 va_list ap;. c
52c0: 68 61 72 20 2a 7a 32 3b 0a 20 20 69 66 28 20 63 har *z2;. if( c
52d0: 67 69 5f 70 61 72 61 6d 65 74 65 72 28 7a 2c 30 gi_parameter(z,0
52e0: 29 3d 3d 30 20 29 20 72 65 74 75 72 6e 20 30 3b )==0 ) return 0;
52f0: 0a 20 20 76 61 5f 73 74 61 72 74 28 61 70 2c 20 . va_start(ap,
5300: 7a 29 3b 0a 20 20 77 68 69 6c 65 28 20 28 7a 32 z);. while( (z2
5310: 20 3d 20 76 61 5f 61 72 67 28 61 70 2c 20 63 68 = va_arg(ap, ch
5320: 61 72 2a 29 29 3d 3d 30 20 29 7b 0a 20 20 20 20 ar*))==0 ){.
5330: 69 66 28 20 63 67 69 5f 70 61 72 61 6d 65 74 65 if( cgi_paramete
5340: 72 28 7a 32 2c 30 29 3d 3d 30 20 29 20 72 65 74 r(z2,0)==0 ) ret
5350: 75 72 6e 20 30 3b 0a 20 20 7d 0a 20 20 76 61 5f urn 0;. }. va_
5360: 65 6e 64 28 61 70 29 3b 0a 20 20 72 65 74 75 72 end(ap);. retur
5370: 6e 20 31 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 50 72 n 1;.}../*.** Pr
5380: 69 6e 74 20 61 6c 6c 20 71 75 65 72 79 20 70 61 int all query pa
5390: 72 61 6d 65 74 65 72 73 20 6f 6e 20 73 74 61 6e rameters on stan
53a0: 64 61 72 64 20 6f 75 74 70 75 74 2e 20 20 46 6f dard output. Fo
53b0: 72 6d 61 74 20 74 68 65 0a 2a 2a 20 70 61 72 61 rmat the.** para
53c0: 6d 65 74 65 72 73 20 61 73 20 48 54 4d 4c 2e 20 meters as HTML.
53d0: 20 54 68 69 73 20 69 73 20 75 73 65 64 20 66 6f This is used fo
53e0: 72 20 74 65 73 74 69 6e 67 20 61 6e 64 20 64 65 r testing and de
53f0: 62 75 67 67 69 6e 67 2e 0a 2a 2f 0a 76 6f 69 64 bugging..*/.void
5400: 20 63 67 69 5f 70 72 69 6e 74 5f 61 6c 6c 28 76 cgi_print_all(v
5410: 6f 69 64 29 7b 0a 20 20 69 6e 74 20 69 3b 0a 20 oid){. int i;.
5420: 20 63 67 69 5f 70 61 72 61 6d 65 74 65 72 28 22 cgi_parameter("
5430: 22 2c 22 22 29 3b 20 20 2f 2a 20 46 6f 72 63 65 ",""); /* Force
5440: 20 74 68 65 20 70 61 72 61 6d 65 74 65 72 73 20 the parameters
5450: 69 6e 74 6f 20 73 6f 72 74 65 64 20 6f 72 64 65 into sorted orde
5460: 72 20 2a 2f 0a 20 20 66 6f 72 28 69 3d 30 3b 20 r */. for(i=0;
5470: 69 3c 6e 55 73 65 64 51 50 3b 20 69 2b 2b 29 7b i<nUsedQP; i++){
5480: 0a 20 20 20 20 63 67 69 5f 70 72 69 6e 74 66 28 . cgi_printf(
5490: 22 25 73 20 3d 20 25 73 20 20 3c 62 72 20 2f 3e "%s = %s <br />
54a0: 5c 6e 22 2c 0a 20 20 20 20 20 20 20 68 74 6d 6c \n",. html
54b0: 69 7a 65 28 61 50 61 72 61 6d 51 50 5b 69 5d 2e ize(aParamQP[i].
54c0: 7a 4e 61 6d 65 2c 20 2d 31 29 2c 20 68 74 6d 6c zName, -1), html
54d0: 69 7a 65 28 61 50 61 72 61 6d 51 50 5b 69 5d 2e ize(aParamQP[i].
54e0: 7a 56 61 6c 75 65 2c 20 2d 31 29 29 3b 0a 20 20 zValue, -1));.
54f0: 7d 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 57 72 69 74 65 }.}../*.** Write
5500: 20 48 54 4d 4c 20 74 65 78 74 20 66 6f 72 20 61 HTML text for a
5510: 6e 20 6f 70 74 69 6f 6e 20 6d 65 6e 75 20 74 6f n option menu to
5520: 20 73 74 61 6e 64 61 72 64 20 6f 75 74 70 75 74 standard output
5530: 2e 20 20 7a 50 61 72 61 6d 0a 2a 2a 20 69 73 20 . zParam.** is
5540: 74 68 65 20 71 75 65 72 79 20 70 61 72 61 6d 65 the query parame
5550: 74 65 72 20 74 68 61 74 20 74 68 65 20 6f 70 74 ter that the opt
5560: 69 6f 6e 20 6d 65 6e 75 20 73 65 74 73 2e 20 20 ion menu sets.
5570: 7a 44 66 6c 74 20 69 73 20 74 68 65 0a 2a 2a 20 zDflt is the.**
5580: 69 6e 69 74 69 61 6c 20 76 61 6c 75 65 20 6f 66 initial value of
5590: 20 74 68 65 20 6f 70 74 69 6f 6e 20 6d 65 6e 75 the option menu
55a0: 2e 20 20 41 64 64 69 74 69 6f 6e 20 61 72 67 75 . Addition argu
55b0: 6d 65 6e 74 73 20 61 72 65 20 6e 61 6d 65 2f 76 ments are name/v
55c0: 61 6c 75 65 0a 2a 2a 20 70 61 69 72 73 20 74 68 alue.** pairs th
55d0: 61 74 20 64 65 66 69 6e 65 20 76 61 6c 75 65 73 at define values
55e0: 20 6f 6e 20 74 68 65 20 6d 65 6e 75 2e 20 20 54 on the menu. T
55f0: 68 65 20 6c 69 73 74 20 69 73 20 74 65 72 6d 69 he list is termi
5600: 6e 61 74 65 64 20 77 69 74 68 0a 2a 2a 20 61 20 nated with.** a
5610: 73 69 6e 67 6c 65 20 4e 55 4c 4c 20 61 72 67 75 single NULL argu
5620: 6d 65 6e 74 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 ment..*/.void cg
5630: 69 5f 6f 70 74 69 6f 6e 6d 65 6e 75 28 69 6e 74 i_optionmenu(int
5640: 20 69 6e 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 in, const char
5650: 2a 7a 50 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 *zP, const char
5660: 2a 7a 44 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f *zD, ...){. va_
5670: 6c 69 73 74 20 61 70 3b 0a 20 20 63 68 61 72 20 list ap;. char
5680: 2a 7a 4e 61 6d 65 2c 20 2a 7a 56 61 6c 3b 0a 20 *zName, *zVal;.
5690: 20 69 6e 74 20 64 66 6c 74 53 65 65 6e 20 3d 20 int dfltSeen =
56a0: 30 3b 0a 20 20 63 67 69 5f 70 72 69 6e 74 66 28 0;. cgi_printf(
56b0: 22 25 2a 73 3c 73 65 6c 65 63 74 20 73 69 7a 65 "%*s<select size
56c0: 3d 31 20 6e 61 6d 65 3d 5c 22 25 73 5c 22 3e 5c =1 name=\"%s\">\
56d0: 6e 22 2c 20 69 6e 2c 20 22 22 2c 20 7a 50 29 3b n", in, "", zP);
56e0: 0a 20 20 76 61 5f 73 74 61 72 74 28 61 70 2c 20 . va_start(ap,
56f0: 7a 44 29 3b 0a 20 20 77 68 69 6c 65 28 20 28 7a zD);. while( (z
5700: 4e 61 6d 65 20 3d 20 76 61 5f 61 72 67 28 61 70 Name = va_arg(ap
5710: 2c 20 63 68 61 72 2a 29 29 21 3d 30 20 26 26 20 , char*))!=0 &&
5720: 28 7a 56 61 6c 20 3d 20 76 61 5f 61 72 67 28 61 (zVal = va_arg(a
5730: 70 2c 20 63 68 61 72 2a 29 29 21 3d 30 20 29 7b p, char*))!=0 ){
5740: 0a 20 20 20 20 69 66 28 20 73 74 72 63 6d 70 28 . if( strcmp(
5750: 7a 56 61 6c 2c 7a 44 29 3d 3d 30 20 29 7b 20 64 zVal,zD)==0 ){ d
5760: 66 6c 74 53 65 65 6e 20 3d 20 31 3b 20 62 72 65 fltSeen = 1; bre
5770: 61 6b 3b 20 7d 0a 20 20 7d 0a 20 20 76 61 5f 65 ak; }. }. va_e
5780: 6e 64 28 61 70 29 3b 0a 20 20 69 66 28 20 21 64 nd(ap);. if( !d
5790: 66 6c 74 53 65 65 6e 20 29 7b 0a 20 20 20 20 69 fltSeen ){. i
57a0: 66 28 20 7a 44 5b 30 5d 20 29 7b 0a 20 20 20 20 f( zD[0] ){.
57b0: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a cgi_printf("%*
57c0: 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c s<option value=\
57d0: 22 25 68 5c 22 20 73 65 6c 65 63 74 65 64 3e 25 "%h\" selected>%
57e0: 68 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 h</option>\n",.
57f0: 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c in+2, "",
5800: 20 7a 44 2c 20 7a 44 29 3b 0a 20 20 20 20 7d 65 zD, zD);. }e
5810: 6c 73 65 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 lse{. cgi_p
5820: 72 69 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f rintf("%*s<optio
5830: 6e 20 76 61 6c 75 65 3d 5c 22 5c 22 20 73 65 6c n value=\"\" sel
5840: 65 63 74 65 64 3e 26 6e 62 73 70 3b 3c 2f 6f 70 ected> </op
5850: 74 69 6f 6e 3e 5c 6e 22 2c 20 69 6e 2b 32 2c 20 tion>\n", in+2,
5860: 22 22 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 "");. }. }.
5870: 20 76 61 5f 73 74 61 72 74 28 61 70 2c 20 7a 44 va_start(ap, zD
5880: 29 3b 0a 20 20 77 68 69 6c 65 28 20 28 7a 4e 61 );. while( (zNa
5890: 6d 65 20 3d 20 76 61 5f 61 72 67 28 61 70 2c 20 me = va_arg(ap,
58a0: 63 68 61 72 2a 29 29 21 3d 30 20 26 26 20 28 7a char*))!=0 && (z
58b0: 56 61 6c 20 3d 20 76 61 5f 61 72 67 28 61 70 2c Val = va_arg(ap,
58c0: 20 63 68 61 72 2a 29 29 21 3d 30 20 29 7b 0a 20 char*))!=0 ){.
58d0: 20 20 20 69 66 28 20 7a 4e 61 6d 65 5b 30 5d 20 if( zName[0]
58e0: 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69 ){. cgi_pri
58f0: 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20 ntf("%*s<option
5900: 76 61 6c 75 65 3d 5c 22 25 68 5c 22 25 73 3e 25 value=\"%h\"%s>%
5910: 68 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 h</option>\n",.
5920: 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c in+2, "",
5930: 0a 20 20 20 20 20 20 20 20 7a 56 61 6c 2c 0a 20 . zVal,.
5940: 20 20 20 20 20 20 20 73 74 72 63 6d 70 28 7a 56 strcmp(zV
5950: 61 6c 2c 20 7a 44 29 20 3f 20 22 22 20 3a 20 22 al, zD) ? "" : "
5960: 20 73 65 6c 65 63 74 65 64 22 2c 0a 20 20 20 20 selected",.
5970: 20 20 20 20 7a 4e 61 6d 65 0a 20 20 20 20 20 20 zName.
5980: 29 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 );. }else{.
5990: 20 20 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 cgi_printf("
59a0: 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65 %*s<option value
59b0: 3d 5c 22 5c 22 25 73 3e 26 6e 62 73 70 3b 3c 2f =\"\"%s> </
59c0: 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 option>\n",.
59d0: 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c 0a 20 20 in+2, "",.
59e0: 20 20 20 20 20 20 73 74 72 63 6d 70 28 7a 56 61 strcmp(zVa
59f0: 6c 2c 20 7a 44 29 20 3f 20 22 22 20 3a 20 22 20 l, zD) ? "" : "
5a00: 73 65 6c 65 63 74 65 64 22 0a 20 20 20 20 20 20 selected".
5a10: 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 76 );. }. }. v
5a20: 61 5f 65 6e 64 28 61 70 29 3b 0a 20 20 63 67 69 a_end(ap);. cgi
5a30: 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 2f 73 65 _printf("%*s</se
5a40: 6c 65 63 74 3e 5c 6e 22 2c 20 69 6e 2c 20 22 22 lect>\n", in, ""
5a50: 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 69 73 );.}../*.** This
5a60: 20 72 6f 75 74 69 6e 65 20 77 6f 72 6b 73 20 61 routine works a
5a70: 20 6c 6f 74 20 6c 69 6b 65 20 63 67 69 5f 6f 70 lot like cgi_op
5a80: 74 69 6f 6e 6d 65 6e 75 28 29 20 65 78 63 65 70 tionmenu() excep
5a90: 74 20 74 68 61 74 20 74 68 65 20 6c 69 73 74 20 t that the list
5aa0: 6f 66 0a 2a 2a 20 76 61 6c 75 65 73 20 69 73 20 of.** values is
5ab0: 63 6f 6e 74 61 69 6e 65 64 20 69 6e 20 61 6e 20 contained in an
5ac0: 61 72 72 61 79 2e 20 20 41 6c 73 6f 2c 20 74 68 array. Also, th
5ad0: 65 20 76 61 6c 75 65 73 20 61 72 65 20 6a 75 73 e values are jus
5ae0: 74 20 76 61 6c 75 65 73 2c 20 6e 6f 74 0a 2a 2a t values, not.**
5af0: 20 6e 61 6d 65 2f 76 61 6c 75 65 20 70 61 69 72 name/value pair
5b00: 73 20 61 73 20 69 6e 20 63 67 69 5f 6f 70 74 69 s as in cgi_opti
5b10: 6f 6e 6d 65 6e 75 2e 0a 2a 2f 0a 76 6f 69 64 20 onmenu..*/.void
5b20: 63 67 69 5f 76 5f 6f 70 74 69 6f 6e 6d 65 6e 75 cgi_v_optionmenu
5b30: 28 0a 20 20 69 6e 74 20 69 6e 2c 20 20 20 20 20 (. int in,
5b40: 20 20 20 20 20 20 20 20 20 2f 2a 20 49 6e 64 65 /* Inde
5b50: 6e 74 20 62 79 20 74 68 69 73 20 61 6d 6f 75 6e nt by this amoun
5b60: 74 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 t */. const cha
5b70: 72 20 2a 7a 50 2c 20 20 20 20 20 20 2f 2a 20 54 r *zP, /* T
5b80: 68 65 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 he query paramet
5b90: 65 72 20 6e 61 6d 65 20 2a 2f 0a 20 20 63 6f 6e er name */. con
5ba0: 73 74 20 63 68 61 72 20 2a 7a 44 2c 20 20 20 20 st char *zD,
5bb0: 20 20 2f 2a 20 44 65 66 61 75 6c 74 20 76 61 6c /* Default val
5bc0: 75 65 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 ue */. const ch
5bd0: 61 72 20 2a 2a 61 7a 20 20 20 20 20 20 2f 2a 20 ar **az /*
5be0: 4e 55 4c 4c 2d 74 65 72 6d 69 6e 61 74 65 64 20 NULL-terminated
5bf0: 6c 69 73 74 20 6f 66 20 61 6c 6c 6f 77 65 64 20 list of allowed
5c00: 76 61 6c 75 65 73 20 2a 2f 0a 29 7b 0a 20 20 63 values */.){. c
5c10: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 56 61 6c 3b onst char *zVal;
5c20: 0a 20 20 69 6e 74 20 69 3b 0a 20 20 63 67 69 5f . int i;. cgi_
5c30: 70 72 69 6e 74 66 28 22 25 2a 73 3c 73 65 6c 65 printf("%*s<sele
5c40: 63 74 20 73 69 7a 65 3d 31 20 6e 61 6d 65 3d 5c ct size=1 name=\
5c50: 22 25 73 5c 22 3e 5c 6e 22 2c 20 69 6e 2c 20 22 "%s\">\n", in, "
5c60: 22 2c 20 7a 50 29 3b 0a 20 20 66 6f 72 28 69 3d ", zP);. for(i=
5c70: 30 3b 20 61 7a 5b 69 5d 3b 20 69 2b 2b 29 7b 0a 0; az[i]; i++){.
5c80: 20 20 20 20 69 66 28 20 73 74 72 63 6d 70 28 61 if( strcmp(a
5c90: 7a 5b 69 5d 2c 7a 44 29 3d 3d 30 20 29 20 62 72 z[i],zD)==0 ) br
5ca0: 65 61 6b 3b 0a 20 20 7d 0a 20 20 69 66 28 20 61 eak;. }. if( a
5cb0: 7a 5b 69 5d 3d 3d 30 20 29 7b 0a 20 20 20 20 69 z[i]==0 ){. i
5cc0: 66 28 20 7a 44 5b 30 5d 3d 3d 30 20 29 7b 0a 20 f( zD[0]==0 ){.
5cd0: 20 20 20 20 20 63 67 69 5f 70 72 69 6e 74 66 28 cgi_printf(
5ce0: 22 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 "%*s<option valu
5cf0: 65 3d 5c 22 5c 22 20 73 65 6c 65 63 74 65 64 3e e=\"\" selected>
5d00: 26 6e 62 73 70 3b 3c 2f 6f 70 74 69 6f 6e 3e 5c </option>\
5d10: 6e 22 2c 0a 20 20 20 20 20 20 20 69 6e 2b 32 2c n",. in+2,
5d20: 20 22 22 29 3b 0a 20 20 20 20 7d 65 6c 73 65 7b "");. }else{
5d30: 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69 6e 74 . cgi_print
5d40: 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 f("%*s<option va
5d50: 6c 75 65 3d 5c 22 25 68 5c 22 20 73 65 6c 65 63 lue=\"%h\" selec
5d60: 74 65 64 3e 25 68 3c 2f 6f 70 74 69 6f 6e 3e 5c ted>%h</option>\
5d70: 6e 22 2c 0a 20 20 20 20 20 20 20 69 6e 2b 32 2c n",. in+2,
5d80: 20 22 22 2c 20 7a 44 2c 20 7a 44 29 3b 0a 20 20 "", zD, zD);.
5d90: 20 20 7d 0a 20 20 7d 0a 20 20 77 68 69 6c 65 28 }. }. while(
5da0: 20 28 7a 56 61 6c 20 3d 20 2a 28 61 7a 2b 2b 29 (zVal = *(az++)
5db0: 29 21 3d 30 20 20 29 7b 0a 20 20 20 20 69 66 28 )!=0 ){. if(
5dc0: 20 7a 56 61 6c 5b 30 5d 20 29 7b 0a 20 20 20 20 zVal[0] ){.
5dd0: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a cgi_printf("%*
5de0: 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c s<option value=\
5df0: 22 25 68 5c 22 25 73 3e 25 68 3c 2f 6f 70 74 69 "%h\"%s>%h</opti
5e00: 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 20 20 20 20 on>\n",.
5e10: 69 6e 2b 32 2c 20 22 22 2c 0a 20 20 20 20 20 20 in+2, "",.
5e20: 20 20 7a 56 61 6c 2c 0a 20 20 20 20 20 20 20 20 zVal,.
5e30: 73 74 72 63 6d 70 28 7a 56 61 6c 2c 20 7a 44 29 strcmp(zVal, zD)
5e40: 20 3f 20 22 22 20 3a 20 22 20 73 65 6c 65 63 74 ? "" : " select
5e50: 65 64 22 2c 0a 20 20 20 20 20 20 20 20 7a 56 61 ed",. zVa
5e60: 6c 0a 20 20 20 20 20 20 29 3b 0a 20 20 20 20 7d l. );. }
5e70: 65 6c 73 65 7b 0a 20 20 20 20 20 20 63 67 69 5f else{. cgi_
5e80: 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 printf("%*s<opti
5e90: 6f 6e 20 76 61 6c 75 65 3d 5c 22 5c 22 25 73 3e on value=\"\"%s>
5ea0: 26 6e 62 73 70 3b 3c 2f 6f 70 74 69 6f 6e 3e 5c </option>\
5eb0: 6e 22 2c 0a 20 20 20 20 20 20 20 20 69 6e 2b 32 n",. in+2
5ec0: 2c 20 22 22 2c 0a 20 20 20 20 20 20 20 20 73 74 , "",. st
5ed0: 72 63 6d 70 28 7a 56 61 6c 2c 20 7a 44 29 20 3f rcmp(zVal, zD) ?
5ee0: 20 22 22 20 3a 20 22 20 73 65 6c 65 63 74 65 64 "" : " selected
5ef0: 22 0a 20 20 20 20 20 20 29 3b 0a 20 20 20 20 7d ". );. }
5f00: 0a 20 20 7d 0a 20 20 63 67 69 5f 70 72 69 6e 74 . }. cgi_print
5f10: 66 28 22 25 2a 73 3c 2f 73 65 6c 65 63 74 3e 5c f("%*s</select>\
5f20: 6e 22 2c 20 69 6e 2c 20 22 22 29 3b 0a 7d 0a 0a n", in, "");.}..
5f30: 2f 2a 0a 2a 2a 20 54 68 69 73 20 72 6f 75 74 69 /*.** This routi
5f40: 6e 65 20 77 6f 72 6b 73 20 61 20 6c 6f 74 20 6c ne works a lot l
5f50: 69 6b 65 20 63 67 69 5f 76 5f 6f 70 74 69 6f 6e ike cgi_v_option
5f60: 6d 65 6e 75 28 29 20 65 78 63 65 70 74 20 74 68 menu() except th
5f70: 61 74 20 74 68 65 20 6c 69 73 74 0a 2a 2a 20 69 at the list.** i
5f80: 73 20 61 20 6c 69 73 74 20 6f 66 20 70 61 69 72 s a list of pair
5f90: 73 2e 20 20 54 68 65 20 66 69 72 73 74 20 65 6c s. The first el
5fa0: 65 6d 65 6e 74 20 6f 66 20 65 61 63 68 20 70 61 ement of each pa
5fb0: 69 72 20 69 73 20 74 68 65 20 76 61 6c 75 65 20 ir is the value
5fc0: 75 73 65 64 0a 2a 2a 20 69 6e 74 65 72 6e 61 6c used.** internal
5fd0: 6c 79 20 61 6e 64 20 74 68 65 20 73 65 63 6f 6e ly and the secon
5fe0: 64 20 65 6c 65 6d 65 6e 74 20 69 73 20 74 68 65 d element is the
5ff0: 20 76 61 6c 75 65 20 64 69 73 70 6c 61 79 65 64 value displayed
6000: 20 74 6f 20 74 68 65 20 75 73 65 72 2e 0a 2a 2f to the user..*/
6010: 0a 76 6f 69 64 20 63 67 69 5f 76 5f 6f 70 74 69 .void cgi_v_opti
6020: 6f 6e 6d 65 6e 75 32 28 0a 20 20 69 6e 74 20 69 onmenu2(. int i
6030: 6e 2c 20 20 20 20 20 20 20 20 20 20 20 20 20 20 n,
6040: 2f 2a 20 49 6e 64 65 6e 74 20 62 79 20 74 68 69 /* Indent by thi
6050: 73 20 61 6d 6f 75 6e 74 20 2a 2f 0a 20 20 63 6f s amount */. co
6060: 6e 73 74 20 63 68 61 72 20 2a 7a 50 2c 20 20 20 nst char *zP,
6070: 20 20 20 2f 2a 20 54 68 65 20 71 75 65 72 79 20 /* The query
6080: 70 61 72 61 6d 65 74 65 72 20 6e 61 6d 65 20 2a parameter name *
6090: 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a /. const char *
60a0: 7a 44 2c 20 20 20 20 20 20 2f 2a 20 44 65 66 61 zD, /* Defa
60b0: 75 6c 74 20 76 61 6c 75 65 20 2a 2f 0a 20 20 63 ult value */. c
60c0: 6f 6e 73 74 20 63 68 61 72 20 2a 2a 61 7a 20 20 onst char **az
60d0: 20 20 20 20 2f 2a 20 4e 55 4c 4c 2d 74 65 72 6d /* NULL-term
60e0: 69 6e 61 74 65 64 20 6c 69 73 74 20 6f 66 20 61 inated list of a
60f0: 6c 6c 6f 77 65 64 20 76 61 6c 75 65 73 20 2a 2f llowed values */
6100: 0a 29 7b 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 .){. const char
6110: 20 2a 7a 56 61 6c 3b 0a 20 20 69 6e 74 20 69 3b *zVal;. int i;
6120: 0a 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 . cgi_printf("%
6130: 2a 73 3c 73 65 6c 65 63 74 20 73 69 7a 65 3d 31 *s<select size=1
6140: 20 6e 61 6d 65 3d 5c 22 25 73 5c 22 3e 5c 6e 22 name=\"%s\">\n"
6150: 2c 20 69 6e 2c 20 22 22 2c 20 7a 50 29 3b 0a 20 , in, "", zP);.
6160: 20 66 6f 72 28 69 3d 30 3b 20 61 7a 5b 69 5d 3b for(i=0; az[i];
6170: 20 69 2b 3d 32 29 7b 0a 20 20 20 20 69 66 28 20 i+=2){. if(
6180: 73 74 72 63 6d 70 28 61 7a 5b 69 5d 2c 7a 44 29 strcmp(az[i],zD)
6190: 3d 3d 30 20 29 20 62 72 65 61 6b 3b 0a 20 20 7d ==0 ) break;. }
61a0: 0a 20 20 69 66 28 20 61 7a 5b 69 5d 3d 3d 30 20 . if( az[i]==0
61b0: 29 7b 0a 20 20 20 20 69 66 28 20 7a 44 5b 30 5d ){. if( zD[0]
61c0: 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 63 67 69 ==0 ){. cgi
61d0: 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f 70 74 _printf("%*s<opt
61e0: 69 6f 6e 20 76 61 6c 75 65 3d 5c 22 5c 22 20 73 ion value=\"\" s
61f0: 65 6c 65 63 74 65 64 3e 26 6e 62 73 70 3b 3c 2f elected> </
6200: 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 option>\n",.
6210: 20 20 20 69 6e 2b 32 2c 20 22 22 29 3b 0a 20 20 in+2, "");.
6220: 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20 20 63 }else{. c
6230: 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f gi_printf("%*s<o
6240: 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c 22 25 68 ption value=\"%h
6250: 5c 22 20 73 65 6c 65 63 74 65 64 3e 25 68 3c 2f \" selected>%h</
6260: 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 option>\n",.
6270: 20 20 20 69 6e 2b 32 2c 20 22 22 2c 20 7a 44 2c in+2, "", zD,
6280: 20 7a 44 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a zD);. }. }.
6290: 20 20 77 68 69 6c 65 28 20 28 7a 56 61 6c 20 3d while( (zVal =
62a0: 20 2a 28 61 7a 2b 2b 29 29 21 3d 30 20 20 29 7b *(az++))!=0 ){
62b0: 0a 20 20 20 20 63 6f 6e 73 74 20 63 68 61 72 20 . const char
62c0: 2a 7a 4e 61 6d 65 20 3d 20 2a 28 61 7a 2b 2b 29 *zName = *(az++)
62d0: 3b 0a 20 20 20 20 69 66 28 20 7a 4e 61 6d 65 5b ;. if( zName[
62e0: 30 5d 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 0] ){. cgi_
62f0: 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 printf("%*s<opti
6300: 6f 6e 20 76 61 6c 75 65 3d 5c 22 25 68 5c 22 25 on value=\"%h\"%
6310: 73 3e 25 68 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 s>%h</option>\n"
6320: 2c 0a 20 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 ,. in+2,
6330: 22 22 2c 0a 20 20 20 20 20 20 20 20 7a 56 61 6c "",. zVal
6340: 2c 0a 20 20 20 20 20 20 20 20 73 74 72 63 6d 70 ,. strcmp
6350: 28 7a 56 61 6c 2c 20 7a 44 29 20 3f 20 22 22 20 (zVal, zD) ? ""
6360: 3a 20 22 20 73 65 6c 65 63 74 65 64 22 2c 0a 20 : " selected",.
6370: 20 20 20 20 20 20 20 7a 4e 61 6d 65 0a 20 20 20 zName.
6380: 20 20 20 29 3b 0a 20 20 20 20 7d 65 6c 73 65 7b );. }else{
6390: 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69 6e 74 . cgi_print
63a0: 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 f("%*s<option va
63b0: 6c 75 65 3d 5c 22 25 68 5c 22 25 73 3e 26 6e 62 lue=\"%h\"%s>&nb
63c0: 73 70 3b 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c sp;</option>\n",
63d0: 0a 20 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 . in+2, "
63e0: 22 2c 0a 20 20 20 20 20 20 20 20 7a 56 61 6c 2c ",. zVal,
63f0: 0a 20 20 20 20 20 20 20 20 73 74 72 63 6d 70 28 . strcmp(
6400: 7a 56 61 6c 2c 20 7a 44 29 20 3f 20 22 22 20 3a zVal, zD) ? "" :
6410: 20 22 20 73 65 6c 65 63 74 65 64 22 0a 20 20 20 " selected".
6420: 20 20 20 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a );. }. }.
6430: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a cgi_printf("%*
6440: 73 3c 2f 73 65 6c 65 63 74 3e 5c 6e 22 2c 20 69 s</select>\n", i
6450: 6e 2c 20 22 22 29 3b 0a 7d 0a 0a 2f 2a 20 0a 2a n, "");.}../* .*
6460: 2a 20 54 68 69 73 20 66 75 6e 63 74 69 6f 6e 20 * This function
6470: 69 6d 70 6c 65 6d 65 6e 74 73 20 74 68 65 20 63 implements the c
6480: 61 6c 6c 62 61 63 6b 20 66 72 6f 6d 20 76 78 70 allback from vxp
6490: 72 69 6e 74 66 2e 20 0a 2a 2a 0a 2a 2a 20 54 68 rintf. .**.** Th
64a0: 69 73 20 72 6f 75 74 69 6e 65 20 73 65 6e 64 73 is routine sends
64b0: 20 6e 4e 65 77 43 68 61 72 20 63 68 61 72 61 63 nNewChar charac
64c0: 74 65 72 73 20 6f 66 20 74 65 78 74 20 69 6e 20 ters of text in
64d0: 7a 4e 65 77 54 65 78 74 20 74 6f 0a 2a 2a 20 43 zNewText to.** C
64e0: 47 49 20 72 65 70 6c 79 20 63 6f 6e 74 65 6e 74 GI reply content
64f0: 20 62 75 66 66 65 72 2e 0a 2a 2f 0a 73 74 61 74 buffer..*/.stat
6500: 69 63 20 76 6f 69 64 20 73 6f 75 74 28 76 6f 69 ic void sout(voi
6510: 64 20 2a 4e 6f 74 55 73 65 64 2c 20 63 6f 6e 73 d *NotUsed, cons
6520: 74 20 63 68 61 72 20 2a 7a 4e 65 77 54 65 78 74 t char *zNewText
6530: 2c 20 69 6e 74 20 6e 4e 65 77 43 68 61 72 29 7b , int nNewChar){
6540: 0a 20 20 63 67 69 5f 61 70 70 65 6e 64 5f 63 6f . cgi_append_co
6550: 6e 74 65 6e 74 28 7a 4e 65 77 54 65 78 74 2c 20 ntent(zNewText,
6560: 6e 4e 65 77 43 68 61 72 29 3b 0a 7d 0a 0a 2f 2a nNewChar);.}../*
6570: 0a 2a 2a 20 54 68 69 73 20 72 6f 75 74 69 6e 65 .** This routine
6580: 20 77 6f 72 6b 73 20 6c 69 6b 65 20 22 70 72 69 works like "pri
6590: 6e 74 66 22 20 65 78 63 65 70 74 20 74 68 61 74 ntf" except that
65a0: 20 69 74 20 68 61 73 20 74 68 65 0a 2a 2a 20 65 it has the.** e
65b0: 78 74 72 61 20 66 6f 72 6d 61 74 74 69 6e 67 20 xtra formatting
65c0: 63 61 70 61 62 69 6c 69 74 69 65 73 20 73 75 63 capabilities suc
65d0: 68 20 61 73 20 25 68 20 61 6e 64 20 25 74 2e 0a h as %h and %t..
65e0: 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 70 72 69 6e */.void cgi_prin
65f0: 74 66 28 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a tf(const char *z
6600: 46 6f 72 6d 61 74 2c 20 2e 2e 2e 29 7b 0a 20 20 Format, ...){.
6610: 76 61 5f 6c 69 73 74 20 61 70 3b 0a 20 20 76 61 va_list ap;. va
6620: 5f 73 74 61 72 74 28 61 70 2c 7a 46 6f 72 6d 61 _start(ap,zForma
6630: 74 29 3b 0a 20 20 76 78 70 72 69 6e 74 66 28 73 t);. vxprintf(s
6640: 6f 75 74 2c 30 2c 7a 46 6f 72 6d 61 74 2c 61 70 out,0,zFormat,ap
6650: 29 3b 0a 20 20 76 61 5f 65 6e 64 28 61 70 29 3b );. va_end(ap);
6660: 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 69 73 20 72 .}../*.** This r
6670: 6f 75 74 69 6e 65 20 77 6f 72 6b 73 20 6c 69 6b outine works lik
6680: 65 20 22 76 70 72 69 6e 74 66 22 20 65 78 63 65 e "vprintf" exce
6690: 70 74 20 74 68 61 74 20 69 74 20 68 61 73 20 74 pt that it has t
66a0: 68 65 0a 2a 2a 20 65 78 74 72 61 20 66 6f 72 6d he.** extra form
66b0: 61 74 74 69 6e 67 20 63 61 70 61 62 69 6c 69 74 atting capabilit
66c0: 69 65 73 20 73 75 63 68 20 61 73 20 25 68 20 61 ies such as %h a
66d0: 6e 64 20 25 74 2e 0a 2a 2f 0a 76 6f 69 64 20 63 nd %t..*/.void c
66e0: 67 69 5f 76 70 72 69 6e 74 66 28 63 6f 6e 73 74 gi_vprintf(const
66f0: 20 63 68 61 72 20 2a 7a 46 6f 72 6d 61 74 2c 20 char *zFormat,
6700: 76 61 5f 6c 69 73 74 20 61 70 29 7b 0a 20 20 76 va_list ap){. v
6710: 78 70 72 69 6e 74 66 28 73 6f 75 74 2c 30 2c 7a xprintf(sout,0,z
6720: 46 6f 72 6d 61 74 2c 61 70 29 3b 0a 7d 0a 0a 0a Format,ap);.}...
6730: 2f 2a 0a 2a 2a 20 53 65 6e 64 20 61 20 72 65 70 /*.** Send a rep
6740: 6c 79 20 69 6e 64 69 63 61 74 69 6e 67 20 74 68 ly indicating th
6750: 61 74 20 74 68 65 20 48 54 54 50 20 72 65 71 75 at the HTTP requ
6760: 65 73 74 20 77 61 73 20 6d 61 6c 66 6f 72 6d 65 est was malforme
6770: 64 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 d.*/.static void
6780: 20 6d 61 6c 66 6f 72 6d 65 64 5f 72 65 71 75 65 malformed_reque
6790: 73 74 28 76 6f 69 64 29 7b 0a 20 20 63 67 69 5f st(void){. cgi_
67a0: 73 65 74 5f 73 74 61 74 75 73 28 35 30 31 2c 20 set_status(501,
67b0: 22 4e 6f 74 20 49 6d 70 6c 65 6d 65 6e 74 65 64 "Not Implemented
67c0: 22 29 3b 0a 20 20 63 67 69 5f 70 72 69 6e 74 66 ");. cgi_printf
67d0: 28 0a 20 20 20 20 22 3c 68 74 6d 6c 3e 3c 62 6f (. "<html><bo
67e0: 64 79 3e 55 6e 72 65 63 6f 67 6e 69 7a 65 64 20 dy>Unrecognized
67f0: 48 54 54 50 20 52 65 71 75 65 73 74 3c 2f 62 6f HTTP Request</bo
6800: 64 79 3e 3c 2f 68 74 6d 6c 3e 5c 6e 22 0a 20 20 dy></html>\n".
6810: 29 3b 0a 20 20 63 67 69 5f 72 65 70 6c 79 28 29 );. cgi_reply()
6820: 3b 0a 20 20 65 78 69 74 28 30 29 3b 0a 7d 0a 0a ;. exit(0);.}..
6830: 2f 2a 0a 2a 2a 20 50 61 6e 69 63 20 61 6e 64 20 /*.** Panic and
6840: 64 69 65 20 77 68 69 6c 65 20 70 72 6f 63 65 73 die while proces
6850: 73 69 6e 67 20 61 20 77 65 62 70 61 67 65 2e 0a sing a webpage..
6860: 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 70 61 6e 69 */.void cgi_pani
6870: 63 28 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 46 c(const char *zF
6880: 6f 72 6d 61 74 2c 20 2e 2e 2e 29 7b 0a 20 20 76 ormat, ...){. v
6890: 61 5f 6c 69 73 74 20 61 70 3b 0a 20 20 63 67 69 a_list ap;. cgi
68a0: 5f 72 65 73 65 74 5f 63 6f 6e 74 65 6e 74 28 29 _reset_content()
68b0: 3b 0a 20 20 63 67 69 5f 73 65 74 5f 73 74 61 74 ;. cgi_set_stat
68c0: 75 73 28 35 30 30 2c 20 22 49 6e 74 65 72 6e 61 us(500, "Interna
68d0: 6c 20 53 65 72 76 65 72 20 45 72 72 6f 72 22 29 l Server Error")
68e0: 3b 0a 20 20 63 67 69 5f 70 72 69 6e 74 66 28 0a ;. cgi_printf(.
68f0: 20 20 20 20 22 3c 68 74 6d 6c 3e 3c 62 6f 64 79 "<html><body
6900: 3e 3c 68 31 3e 49 6e 74 65 72 6e 61 6c 20 53 65 ><h1>Internal Se
6910: 72 76 65 72 20 45 72 72 6f 72 3c 2f 68 31 3e 5c rver Error</h1>\
6920: 6e 22 0a 20 20 20 20 22 3c 70 6c 61 69 6e 74 65 n". "<plainte
6930: 78 74 3e 22 0a 20 20 29 3b 0a 20 20 76 61 5f 73 xt>". );. va_s
6940: 74 61 72 74 28 61 70 2c 20 7a 46 6f 72 6d 61 74 tart(ap, zFormat
6950: 29 3b 0a 20 20 76 78 70 72 69 6e 74 66 28 73 6f );. vxprintf(so
6960: 75 74 2c 30 2c 7a 46 6f 72 6d 61 74 2c 61 70 29 ut,0,zFormat,ap)
6970: 3b 0a 20 20 76 61 5f 65 6e 64 28 61 70 29 3b 0a ;. va_end(ap);.
6980: 20 20 63 67 69 5f 72 65 70 6c 79 28 29 3b 0a 20 cgi_reply();.
6990: 20 65 78 69 74 28 31 29 3b 0a 7d 0a 0a 2f 2a 0a exit(1);.}../*.
69a0: 2a 2a 20 52 65 6d 6f 76 65 20 74 68 65 20 66 69 ** Remove the fi
69b0: 72 73 74 20 73 70 61 63 65 2d 64 65 6c 69 6d 69 rst space-delimi
69c0: 74 65 64 20 74 6f 6b 65 6e 20 66 72 6f 6d 20 61 ted token from a
69d0: 20 73 74 72 69 6e 67 20 61 6e 64 20 72 65 74 75 string and retu
69e0: 72 6e 0a 2a 2a 20 61 20 70 6f 69 6e 74 65 72 20 rn.** a pointer
69f0: 74 6f 20 69 74 2e 20 20 41 64 64 20 61 20 4e 55 to it. Add a NU
6a00: 4c 4c 20 74 6f 20 74 68 65 20 73 74 72 69 6e 67 LL to the string
6a10: 20 74 6f 20 74 65 72 6d 69 6e 61 74 65 20 74 68 to terminate th
6a20: 65 20 74 6f 6b 65 6e 2e 0a 2a 2a 20 4d 61 6b 65 e token..** Make
6a30: 20 2a 7a 4c 65 66 74 4f 76 65 72 20 70 6f 69 6e *zLeftOver poin
6a40: 74 20 74 6f 20 74 68 65 20 73 74 61 72 74 20 6f t to the start o
6a50: 66 20 74 68 65 20 6e 65 78 74 20 74 6f 6b 65 6e f the next token
6a60: 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 63 68 61 72 ..*/.static char
6a70: 20 2a 65 78 74 72 61 63 74 5f 74 6f 6b 65 6e 28 *extract_token(
6a80: 63 68 61 72 20 2a 7a 49 6e 70 75 74 2c 20 63 68 char *zInput, ch
6a90: 61 72 20 2a 2a 7a 4c 65 66 74 4f 76 65 72 29 7b ar **zLeftOver){
6aa0: 0a 20 20 63 68 61 72 20 2a 7a 52 65 73 75 6c 74 . char *zResult
6ab0: 20 3d 20 30 3b 0a 20 20 69 66 28 20 7a 49 6e 70 = 0;. if( zInp
6ac0: 75 74 3d 3d 30 20 29 7b 0a 20 20 20 20 69 66 28 ut==0 ){. if(
6ad0: 20 7a 4c 65 66 74 4f 76 65 72 20 29 20 2a 7a 4c zLeftOver ) *zL
6ae0: 65 66 74 4f 76 65 72 20 3d 20 30 3b 0a 20 20 20 eftOver = 0;.
6af0: 20 72 65 74 75 72 6e 20 30 3b 0a 20 20 7d 0a 20 return 0;. }.
6b00: 20 77 68 69 6c 65 28 20 69 73 73 70 61 63 65 28 while( isspace(
6b10: 2a 7a 49 6e 70 75 74 29 20 29 7b 20 7a 49 6e 70 *zInput) ){ zInp
6b20: 75 74 2b 2b 3b 20 7d 0a 20 20 7a 52 65 73 75 6c ut++; }. zResul
6b30: 74 20 3d 20 7a 49 6e 70 75 74 3b 0a 20 20 77 68 t = zInput;. wh
6b40: 69 6c 65 28 20 2a 7a 49 6e 70 75 74 20 26 26 20 ile( *zInput &&
6b50: 21 69 73 73 70 61 63 65 28 2a 7a 49 6e 70 75 74 !isspace(*zInput
6b60: 29 20 29 7b 20 7a 49 6e 70 75 74 2b 2b 3b 20 7d ) ){ zInput++; }
6b70: 0a 20 20 69 66 28 20 2a 7a 49 6e 70 75 74 20 29 . if( *zInput )
6b80: 7b 0a 20 20 20 20 2a 7a 49 6e 70 75 74 20 3d 20 {. *zInput =
6b90: 30 3b 0a 20 20 20 20 7a 49 6e 70 75 74 2b 2b 3b 0;. zInput++;
6ba0: 0a 20 20 20 20 77 68 69 6c 65 28 20 69 73 73 70 . while( issp
6bb0: 61 63 65 28 2a 7a 49 6e 70 75 74 29 20 29 7b 20 ace(*zInput) ){
6bc0: 7a 49 6e 70 75 74 2b 2b 3b 20 7d 0a 20 20 7d 0a zInput++; }. }.
6bd0: 20 20 69 66 28 20 7a 4c 65 66 74 4f 76 65 72 20 if( zLeftOver
6be0: 29 7b 20 2a 7a 4c 65 66 74 4f 76 65 72 20 3d 20 ){ *zLeftOver =
6bf0: 7a 49 6e 70 75 74 3b 20 7d 0a 20 20 72 65 74 75 zInput; }. retu
6c00: 72 6e 20 7a 52 65 73 75 6c 74 3b 0a 7d 0a 0a 2f rn zResult;.}../
6c10: 2a 0a 2a 2a 20 54 68 69 73 20 72 6f 75 74 69 6e *.** This routin
6c20: 65 20 68 61 6e 64 6c 65 73 20 61 20 73 69 6e 67 e handles a sing
6c30: 6c 65 20 48 54 54 50 20 72 65 71 75 65 73 74 20 le HTTP request
6c40: 77 68 69 63 68 20 69 73 20 63 6f 6d 69 6e 67 20 which is coming
6c50: 69 6e 20 6f 6e 0a 2a 2a 20 73 74 61 6e 64 61 72 in on.** standar
6c60: 64 20 69 6e 70 75 74 20 61 6e 64 20 77 68 69 63 d input and whic
6c70: 68 20 72 65 70 6c 69 65 73 20 6f 6e 20 73 74 61 h replies on sta
6c80: 6e 64 61 72 64 20 6f 75 74 70 75 74 2e 0a 2a 2a ndard output..**
6c90: 0a 2a 2a 20 54 68 65 20 48 54 54 50 20 72 65 71 .** The HTTP req
6ca0: 75 65 73 74 20 69 73 20 72 65 61 64 20 66 72 6f uest is read fro
6cb0: 6d 20 73 74 61 6e 64 61 72 64 20 69 6e 70 75 74 m standard input
6cc0: 20 61 6e 64 20 69 73 20 75 73 65 64 20 74 6f 20 and is used to
6cd0: 69 6e 69 74 69 61 6c 69 7a 65 0a 2a 2a 20 65 6e initialize.** en
6ce0: 76 69 72 6f 6e 6d 65 6e 74 20 76 61 72 69 61 62 vironment variab
6cf0: 6c 65 73 20 61 73 20 70 65 72 20 43 47 49 2e 20 les as per CGI.
6d00: 20 54 68 65 20 63 67 69 5f 69 6e 69 74 28 29 20 The cgi_init()
6d10: 72 6f 75 74 69 6e 65 20 74 6f 20 63 6f 6d 70 6c routine to compl
6d20: 65 74 65 0a 2a 2a 20 74 68 65 20 73 65 74 75 70 ete.** the setup
6d30: 2e 20 20 4f 6e 63 65 20 61 6c 6c 20 74 68 65 20 . Once all the
6d40: 73 65 74 75 70 20 69 73 20 66 69 6e 69 73 68 65 setup is finishe
6d50: 64 2c 20 74 68 69 73 20 70 72 6f 63 65 64 75 72 d, this procedur
6d60: 65 20 72 65 74 75 72 6e 73 0a 2a 2a 20 61 6e 64 e returns.** and
6d70: 20 73 75 62 73 65 71 75 65 6e 74 20 63 6f 64 65 subsequent code
6d80: 20 68 61 6e 64 6c 65 73 20 74 68 65 20 61 63 74 handles the act
6d90: 75 61 6c 20 67 65 6e 65 72 61 74 69 6f 6e 20 6f ual generation o
6da0: 66 20 74 68 65 20 77 65 62 70 61 67 65 2e 0a 2a f the webpage..*
6db0: 2f 0a 76 6f 69 64 20 63 67 69 5f 68 61 6e 64 6c /.void cgi_handl
6dc0: 65 5f 68 74 74 70 5f 72 65 71 75 65 73 74 28 76 e_http_request(v
6dd0: 6f 69 64 29 7b 0a 20 20 63 68 61 72 20 2a 7a 2c oid){. char *z,
6de0: 20 2a 7a 54 6f 6b 65 6e 3b 0a 20 20 69 6e 74 20 *zToken;. int
6df0: 69 3b 0a 20 20 73 74 72 75 63 74 20 73 6f 63 6b i;. struct sock
6e00: 61 64 64 72 5f 69 6e 20 72 65 6d 6f 74 65 4e 61 addr_in remoteNa
6e10: 6d 65 3b 0a 20 20 73 69 7a 65 5f 74 20 73 69 7a me;. size_t siz
6e20: 65 20 3d 20 73 69 7a 65 6f 66 28 73 74 72 75 63 e = sizeof(struc
6e30: 74 20 73 6f 63 6b 61 64 64 72 5f 69 6e 29 3b 0a t sockaddr_in);.
6e40: 20 20 63 68 61 72 20 7a 4c 69 6e 65 5b 32 30 30 char zLine[200
6e50: 30 5d 3b 20 20 20 20 20 2f 2a 20 41 20 73 69 6e 0]; /* A sin
6e60: 67 6c 65 20 6c 69 6e 65 20 6f 66 20 69 6e 70 75 gle line of inpu
6e70: 74 2e 20 2a 2f 0a 0a 20 20 66 75 6c 6c 48 74 74 t. */.. fullHtt
6e80: 70 52 65 70 6c 79 20 3d 20 31 3b 0a 20 20 69 66 pReply = 1;. if
6e90: 28 20 66 67 65 74 73 28 7a 4c 69 6e 65 2c 20 73 ( fgets(zLine, s
6ea0: 69 7a 65 6f 66 28 7a 4c 69 6e 65 29 2c 20 73 74 izeof(zLine), st
6eb0: 64 69 6e 29 3d 3d 30 20 29 7b 0a 20 20 20 20 6d din)==0 ){. m
6ec0: 61 6c 66 6f 72 6d 65 64 5f 72 65 71 75 65 73 74 alformed_request
6ed0: 28 29 3b 0a 20 20 7d 0a 20 20 7a 54 6f 6b 65 6e ();. }. zToken
6ee0: 20 3d 20 65 78 74 72 61 63 74 5f 74 6f 6b 65 6e = extract_token
6ef0: 28 7a 4c 69 6e 65 2c 20 26 7a 29 3b 0a 20 20 69 (zLine, &z);. i
6f00: 66 28 20 7a 54 6f 6b 65 6e 3d 3d 30 20 29 7b 0a f( zToken==0 ){.
6f10: 20 20 20 20 6d 61 6c 66 6f 72 6d 65 64 5f 72 65 malformed_re
6f20: 71 75 65 73 74 28 29 3b 0a 20 20 7d 0a 20 20 69 quest();. }. i
6f30: 66 28 20 73 74 72 63 6d 70 28 7a 54 6f 6b 65 6e f( strcmp(zToken
6f40: 2c 22 47 45 54 22 29 21 3d 30 20 26 26 20 73 74 ,"GET")!=0 && st
6f50: 72 63 6d 70 28 7a 54 6f 6b 65 6e 2c 22 50 4f 53 rcmp(zToken,"POS
6f60: 54 22 29 21 3d 30 0a 20 20 20 20 20 20 26 26 20 T")!=0. &&
6f70: 73 74 72 63 6d 70 28 7a 54 6f 6b 65 6e 2c 22 48 strcmp(zToken,"H
6f80: 45 41 44 22 29 21 3d 30 20 29 7b 0a 20 20 20 20 EAD")!=0 ){.
6f90: 6d 61 6c 66 6f 72 6d 65 64 5f 72 65 71 75 65 73 malformed_reques
6fa0: 74 28 29 3b 0a 20 20 7d 0a 20 20 63 67 69 5f 73 t();. }. cgi_s
6fb0: 65 74 65 6e 76 28 22 47 41 54 45 57 41 59 5f 49 etenv("GATEWAY_I
6fc0: 4e 54 45 52 46 41 43 45 22 2c 22 43 47 49 2f 31 NTERFACE","CGI/1
6fd0: 2e 30 22 29 3b 0a 20 20 63 67 69 5f 73 65 74 65 .0");. cgi_sete
6fe0: 6e 76 28 22 52 45 51 55 45 53 54 5f 4d 45 54 48 nv("REQUEST_METH
6ff0: 4f 44 22 2c 7a 54 6f 6b 65 6e 29 3b 0a 20 20 7a OD",zToken);. z
7000: 54 6f 6b 65 6e 20 3d 20 65 78 74 72 61 63 74 5f Token = extract_
7010: 74 6f 6b 65 6e 28 7a 2c 20 26 7a 29 3b 0a 20 20 token(z, &z);.
7020: 69 66 28 20 7a 54 6f 6b 65 6e 3d 3d 30 20 29 7b if( zToken==0 ){
7030: 0a 20 20 20 20 6d 61 6c 66 6f 72 6d 65 64 5f 72 . malformed_r
7040: 65 71 75 65 73 74 28 29 3b 0a 20 20 7d 0a 20 20 equest();. }.
7050: 63 67 69 5f 73 65 74 65 6e 76 28 22 52 45 51 55 cgi_setenv("REQU
7060: 45 53 54 5f 55 52 49 22 2c 20 7a 54 6f 6b 65 6e EST_URI", zToken
7070: 29 3b 0a 20 20 66 6f 72 28 69 3d 30 3b 20 7a 54 );. for(i=0; zT
7080: 6f 6b 65 6e 5b 69 5d 20 26 26 20 7a 54 6f 6b 65 oken[i] && zToke
7090: 6e 5b 69 5d 21 3d 27 3f 27 3b 20 69 2b 2b 29 7b n[i]!='?'; i++){
70a0: 7d 0a 20 20 69 66 28 20 7a 54 6f 6b 65 6e 5b 69 }. if( zToken[i
70b0: 5d 20 29 20 7a 54 6f 6b 65 6e 5b 69 2b 2b 5d 20 ] ) zToken[i++]
70c0: 3d 20 30 3b 0a 20 20 63 67 69 5f 73 65 74 65 6e = 0;. cgi_seten
70d0: 76 28 22 50 41 54 48 5f 49 4e 46 4f 22 2c 20 7a v("PATH_INFO", z
70e0: 54 6f 6b 65 6e 29 3b 0a 20 20 63 67 69 5f 73 65 Token);. cgi_se
70f0: 74 65 6e 76 28 22 51 55 45 52 59 5f 53 54 52 49 tenv("QUERY_STRI
7100: 4e 47 22 2c 20 26 7a 54 6f 6b 65 6e 5b 69 5d 29 NG", &zToken[i])
7110: 3b 0a 20 20 69 66 28 20 67 65 74 70 65 65 72 6e ;. if( getpeern
7120: 61 6d 65 28 66 69 6c 65 6e 6f 28 73 74 64 69 6e ame(fileno(stdin
7130: 29 2c 20 28 73 74 72 75 63 74 20 73 6f 63 6b 61 ), (struct socka
7140: 64 64 72 2a 29 26 72 65 6d 6f 74 65 4e 61 6d 65 ddr*)&remoteName
7150: 2c 20 26 73 69 7a 65 29 3e 3d 30 20 29 7b 0a 20 , &size)>=0 ){.
7160: 20 20 20 63 67 69 5f 73 65 74 65 6e 76 28 22 52 cgi_setenv("R
7170: 45 4d 4f 54 45 5f 41 44 44 52 22 2c 20 69 6e 65 EMOTE_ADDR", ine
7180: 74 5f 6e 74 6f 61 28 72 65 6d 6f 74 65 4e 61 6d t_ntoa(remoteNam
7190: 65 2e 73 69 6e 5f 61 64 64 72 29 29 3b 0a 20 20 e.sin_addr));.
71a0: 7d 0a 20 0a 20 20 2f 2a 20 47 65 74 20 61 6c 6c }. . /* Get all
71b0: 20 74 68 65 20 6f 70 74 69 6f 6e 61 6c 20 66 69 the optional fi
71c0: 65 6c 64 73 20 74 68 61 74 20 66 6f 6c 6c 6f 77 elds that follow
71d0: 20 74 68 65 20 66 69 72 73 74 20 6c 69 6e 65 2e the first line.
71e0: 0a 20 20 2a 2f 0a 20 20 77 68 69 6c 65 28 20 66 . */. while( f
71f0: 67 65 74 73 28 7a 4c 69 6e 65 2c 73 69 7a 65 6f gets(zLine,sizeo
7200: 66 28 7a 4c 69 6e 65 29 2c 73 74 64 69 6e 29 20 f(zLine),stdin)
7210: 29 7b 0a 20 20 20 20 63 68 61 72 20 2a 7a 46 69 ){. char *zFi
7220: 65 6c 64 4e 61 6d 65 3b 0a 20 20 20 20 63 68 61 eldName;. cha
7230: 72 20 2a 7a 56 61 6c 3b 0a 0a 20 20 20 20 7a 46 r *zVal;.. zF
7240: 69 65 6c 64 4e 61 6d 65 20 3d 20 65 78 74 72 61 ieldName = extra
7250: 63 74 5f 74 6f 6b 65 6e 28 7a 4c 69 6e 65 2c 26 ct_token(zLine,&
7260: 7a 56 61 6c 29 3b 0a 20 20 20 20 69 66 28 20 7a zVal);. if( z
7270: 46 69 65 6c 64 4e 61 6d 65 3d 3d 30 20 7c 7c 20 FieldName==0 ||
7280: 2a 7a 46 69 65 6c 64 4e 61 6d 65 3d 3d 30 20 29 *zFieldName==0 )
7290: 20 62 72 65 61 6b 3b 0a 20 20 20 20 77 68 69 6c break;. whil
72a0: 65 28 20 69 73 73 70 61 63 65 28 2a 7a 56 61 6c e( isspace(*zVal
72b0: 29 20 29 7b 20 7a 56 61 6c 2b 2b 3b 20 7d 0a 20 ) ){ zVal++; }.
72c0: 20 20 20 69 20 3d 20 73 74 72 6c 65 6e 28 7a 56 i = strlen(zV
72d0: 61 6c 29 3b 0a 20 20 20 20 77 68 69 6c 65 28 20 al);. while(
72e0: 69 3e 30 20 26 26 20 69 73 73 70 61 63 65 28 7a i>0 && isspace(z
72f0: 56 61 6c 5b 69 2d 31 5d 29 20 29 7b 20 69 2d 2d Val[i-1]) ){ i--
7300: 3b 20 7d 0a 20 20 20 20 7a 56 61 6c 5b 69 5d 20 ; }. zVal[i]
7310: 3d 20 30 3b 0a 20 20 20 20 66 6f 72 28 69 3d 30 = 0;. for(i=0
7320: 3b 20 7a 46 69 65 6c 64 4e 61 6d 65 5b 69 5d 3b ; zFieldName[i];
7330: 20 69 2b 2b 29 7b 20 7a 46 69 65 6c 64 4e 61 6d i++){ zFieldNam
7340: 65 5b 69 5d 20 3d 20 74 6f 6c 6f 77 65 72 28 7a e[i] = tolower(z
7350: 46 69 65 6c 64 4e 61 6d 65 5b 69 5d 29 3b 20 7d FieldName[i]); }
7360: 0a 20 20 20 20 69 66 28 20 73 74 72 63 6d 70 28 . if( strcmp(
7370: 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 75 73 65 72 zFieldName,"user
7380: 2d 61 67 65 6e 74 3a 22 29 3d 3d 30 20 29 7b 0a -agent:")==0 ){.
7390: 20 20 20 20 20 20 63 67 69 5f 73 65 74 65 6e 76 cgi_setenv
73a0: 28 22 48 54 54 50 5f 55 53 45 52 5f 41 47 45 4e ("HTTP_USER_AGEN
73b0: 54 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d T", zVal);. }
73c0: 65 6c 73 65 20 69 66 28 20 73 74 72 63 6d 70 28 else if( strcmp(
73d0: 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 63 6f 6e 74 zFieldName,"cont
73e0: 65 6e 74 2d 6c 65 6e 67 74 68 3a 22 29 3d 3d 30 ent-length:")==0
73f0: 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 65 ){. cgi_se
7400: 74 65 6e 76 28 22 43 4f 4e 54 45 4e 54 5f 4c 45 tenv("CONTENT_LE
7410: 4e 47 54 48 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 NGTH", zVal);.
7420: 20 20 7d 65 6c 73 65 20 69 66 28 20 73 74 72 63 }else if( strc
7430: 6d 70 28 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 72 mp(zFieldName,"r
7440: 65 66 65 72 65 72 3a 22 29 3d 3d 30 20 29 7b 0a eferer:")==0 ){.
7450: 20 20 20 20 20 20 63 67 69 5f 73 65 74 65 6e 76 cgi_setenv
7460: 28 22 48 54 54 50 5f 52 45 46 45 52 45 52 22 2c ("HTTP_REFERER",
7470: 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d 65 6c 73 zVal);. }els
7480: 65 20 69 66 28 20 73 74 72 63 6d 70 28 7a 46 69 e if( strcmp(zFi
7490: 65 6c 64 4e 61 6d 65 2c 22 68 6f 73 74 3a 22 29 eldName,"host:")
74a0: 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 63 67 69 ==0 ){. cgi
74b0: 5f 73 65 74 65 6e 76 28 22 48 54 54 50 5f 48 4f _setenv("HTTP_HO
74c0: 53 54 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 ST", zVal);.
74d0: 7d 65 6c 73 65 20 69 66 28 20 73 74 72 63 6d 70 }else if( strcmp
74e0: 28 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 63 6f 6e (zFieldName,"con
74f0: 74 65 6e 74 2d 74 79 70 65 3a 22 29 3d 3d 30 20 tent-type:")==0
7500: 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 65 74 ){. cgi_set
7510: 65 6e 76 28 22 43 4f 4e 54 45 4e 54 5f 54 59 50 env("CONTENT_TYP
7520: 45 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d E", zVal);. }
7530: 65 6c 73 65 20 69 66 28 20 73 74 72 63 6d 70 28 else if( strcmp(
7540: 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 63 6f 6f 6b zFieldName,"cook
7550: 69 65 3a 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 ie:")==0 ){.
7560: 20 20 63 67 69 5f 73 65 74 65 6e 76 28 22 48 54 cgi_setenv("HT
7570: 54 50 5f 43 4f 4f 4b 49 45 22 2c 20 7a 56 61 6c TP_COOKIE", zVal
7580: 29 3b 0a 20 20 20 20 7d 65 6c 73 65 20 69 66 28 );. }else if(
7590: 20 73 74 72 63 6d 70 28 7a 46 69 65 6c 64 4e 61 strcmp(zFieldNa
75a0: 6d 65 2c 22 69 66 2d 6e 6f 6e 65 2d 6d 61 74 63 me,"if-none-matc
75b0: 68 3a 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 h:")==0 ){.
75c0: 20 63 67 69 5f 73 65 74 65 6e 76 28 22 48 54 54 cgi_setenv("HTT
75d0: 50 5f 49 46 5f 4e 4f 4e 45 5f 4d 41 54 43 48 22 P_IF_NONE_MATCH"
75e0: 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d 65 6c , zVal);. }el
75f0: 73 65 20 69 66 28 20 73 74 72 63 6d 70 28 7a 46 se if( strcmp(zF
7600: 69 65 6c 64 4e 61 6d 65 2c 22 69 66 2d 6d 6f 64 ieldName,"if-mod
7610: 69 66 69 65 64 2d 73 69 6e 63 65 3a 22 29 3d 3d ified-since:")==
7620: 30 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 0 ){. cgi_s
7630: 65 74 65 6e 76 28 22 48 54 54 50 5f 49 46 5f 4d etenv("HTTP_IF_M
7640: 4f 44 49 46 49 45 44 5f 53 49 4e 43 45 22 2c 20 ODIFIED_SINCE",
7650: 7a 56 61 6c 29 3b 0a 20 20 20 20 7d 0a 20 20 7d zVal);. }. }
7660: 0a 0a 20 20 63 67 69 5f 69 6e 69 74 28 29 3b 0a .. cgi_init();.
7670: 7d 0a 0a 2f 2a 0a 2a 2a 20 4d 61 78 69 6d 75 6d }../*.** Maximum
7680: 20 6e 75 6d 62 65 72 20 6f 66 20 63 68 69 6c 64 number of child
7690: 20 70 72 6f 63 65 73 73 65 73 20 74 68 61 74 20 processes that
76a0: 77 65 20 63 61 6e 20 68 61 76 65 20 72 75 6e 6e we can have runn
76b0: 69 6e 67 0a 2a 2a 20 61 74 20 6f 6e 65 20 74 69 ing.** at one ti
76c0: 6d 65 20 62 65 66 6f 72 65 20 77 65 20 73 74 61 me before we sta
76d0: 72 74 20 73 6c 6f 77 69 6e 67 20 74 68 69 6e 67 rt slowing thing
76e0: 73 20 64 6f 77 6e 2e 0a 2a 2f 0a 23 64 65 66 69 s down..*/.#defi
76f0: 6e 65 20 4d 41 58 5f 50 41 52 41 4c 4c 45 4c 20 ne MAX_PARALLEL
7700: 32 0a 0a 2f 2a 0a 2a 2a 20 49 6d 70 6c 65 6d 65 2../*.** Impleme
7710: 6e 74 20 61 6e 20 48 54 54 50 20 73 65 72 76 65 nt an HTTP serve
7720: 72 20 64 61 65 6d 6f 6e 20 6c 69 73 74 65 6e 69 r daemon listeni
7730: 6e 67 20 6f 6e 20 70 6f 72 74 20 69 50 6f 72 74 ng on port iPort
7740: 2e 0a 2a 2a 0a 2a 2a 20 41 73 20 6e 65 77 20 63 ..**.** As new c
7750: 6f 6e 6e 65 63 74 69 6f 6e 73 20 61 72 72 69 76 onnections arriv
7760: 65 2c 20 66 6f 72 6b 20 61 20 63 68 69 6c 64 20 e, fork a child
7770: 61 6e 64 20 6c 65 74 20 63 68 69 6c 64 20 72 65 and let child re
7780: 74 75 72 6e 0a 2a 2a 20 6f 75 74 20 6f 66 20 74 turn.** out of t
7790: 68 69 73 20 70 72 6f 63 65 64 75 72 65 20 63 61 his procedure ca
77a0: 6c 6c 2e 20 20 54 68 65 20 63 68 69 6c 64 20 77 ll. The child w
77b0: 69 6c 6c 20 68 61 6e 64 6c 65 20 74 68 65 20 72 ill handle the r
77c0: 65 71 75 65 73 74 2e 0a 2a 2a 20 54 68 65 20 70 equest..** The p
77d0: 61 72 65 6e 74 20 6e 65 76 65 72 20 72 65 74 75 arent never retu
77e0: 72 6e 73 20 66 72 6f 6d 20 74 68 69 73 20 70 72 rns from this pr
77f0: 6f 63 65 64 75 72 65 2e 0a 2a 2f 0a 76 6f 69 64 ocedure..*/.void
7800: 20 63 67 69 5f 68 74 74 70 5f 73 65 72 76 65 72 cgi_http_server
7810: 28 69 6e 74 20 69 50 6f 72 74 29 7b 0a 20 20 69 (int iPort){. i
7820: 6e 74 20 6c 69 73 74 65 6e 65 72 3b 20 20 20 20 nt listener;
7830: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 54 /* T
7840: 68 65 20 73 65 72 76 65 72 20 73 6f 63 6b 65 74 he server socket
7850: 20 2a 2f 0a 20 20 69 6e 74 20 63 6f 6e 6e 65 63 */. int connec
7860: 74 69 6f 6e 3b 20 20 20 20 20 20 20 20 20 20 20 tion;
7870: 20 20 20 2f 2a 20 41 20 73 6f 63 6b 65 74 20 66 /* A socket f
7880: 6f 72 20 65 61 63 68 20 69 6e 64 69 76 69 64 75 or each individu
7890: 61 6c 20 63 6f 6e 6e 65 63 74 69 6f 6e 20 2a 2f al connection */
78a0: 0a 20 20 66 64 5f 73 65 74 20 72 65 61 64 66 64 . fd_set readfd
78b0: 73 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 s;
78c0: 2f 2a 20 53 65 74 20 6f 66 20 66 69 6c 65 20 64 /* Set of file d
78d0: 65 73 63 72 69 70 74 6f 72 73 20 66 6f 72 20 73 escriptors for s
78e0: 65 6c 65 63 74 28 29 20 2a 2f 0a 20 20 73 69 7a elect() */. siz
78f0: 65 5f 74 20 6c 65 6e 61 64 64 72 3b 20 20 20 20 e_t lenaddr;
7900: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4c 65 6e /* Len
7910: 67 74 68 20 6f 66 20 74 68 65 20 69 6e 61 64 64 gth of the inadd
7920: 72 20 73 74 72 75 63 74 75 72 65 20 2a 2f 0a 20 r structure */.
7930: 20 69 6e 74 20 63 68 69 6c 64 3b 20 20 20 20 20 int child;
7940: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a /*
7950: 20 50 49 44 20 6f 66 20 74 68 65 20 63 68 69 6c PID of the chil
7960: 64 20 70 72 6f 63 65 73 73 20 2a 2f 0a 20 20 69 d process */. i
7970: 6e 74 20 6e 63 68 69 6c 64 72 65 6e 20 3d 20 30 nt nchildren = 0
7980: 3b 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4e ; /* N
7990: 75 6d 62 65 72 20 6f 66 20 63 68 69 6c 64 20 70 umber of child p
79a0: 72 6f 63 65 73 73 65 73 20 2a 2f 0a 20 20 73 74 rocesses */. st
79b0: 72 75 63 74 20 74 69 6d 65 76 61 6c 20 64 65 6c ruct timeval del
79c0: 61 79 3b 20 20 20 20 20 20 20 20 2f 2a 20 48 6f ay; /* Ho
79d0: 77 20 6c 6f 6e 67 20 74 6f 20 77 61 69 74 20 69 w long to wait i
79e0: 6e 73 69 64 65 20 73 65 6c 65 63 74 28 29 20 2a nside select() *
79f0: 2f 0a 20 20 73 74 72 75 63 74 20 73 6f 63 6b 61 /. struct socka
7a00: 64 64 72 5f 69 6e 20 69 6e 61 64 64 72 3b 20 20 ddr_in inaddr;
7a10: 20 2f 2a 20 54 68 65 20 73 6f 63 6b 65 74 20 61 /* The socket a
7a20: 64 64 72 65 73 73 20 2a 2f 0a 20 20 69 6e 74 20 ddress */. int
7a30: 6f 70 74 20 3d 20 31 3b 20 20 20 20 20 20 20 20 opt = 1;
7a40: 20 20 20 20 20 20 20 20 20 2f 2a 20 73 65 74 73 /* sets
7a50: 6f 63 6b 6f 70 74 20 66 6c 61 67 20 2a 2f 0a 0a ockopt flag */..
7a60: 20 20 6d 65 6d 73 65 74 28 26 69 6e 61 64 64 72 memset(&inaddr
7a70: 2c 20 30 2c 20 73 69 7a 65 6f 66 28 69 6e 61 64 , 0, sizeof(inad
7a80: 64 72 29 29 3b 0a 20 20 69 6e 61 64 64 72 2e 73 dr));. inaddr.s
7a90: 69 6e 5f 66 61 6d 69 6c 79 20 3d 20 41 46 5f 49 in_family = AF_I
7aa0: 4e 45 54 3b 0a 20 20 69 6e 61 64 64 72 2e 73 69 NET;. inaddr.si
7ab0: 6e 5f 61 64 64 72 2e 73 5f 61 64 64 72 20 3d 20 n_addr.s_addr =
7ac0: 49 4e 41 44 44 52 5f 41 4e 59 3b 0a 20 20 69 6e INADDR_ANY;. in
7ad0: 61 64 64 72 2e 73 69 6e 5f 70 6f 72 74 20 3d 20 addr.sin_port =
7ae0: 68 74 6f 6e 73 28 69 50 6f 72 74 29 3b 0a 20 20 htons(iPort);.
7af0: 6c 69 73 74 65 6e 65 72 20 3d 20 73 6f 63 6b 65 listener = socke
7b00: 74 28 41 46 5f 49 4e 45 54 2c 20 53 4f 43 4b 5f t(AF_INET, SOCK_
7b10: 53 54 52 45 41 4d 2c 20 30 29 3b 0a 20 20 69 66 STREAM, 0);. if
7b20: 28 20 6c 69 73 74 65 6e 65 72 3c 30 20 29 7b 0a ( listener<0 ){.
7b30: 20 20 20 20 66 70 72 69 6e 74 66 28 73 74 64 65 fprintf(stde
7b40: 72 72 2c 22 43 61 6e 27 74 20 63 72 65 61 74 65 rr,"Can't create
7b50: 20 61 20 73 6f 63 6b 65 74 5c 6e 22 29 3b 0a 20 a socket\n");.
7b60: 20 20 20 65 78 69 74 28 31 29 3b 0a 20 20 7d 0a exit(1);. }.
7b70: 0a 20 20 2f 2a 20 69 66 20 77 65 20 63 61 6e 27 . /* if we can'
7b80: 74 20 74 65 72 6d 69 6e 61 74 65 20 6e 69 63 65 t terminate nice
7b90: 6c 79 2c 20 61 74 20 6c 65 61 73 74 20 61 6c 6c ly, at least all
7ba0: 6f 77 20 74 68 65 20 73 6f 63 6b 65 74 20 74 6f ow the socket to
7bb0: 20 62 65 20 72 65 75 73 65 64 20 2a 2f 0a 20 20 be reused */.
7bc0: 73 65 74 73 6f 63 6b 6f 70 74 28 6c 69 73 74 65 setsockopt(liste
7bd0: 6e 65 72 2c 53 4f 4c 5f 53 4f 43 4b 45 54 2c 53 ner,SOL_SOCKET,S
7be0: 4f 5f 52 45 55 53 45 41 44 44 52 2c 26 6f 70 74 O_REUSEADDR,&opt
7bf0: 2c 73 69 7a 65 6f 66 28 6f 70 74 29 29 3b 0a 0a ,sizeof(opt));..
7c00: 20 20 69 66 28 20 62 69 6e 64 28 6c 69 73 74 65 if( bind(liste
7c10: 6e 65 72 2c 20 28 73 74 72 75 63 74 20 73 6f 63 ner, (struct soc
7c20: 6b 61 64 64 72 2a 29 26 69 6e 61 64 64 72 2c 20 kaddr*)&inaddr,
7c30: 73 69 7a 65 6f 66 28 69 6e 61 64 64 72 29 29 3c sizeof(inaddr))<
7c40: 30 20 29 7b 0a 20 20 20 20 66 70 72 69 6e 74 66 0 ){. fprintf
7c50: 28 73 74 64 65 72 72 2c 22 43 61 6e 27 74 20 62 (stderr,"Can't b
7c60: 69 6e 64 20 74 6f 20 70 6f 72 74 20 25 64 5c 6e ind to port %d\n
7c70: 22 2c 20 69 50 6f 72 74 29 3b 0a 20 20 20 20 65 ", iPort);. e
7c80: 78 69 74 28 31 29 3b 0a 20 20 7d 0a 20 20 6c 69 xit(1);. }. li
7c90: 73 74 65 6e 28 6c 69 73 74 65 6e 65 72 2c 31 30 sten(listener,10
7ca0: 29 3b 0a 20 20 77 68 69 6c 65 28 20 31 20 29 7b );. while( 1 ){
7cb0: 0a 20 20 20 20 69 66 28 20 6e 63 68 69 6c 64 72 . if( nchildr
7cc0: 65 6e 3e 4d 41 58 5f 50 41 52 41 4c 4c 45 4c 20 en>MAX_PARALLEL
7cd0: 29 7b 0a 20 20 20 20 20 20 2f 2a 20 53 6c 6f 77 ){. /* Slow
7ce0: 20 64 6f 77 6e 20 69 66 20 63 6f 6e 6e 65 63 74 down if connect
7cf0: 69 6f 6e 73 20 61 72 65 20 61 72 72 69 76 69 6e ions are arrivin
7d00: 67 20 74 6f 6f 20 66 61 73 74 20 2a 2f 0a 20 20 g too fast */.
7d10: 20 20 20 20 73 6c 65 65 70 28 20 6e 63 68 69 6c sleep( nchil
7d20: 64 72 65 6e 2d 4d 41 58 5f 50 41 52 41 4c 4c 45 dren-MAX_PARALLE
7d30: 4c 20 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20 64 L );. }. d
7d40: 65 6c 61 79 2e 74 76 5f 73 65 63 20 3d 20 36 30 elay.tv_sec = 60
7d50: 3b 0a 20 20 20 20 64 65 6c 61 79 2e 74 76 5f 75 ;. delay.tv_u
7d60: 73 65 63 20 3d 20 30 3b 0a 20 20 20 20 46 44 5f sec = 0;. FD_
7d70: 5a 45 52 4f 28 26 72 65 61 64 66 64 73 29 3b 0a ZERO(&readfds);.
7d80: 20 20 20 20 46 44 5f 53 45 54 28 20 6c 69 73 74 FD_SET( list
7d90: 65 6e 65 72 2c 20 26 72 65 61 64 66 64 73 29 3b ener, &readfds);
7da0: 0a 20 20 20 20 69 66 28 20 73 65 6c 65 63 74 28 . if( select(
7db0: 20 6c 69 73 74 65 6e 65 72 2b 31 2c 20 26 72 65 listener+1, &re
7dc0: 61 64 66 64 73 2c 20 30 2c 20 30 2c 20 26 64 65 adfds, 0, 0, &de
7dd0: 6c 61 79 29 20 29 7b 0a 20 20 20 20 20 20 6c 65 lay) ){. le
7de0: 6e 61 64 64 72 20 3d 20 73 69 7a 65 6f 66 28 69 naddr = sizeof(i
7df0: 6e 61 64 64 72 29 3b 0a 20 20 20 20 20 20 63 6f naddr);. co
7e00: 6e 6e 65 63 74 69 6f 6e 20 3d 20 61 63 63 65 70 nnection = accep
7e10: 74 28 6c 69 73 74 65 6e 65 72 2c 20 28 73 74 72 t(listener, (str
7e20: 75 63 74 20 73 6f 63 6b 61 64 64 72 2a 29 26 69 uct sockaddr*)&i
7e30: 6e 61 64 64 72 2c 20 26 6c 65 6e 61 64 64 72 29 naddr, &lenaddr)
7e40: 3b 0a 20 20 20 20 20 20 69 66 28 20 63 6f 6e 6e ;. if( conn
7e50: 65 63 74 69 6f 6e 3e 3d 30 20 29 7b 0a 20 20 20 ection>=0 ){.
7e60: 20 20 20 20 20 63 68 69 6c 64 20 3d 20 66 6f 72 child = for
7e70: 6b 28 29 3b 0a 20 20 20 20 20 20 20 20 69 66 28 k();. if(
7e80: 20 63 68 69 6c 64 21 3d 30 20 29 7b 0a 20 20 20 child!=0 ){.
7e90: 20 20 20 20 20 20 20 69 66 28 20 63 68 69 6c 64 if( child
7ea0: 3e 30 20 29 20 6e 63 68 69 6c 64 72 65 6e 2b 2b >0 ) nchildren++
7eb0: 3b 0a 20 20 20 20 20 20 20 20 20 20 63 6c 6f 73 ;. clos
7ec0: 65 28 63 6f 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 e(connection);.
7ed0: 20 20 20 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 }else{.
7ee0: 20 20 20 20 20 20 20 20 63 6c 6f 73 65 28 30 29 close(0)
7ef0: 3b 0a 20 20 20 20 20 20 20 20 20 20 64 75 70 28 ;. dup(
7f00: 63 6f 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 20 20 connection);.
7f10: 20 20 20 20 20 20 20 63 6c 6f 73 65 28 31 29 3b close(1);
7f20: 0a 20 20 20 20 20 20 20 20 20 20 64 75 70 28 63 . dup(c
7f30: 6f 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 20 20 20 onnection);.
7f40: 20 20 20 20 20 20 69 66 28 20 21 67 2e 66 48 74 if( !g.fHt
7f50: 74 70 54 72 61 63 65 20 29 7b 0a 20 20 20 20 20 tpTrace ){.
7f60: 20 20 20 20 20 20 20 63 6c 6f 73 65 28 32 29 3b close(2);
7f70: 0a 20 20 20 20 20 20 20 20 20 20 20 20 64 75 70 . dup
7f80: 28 63 6f 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 20 (connection);.
7f90: 20 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 }.
7fa0: 20 20 20 20 63 6c 6f 73 65 28 63 6f 6e 6e 65 63 close(connec
7fb0: 74 69 6f 6e 29 3b 0a 20 20 20 20 20 20 20 20 20 tion);.
7fc0: 20 72 65 74 75 72 6e 3b 0a 20 20 20 20 20 20 20 return;.
7fd0: 20 7d 0a 20 20 20 20 20 20 7d 0a 20 20 20 20 7d }. }. }
7fe0: 0a 20 20 20 20 2f 2a 20 42 75 72 79 20 64 65 61 . /* Bury dea
7ff0: 64 20 63 68 69 6c 64 72 65 6e 20 2a 2f 0a 20 20 d children */.
8000: 20 20 77 68 69 6c 65 28 20 77 61 69 74 70 69 64 while( waitpid
8010: 28 30 2c 20 30 2c 20 57 4e 4f 48 41 4e 47 29 3e (0, 0, WNOHANG)>
8020: 30 20 29 7b 0a 20 20 20 20 20 20 6e 63 68 69 6c 0 ){. nchil
8030: 64 72 65 6e 2d 2d 3b 0a 20 20 20 20 7d 0a 20 20 dren--;. }.
8040: 7d 0a 20 20 2f 2a 20 4e 4f 54 20 52 45 41 43 48 }. /* NOT REACH
8050: 45 44 20 2a 2f 20 20 0a 20 20 65 78 69 74 28 31 ED */ . exit(1
8060: 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 4e 61 6d 65 );.}../*.** Name
8070: 20 6f 66 20 64 61 79 73 20 61 6e 64 20 6d 6f 6e of days and mon
8080: 74 68 73 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 63 ths..*/.static c
8090: 6f 6e 73 74 20 63 68 61 72 20 2a 61 7a 44 61 79 onst char *azDay
80a0: 73 5b 5d 20 3d 0a 20 20 20 20 7b 22 53 75 6e 22 s[] =. {"Sun"
80b0: 2c 20 22 4d 6f 6e 22 2c 20 22 54 75 65 22 2c 20 , "Mon", "Tue",
80c0: 22 57 65 64 22 2c 20 22 54 68 75 22 2c 20 22 46 "Wed", "Thu", "F
80d0: 72 69 22 2c 20 22 53 61 74 22 2c 20 30 7d 3b 0a ri", "Sat", 0};.
80e0: 73 74 61 74 69 63 20 63 6f 6e 73 74 20 63 68 61 static const cha
80f0: 72 20 2a 61 7a 4d 6f 6e 74 68 73 5b 5d 20 3d 0a r *azMonths[] =.
8100: 20 20 20 20 7b 22 4a 61 6e 22 2c 20 22 46 65 62 {"Jan", "Feb
8110: 22 2c 20 22 4d 61 72 22 2c 20 22 41 70 72 22 2c ", "Mar", "Apr",
8120: 20 22 4d 61 79 22 2c 20 22 4a 75 6e 22 2c 0a 20 "May", "Jun",.
8130: 20 20 20 20 22 4a 75 6c 22 2c 20 22 41 75 67 22 "Jul", "Aug"
8140: 2c 20 22 53 65 70 22 2c 20 22 4f 63 74 22 2c 20 , "Sep", "Oct",
8150: 22 4e 6f 76 22 2c 20 22 44 65 63 22 2c 20 30 7d "Nov", "Dec", 0}
8160: 3b 0a 0a 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 6e ;.../*.** Return
8170: 73 20 61 6e 20 52 46 43 38 32 32 2d 66 6f 72 6d s an RFC822-form
8180: 61 74 74 65 64 20 74 69 6d 65 20 73 74 72 69 6e atted time strin
8190: 67 20 73 75 69 74 61 62 6c 65 20 66 6f 72 20 48 g suitable for H
81a0: 54 54 50 20 68 65 61 64 65 72 73 2c 20 61 6d 6f TTP headers, amo
81b0: 6e 67 0a 2a 2a 20 6f 74 68 65 72 20 74 68 69 6e ng.** other thin
81c0: 67 73 2e 0a 2a 2a 20 52 65 74 75 72 6e 65 64 20 gs..** Returned
81d0: 74 69 6d 65 7a 6f 6e 65 20 69 73 20 61 6c 77 61 timezone is alwa
81e0: 79 73 20 47 4d 54 20 61 73 20 72 65 71 75 69 72 ys GMT as requir
81f0: 65 64 20 62 79 20 48 54 54 50 2f 31 2e 31 20 73 ed by HTTP/1.1 s
8200: 70 65 63 69 66 69 63 61 74 69 6f 6e 2e 0a 2a 2a pecification..**
8210: 0a 2a 2a 20 53 65 65 20 68 74 74 70 3a 2f 2f 77 .** See http://w
8220: 77 77 2e 66 61 71 73 2e 6f 72 67 2f 72 66 63 73 ww.faqs.org/rfcs
8230: 2f 72 66 63 38 32 32 2e 68 74 6d 6c 2c 20 73 65 /rfc822.html, se
8240: 63 74 69 6f 6e 20 35 0a 2a 2a 20 61 6e 64 20 68 ction 5.** and h
8250: 74 74 70 3a 2f 2f 77 77 77 2e 66 61 71 73 2e 6f ttp://www.faqs.o
8260: 72 67 2f 72 66 63 73 2f 72 66 63 32 36 31 36 2e rg/rfcs/rfc2616.
8270: 68 74 6d 6c 2c 20 73 65 63 74 69 6f 6e 20 33 2e html, section 3.
8280: 33 2e 0a 2a 2f 0a 63 68 61 72 20 2a 63 67 69 5f 3..*/.char *cgi_
8290: 72 66 63 38 32 32 5f 64 61 74 65 73 74 61 6d 70 rfc822_datestamp
82a0: 28 74 69 6d 65 5f 74 20 6e 6f 77 29 7b 0a 20 20 (time_t now){.
82b0: 73 74 72 75 63 74 20 74 6d 20 2a 70 54 6d 3b 0a struct tm *pTm;.
82c0: 20 20 70 54 6d 20 3d 20 67 6d 74 69 6d 65 28 26 pTm = gmtime(&
82d0: 6e 6f 77 29 3b 0a 20 20 69 66 28 20 70 54 6d 3d now);. if( pTm=
82e0: 3d 30 20 29 20 72 65 74 75 72 6e 20 22 22 3b 0a =0 ) return "";.
82f0: 20 20 72 65 74 75 72 6e 20 6d 70 72 69 6e 74 66 return mprintf
8300: 28 22 25 73 2c 20 25 64 20 25 73 20 25 30 32 64 ("%s, %d %s %02d
8310: 20 25 30 32 64 3a 25 30 32 64 3a 25 30 32 64 20 %02d:%02d:%02d
8320: 47 4d 54 22 2c 0a 20 20 20 20 20 20 20 20 20 20 GMT",.
8330: 20 20 20 20 20 20 20 61 7a 44 61 79 73 5b 70 54 azDays[pT
8340: 6d 2d 3e 74 6d 5f 77 64 61 79 5d 2c 20 70 54 6d m->tm_wday], pTm
8350: 2d 3e 74 6d 5f 6d 64 61 79 2c 20 61 7a 4d 6f 6e ->tm_mday, azMon
8360: 74 68 73 5b 70 54 6d 2d 3e 74 6d 5f 6d 6f 6e 5d ths[pTm->tm_mon]
8370: 2c 0a 20 20 20 20 20 20 20 20 20 20 20 20 20 20 ,.
8380: 20 20 20 70 54 6d 2d 3e 74 6d 5f 79 65 61 72 2b pTm->tm_year+
8390: 31 39 30 30 2c 20 70 54 6d 2d 3e 74 6d 5f 68 6f 1900, pTm->tm_ho
83a0: 75 72 2c 20 70 54 6d 2d 3e 74 6d 5f 6d 69 6e 2c ur, pTm->tm_min,
83b0: 20 70 54 6d 2d 3e 74 6d 5f 73 65 63 29 3b 0a 7d pTm->tm_sec);.}
83c0: 0a 0a 2f 2a 0a 2a 2a 20 50 61 72 73 65 20 61 6e ../*.** Parse an
83d0: 20 52 46 43 38 32 32 2d 66 6f 72 6d 61 74 74 65 RFC822-formatte
83e0: 64 20 74 69 6d 65 73 74 61 6d 70 20 61 73 20 77 d timestamp as w
83f0: 65 27 64 20 65 78 70 65 63 74 20 66 72 6f 6d 20 e'd expect from
8400: 48 54 54 50 20 61 6e 64 20 72 65 74 75 72 6e 0a HTTP and return.
8410: 2a 2a 20 61 20 55 6e 69 78 20 65 70 6f 63 68 20 ** a Unix epoch
8420: 74 69 6d 65 2e 20 3c 3d 20 7a 65 72 6f 20 69 73 time. <= zero is
8430: 20 72 65 74 75 72 6e 65 64 20 6f 6e 20 66 61 69 returned on fai
8440: 6c 75 72 65 2e 0a 2a 2a 0a 2a 2a 20 4e 6f 74 65 lure..**.** Note
8450: 20 74 68 61 74 20 74 68 69 73 20 77 6f 6e 27 74 that this won't
8460: 20 68 61 6e 64 6c 65 20 61 6c 6c 20 74 68 65 20 handle all the
8470: 5f 61 6c 6c 6f 77 65 64 5f 20 48 54 54 50 20 66 _allowed_ HTTP f
8480: 6f 72 6d 61 74 73 2c 20 6a 75 73 74 20 74 68 65 ormats, just the
8490: 0a 2a 2a 20 6d 6f 73 74 20 70 6f 70 75 6c 61 72 .** most popular
84a0: 20 6f 6e 65 20 28 74 68 65 20 6f 6e 65 20 67 65 one (the one ge
84b0: 6e 65 72 61 74 65 64 20 62 79 20 63 67 69 5f 72 nerated by cgi_r
84c0: 66 63 38 32 32 5f 64 61 74 65 73 74 61 6d 70 28 fc822_datestamp(
84d0: 29 2c 20 61 63 74 75 61 6c 6c 79 29 2e 0a 2a 2f ), actually)..*/
84e0: 0a 74 69 6d 65 5f 74 20 63 67 69 5f 72 66 63 38 .time_t cgi_rfc8
84f0: 32 32 5f 70 61 72 73 65 64 61 74 65 28 63 6f 6e 22_parsedate(con
8500: 73 74 20 63 68 61 72 20 2a 7a 44 61 74 65 29 7b st char *zDate){
8510: 0a 20 20 73 74 72 75 63 74 20 74 6d 20 74 3b 0a . struct tm t;.
8520: 20 20 63 68 61 72 20 7a 49 67 6e 6f 72 65 5b 31 char zIgnore[1
8530: 36 5d 3b 0a 20 20 63 68 61 72 20 7a 4d 6f 6e 74 6];. char zMont
8540: 68 5b 31 36 5d 3b 0a 0a 20 20 6d 65 6d 73 65 74 h[16];.. memset
8550: 28 26 74 2c 20 30 2c 20 73 69 7a 65 6f 66 28 74 (&t, 0, sizeof(t
8560: 29 29 3b 0a 20 20 69 66 28 20 37 3d 3d 73 73 63 ));. if( 7==ssc
8570: 61 6e 66 28 7a 44 61 74 65 2c 20 22 25 31 32 5b anf(zDate, "%12[
8580: 41 2d 5a 61 2d 7a 2c 5d 20 25 64 20 25 31 32 5b A-Za-z,] %d %12[
8590: 41 2d 5a 61 2d 7a 5d 20 25 64 20 25 64 3a 25 64 A-Za-z] %d %d:%d
85a0: 3a 25 64 22 2c 20 7a 49 67 6e 6f 72 65 2c 0a 20 :%d", zIgnore,.
85b0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20
85c0: 20 20 20 20 20 20 26 74 2e 74 6d 5f 6d 64 61 79 &t.tm_mday
85d0: 2c 20 7a 4d 6f 6e 74 68 2c 20 26 74 2e 74 6d 5f , zMonth, &t.tm_
85e0: 79 65 61 72 2c 20 26 74 2e 74 6d 5f 68 6f 75 72 year, &t.tm_hour
85f0: 2c 20 26 74 2e 74 6d 5f 6d 69 6e 2c 0a 20 20 20 , &t.tm_min,.
8600: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20
8610: 20 20 20 20 26 74 2e 74 6d 5f 73 65 63 29 29 7b &t.tm_sec)){
8620: 0a 0a 20 20 20 20 69 66 28 20 74 2e 74 6d 5f 79 .. if( t.tm_y
8630: 65 61 72 20 3e 20 31 39 30 30 20 29 20 74 2e 74 ear > 1900 ) t.t
8640: 6d 5f 79 65 61 72 20 2d 3d 20 31 39 30 30 3b 0a m_year -= 1900;.
8650: 20 20 20 20 66 6f 72 28 74 2e 74 6d 5f 6d 6f 6e for(t.tm_mon
8660: 3d 30 3b 20 61 7a 4d 6f 6e 74 68 73 5b 74 2e 74 =0; azMonths[t.t
8670: 6d 5f 6d 6f 6e 5d 3b 20 74 2e 74 6d 5f 6d 6f 6e m_mon]; t.tm_mon
8680: 2b 2b 29 7b 0a 20 20 20 20 20 20 69 66 28 20 21 ++){. if( !
8690: 73 74 72 6e 63 61 73 65 63 6d 70 28 20 61 7a 4d strncasecmp( azM
86a0: 6f 6e 74 68 73 5b 74 2e 74 6d 5f 6d 6f 6e 5d 2c onths[t.tm_mon],
86b0: 20 7a 4d 6f 6e 74 68 2c 20 33 20 29 29 7b 0a 20 zMonth, 3 )){.
86c0: 20 20 20 20 20 20 20 72 65 74 75 72 6e 20 6d 6b return mk
86d0: 67 6d 74 69 6d 65 28 26 74 29 3b 0a 20 20 20 20 gmtime(&t);.
86e0: 20 20 7d 0a 20 20 20 20 7d 0a 20 20 7d 0a 0a 20 }. }. }..
86f0: 20 72 65 74 75 72 6e 20 30 3b 0a 7d 0a 0a 2f 2a return 0;.}../*
8700: 0a 2a 2a 20 43 6f 6e 76 65 72 74 20 61 20 73 74 .** Convert a st
8710: 72 75 63 74 20 74 6d 2a 20 74 68 61 74 20 72 65 ruct tm* that re
8720: 70 72 65 73 65 6e 74 73 20 61 20 6d 6f 6d 65 6e presents a momen
8730: 74 20 69 6e 20 55 54 43 20 69 6e 74 6f 20 74 68 t in UTC into th
8740: 65 20 6e 75 6d 62 65 72 0a 2a 2a 20 6f 66 20 73 e number.** of s
8750: 65 63 6f 6e 64 73 20 69 6e 20 31 39 37 30 2c 20 econds in 1970,
8760: 55 54 43 2e 0a 2a 2f 0a 74 69 6d 65 5f 74 20 6d UTC..*/.time_t m
8770: 6b 67 6d 74 69 6d 65 28 73 74 72 75 63 74 20 74 kgmtime(struct t
8780: 6d 20 2a 70 29 7b 0a 20 20 74 69 6d 65 5f 74 20 m *p){. time_t
8790: 74 3b 0a 20 20 69 6e 74 20 6e 44 61 79 3b 0a 20 t;. int nDay;.
87a0: 20 69 6e 74 20 69 73 4c 65 61 70 59 72 3b 0a 20 int isLeapYr;.
87b0: 20 2f 2a 20 44 61 79 73 20 69 6e 20 65 61 63 68 /* Days in each
87c0: 20 6d 6f 6e 74 68 3a 20 20 20 20 20 20 20 33 31 month: 31
87d0: 2c 20 32 38 2c 20 33 31 2c 20 33 30 2c 20 33 31 , 28, 31, 30, 31
87e0: 2c 20 33 30 2c 20 33 31 2c 20 33 31 2c 20 33 30 , 30, 31, 31, 30
87f0: 2c 20 33 31 2c 20 33 30 2c 20 33 31 20 2a 2f 0a , 31, 30, 31 */.
8800: 20 20 73 74 61 74 69 63 20 69 6e 74 20 70 72 69 static int pri
8810: 6f 72 44 61 79 73 5b 5d 20 20 20 3d 20 7b 20 20 orDays[] = {
8820: 30 2c 20 33 31 2c 20 35 39 2c 20 39 30 2c 31 32 0, 31, 59, 90,12
8830: 30 2c 31 35 31 2c 31 38 31 2c 32 31 32 2c 32 34 0,151,181,212,24
8840: 33 2c 32 37 33 2c 33 30 34 2c 33 33 34 20 7d 3b 3,273,304,334 };
8850: 0a 20 20 69 66 28 20 70 2d 3e 74 6d 5f 6d 6f 6e . if( p->tm_mon
8860: 3c 30 20 29 7b 0a 20 20 20 20 69 6e 74 20 6e 59 <0 ){. int nY
8870: 65 61 72 20 3d 20 28 31 31 20 2d 20 70 2d 3e 74 ear = (11 - p->t
8880: 6d 5f 6d 6f 6e 29 2f 31 32 3b 0a 20 20 20 20 70 m_mon)/12;. p
8890: 2d 3e 74 6d 5f 79 65 61 72 20 2d 3d 20 6e 59 65 ->tm_year -= nYe
88a0: 61 72 3b 0a 20 20 20 20 70 2d 3e 74 6d 5f 6d 6f ar;. p->tm_mo
88b0: 6e 20 2b 3d 20 6e 59 65 61 72 2a 31 32 3b 0a 20 n += nYear*12;.
88c0: 20 7d 65 6c 73 65 20 69 66 28 20 70 2d 3e 74 6d }else if( p->tm
88d0: 5f 6d 6f 6e 3e 31 31 20 29 7b 0a 20 20 20 20 70 _mon>11 ){. p
88e0: 2d 3e 74 6d 5f 79 65 61 72 20 2b 3d 20 70 2d 3e ->tm_year += p->
88f0: 74 6d 5f 6d 6f 6e 2f 31 32 3b 0a 20 20 20 20 70 tm_mon/12;. p
8900: 2d 3e 74 6d 5f 6d 6f 6e 20 25 3d 20 31 32 3b 0a ->tm_mon %= 12;.
8910: 20 20 7d 0a 20 20 69 73 4c 65 61 70 59 72 20 3d }. isLeapYr =
8920: 20 70 2d 3e 74 6d 5f 79 65 61 72 25 34 3d 3d 30 p->tm_year%4==0
8930: 20 26 26 20 28 70 2d 3e 74 6d 5f 79 65 61 72 25 && (p->tm_year%
8940: 31 30 30 21 3d 30 20 7c 7c 20 28 70 2d 3e 74 6d 100!=0 || (p->tm
8950: 5f 79 65 61 72 2b 33 30 30 29 25 34 30 30 3d 3d _year+300)%400==
8960: 30 29 3b 0a 20 20 70 2d 3e 74 6d 5f 79 64 61 79 0);. p->tm_yday
8970: 20 3d 20 70 72 69 6f 72 44 61 79 73 5b 70 2d 3e = priorDays[p->
8980: 74 6d 5f 6d 6f 6e 5d 20 2b 20 70 2d 3e 74 6d 5f tm_mon] + p->tm_
8990: 6d 64 61 79 20 2d 20 31 3b 0a 20 20 69 66 28 20 mday - 1;. if(
89a0: 69 73 4c 65 61 70 59 72 20 26 26 20 70 2d 3e 74 isLeapYr && p->t
89b0: 6d 5f 6d 6f 6e 3e 31 20 29 20 70 2d 3e 74 6d 5f m_mon>1 ) p->tm_
89c0: 79 64 61 79 2b 2b 3b 0a 20 20 6e 44 61 79 20 3d yday++;. nDay =
89d0: 20 28 70 2d 3e 74 6d 5f 79 65 61 72 2d 37 30 29 (p->tm_year-70)
89e0: 2a 33 36 35 20 2b 20 28 70 2d 3e 74 6d 5f 79 65 *365 + (p->tm_ye
89f0: 61 72 2d 36 39 29 2f 34 20 2d 70 2d 3e 74 6d 5f ar-69)/4 -p->tm_
8a00: 79 65 61 72 2f 31 30 30 20 2b 20 0a 20 20 20 20 year/100 + .
8a10: 20 20 20 20 20 28 70 2d 3e 74 6d 5f 79 65 61 72 (p->tm_year
8a20: 2b 33 30 30 29 2f 34 30 30 20 2b 20 70 2d 3e 74 +300)/400 + p->t
8a30: 6d 5f 79 64 61 79 3b 0a 20 20 74 20 3d 20 28 28 m_yday;. t = ((
8a40: 6e 44 61 79 2a 32 34 20 2b 20 70 2d 3e 74 6d 5f nDay*24 + p->tm_
8a50: 68 6f 75 72 29 2a 36 30 20 2b 20 70 2d 3e 74 6d hour)*60 + p->tm
8a60: 5f 6d 69 6e 29 2a 36 30 20 2b 20 70 2d 3e 74 6d _min)*60 + p->tm
8a70: 5f 73 65 63 3b 0a 20 20 72 65 74 75 72 6e 20 74 _sec;. return t
8a80: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 43 68 65 63 6b ;.}../*.** Check
8a90: 20 74 68 65 20 6f 62 6a 65 63 74 54 69 6d 65 20 the objectTime
8aa0: 61 67 61 69 6e 73 74 20 74 68 65 20 49 66 2d 4d against the If-M
8ab0: 6f 64 69 66 69 65 64 2d 53 69 6e 63 65 20 72 65 odified-Since re
8ac0: 71 75 65 73 74 20 68 65 61 64 65 72 2e 20 49 66 quest header. If
8ad0: 20 74 68 65 0a 2a 2a 20 6f 62 6a 65 63 74 20 74 the.** object t
8ae0: 69 6d 65 20 69 73 6e 27 74 20 61 6e 79 20 6e 65 ime isn't any ne
8af0: 77 65 72 20 74 68 61 6e 20 74 68 65 20 68 65 61 wer than the hea
8b00: 64 65 72 2c 20 77 65 20 69 6d 6d 65 64 69 61 74 der, we immediat
8b10: 65 6c 79 20 73 65 6e 64 20 62 61 63 6b 0a 2a 2a ely send back.**
8b20: 20 61 20 33 30 34 20 72 65 70 6c 79 20 61 6e 64 a 304 reply and
8b30: 20 65 78 69 74 2e 0a 2a 2f 0a 76 6f 69 64 20 63 exit..*/.void c
8b40: 67 69 5f 6d 6f 64 69 66 69 65 64 5f 73 69 6e 63 gi_modified_sinc
8b50: 65 28 74 69 6d 65 5f 74 20 6f 62 6a 65 63 74 54 e(time_t objectT
8b60: 69 6d 65 29 7b 0a 20 20 63 6f 6e 73 74 20 63 68 ime){. const ch
8b70: 61 72 20 2a 7a 49 66 20 3d 20 50 28 22 48 54 54 ar *zIf = P("HTT
8b80: 50 5f 49 46 5f 4d 4f 44 49 46 49 45 44 5f 53 49 P_IF_MODIFIED_SI
8b90: 4e 43 45 22 29 3b 0a 20 20 69 66 28 20 7a 49 66 NCE");. if( zIf
8ba0: 3d 3d 30 20 29 20 72 65 74 75 72 6e 3b 0a 20 20 ==0 ) return;.
8bb0: 69 66 28 20 6f 62 6a 65 63 74 54 69 6d 65 20 3e if( objectTime >
8bc0: 20 63 67 69 5f 72 66 63 38 32 32 5f 70 61 72 73 cgi_rfc822_pars
8bd0: 65 64 61 74 65 28 7a 49 66 29 20 29 20 72 65 74 edate(zIf) ) ret
8be0: 75 72 6e 3b 0a 20 20 63 67 69 5f 73 65 74 5f 73 urn;. cgi_set_s
8bf0: 74 61 74 75 73 28 33 30 34 2c 22 4e 6f 74 20 4d tatus(304,"Not M
8c00: 6f 64 69 66 69 65 64 22 29 3b 0a 20 20 63 67 69 odified");. cgi
8c10: 5f 72 65 73 65 74 5f 63 6f 6e 74 65 6e 74 28 29 _reset_content()
8c20: 3b 0a 20 20 63 67 69 5f 72 65 70 6c 79 28 29 3b ;. cgi_reply();
8c30: 0a 20 20 65 78 69 74 28 30 29 3b 0a 7d 0a . exit(0);.}.