Hex Artifact Content
Not logged in

Artifact db2f7cb7a920014048a0c58b8252caeed9f90fea:

File src/cgi.c part of check-in [e2e016c31f] - Get the "server" command running under windows. by drh on 2008-05-17 17:43:22.

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 66 64   "config.h".#ifd
04b0: 65 66 20 5f 5f 4d 49 4e 47 57 33 32 5f 5f 0a 23  ef __MINGW32__.#
04c0: 20 20 69 6e 63 6c 75 64 65 20 3c 77 69 6e 64 6f    include <windo
04d0: 77 73 2e 68 3e 20 20 20 20 20 20 20 20 20 20 20  ws.h>           
04e0: 2f 2a 20 66 6f 72 20 53 6c 65 65 70 20 6f 6e 63  /* for Sleep onc
04f0: 65 20 73 65 72 76 65 72 20 77 6f 72 6b 73 20 61  e server works a
0500: 67 61 69 6e 20 2a 2f 0a 23 20 20 69 6e 63 6c 75  gain */.#  inclu
0510: 64 65 20 3c 77 69 6e 73 6f 63 6b 32 2e 68 3e 20  de <winsock2.h> 
0520: 20 20 20 20 20 20 20 20 20 2f 2a 20 73 6f 63 6b           /* sock
0530: 65 74 20 6f 70 65 72 61 74 69 6f 6e 73 20 2a 2f  et operations */
0540: 0a 23 20 20 64 65 66 69 6e 65 20 73 6c 65 65 70  .#  define sleep
0550: 20 53 6c 65 65 70 20 20 20 20 20 20 20 20 20 20   Sleep          
0560: 20 20 2f 2a 20 77 69 6e 64 6f 77 73 20 64 6f 65    /* windows doe
0570: 73 20 6e 6f 74 20 68 61 76 65 20 73 6c 65 65 70  s not have sleep
0580: 2c 20 62 75 74 20 53 6c 65 65 70 20 2a 2f 0a 23  , but Sleep */.#
0590: 20 20 69 6e 63 6c 75 64 65 20 3c 77 73 32 74 63    include <ws2tc
05a0: 70 69 70 2e 68 3e 20 20 20 20 20 20 20 20 20 20  pip.h>          
05b0: 0a 23 65 6c 73 65 0a 23 20 20 69 6e 63 6c 75 64  .#else.#  includ
05c0: 65 20 3c 73 79 73 2f 73 6f 63 6b 65 74 2e 68 3e  e <sys/socket.h>
05d0: 0a 23 20 20 69 6e 63 6c 75 64 65 20 3c 6e 65 74  .#  include <net
05e0: 69 6e 65 74 2f 69 6e 2e 68 3e 0a 23 20 20 69 6e  inet/in.h>.#  in
05f0: 63 6c 75 64 65 20 3c 61 72 70 61 2f 69 6e 65 74  clude <arpa/inet
0600: 2e 68 3e 0a 23 20 20 69 6e 63 6c 75 64 65 20 3c  .h>.#  include <
0610: 73 79 73 2f 74 69 6d 65 73 2e 68 3e 0a 23 20 20  sys/times.h>.#  
0620: 69 6e 63 6c 75 64 65 20 3c 73 79 73 2f 74 69 6d  include <sys/tim
0630: 65 2e 68 3e 0a 23 20 20 69 6e 63 6c 75 64 65 20  e.h>.#  include 
0640: 3c 73 79 73 2f 77 61 69 74 2e 68 3e 0a 23 20 20  <sys/wait.h>.#  
0650: 69 6e 63 6c 75 64 65 20 3c 73 79 73 2f 73 65 6c  include <sys/sel
0660: 65 63 74 2e 68 3e 0a 23 65 6e 64 69 66 0a 23 69  ect.h>.#endif.#i
0670: 6e 63 6c 75 64 65 20 3c 74 69 6d 65 2e 68 3e 0a  nclude <time.h>.
0680: 23 69 6e 63 6c 75 64 65 20 3c 73 74 64 69 6f 2e  #include <stdio.
0690: 68 3e 0a 23 69 6e 63 6c 75 64 65 20 3c 73 74 64  h>.#include <std
06a0: 6c 69 62 2e 68 3e 0a 23 69 6e 63 6c 75 64 65 20  lib.h>.#include 
06b0: 3c 75 6e 69 73 74 64 2e 68 3e 0a 23 69 6e 63 6c  <unistd.h>.#incl
06c0: 75 64 65 20 22 63 67 69 2e 68 22 0a 0a 23 69 66  ude "cgi.h"..#if
06d0: 20 49 4e 54 45 52 46 41 43 45 0a 2f 2a 0a 2a 2a   INTERFACE./*.**
06e0: 20 53 68 6f 72 74 63 75 74 73 20 66 6f 72 20 63   Shortcuts for c
06f0: 67 69 5f 70 61 72 61 6d 65 74 65 72 2e 20 20 50  gi_parameter.  P
0700: 28 22 78 22 29 20 72 65 74 75 72 6e 73 20 74 68  ("x") returns th
0710: 65 20 76 61 6c 75 65 20 6f 66 20 71 75 65 72 79  e value of query
0720: 20 70 61 72 61 6d 65 74 65 72 0a 2a 2a 20 6f 72   parameter.** or
0730: 20 63 6f 6f 6b 69 65 20 22 78 22 2c 20 6f 72 20   cookie "x", or 
0740: 4e 55 4c 4c 20 69 66 20 74 68 65 72 65 20 69 73  NULL if there is
0750: 20 6e 6f 20 73 75 63 68 20 70 61 72 61 6d 65 74   no such paramet
0760: 65 72 20 6f 72 20 63 6f 6f 6b 69 65 2e 20 20 50  er or cookie.  P
0770: 44 28 22 78 22 2c 22 79 22 29 0a 2a 2a 20 64 6f  D("x","y").** do
0780: 65 73 20 74 68 65 20 73 61 6d 65 20 65 78 63 65  es the same exce
0790: 70 74 20 22 79 22 20 69 73 20 72 65 74 75 72 6e  pt "y" is return
07a0: 65 64 20 69 6e 20 70 6c 61 63 65 20 6f 66 20 4e  ed in place of N
07b0: 55 4c 4c 20 69 66 20 74 68 65 72 65 20 69 73 20  ULL if there is 
07c0: 6e 6f 74 20 6d 61 74 63 68 2e 0a 2a 2f 0a 23 64  not match..*/.#d
07d0: 65 66 69 6e 65 20 50 28 78 29 20 20 20 20 20 20  efine P(x)      
07e0: 20 20 63 67 69 5f 70 61 72 61 6d 65 74 65 72 28    cgi_parameter(
07f0: 28 78 29 2c 30 29 0a 23 64 65 66 69 6e 65 20 50  (x),0).#define P
0800: 44 28 78 2c 79 29 20 20 20 20 20 63 67 69 5f 70  D(x,y)     cgi_p
0810: 61 72 61 6d 65 74 65 72 28 28 78 29 2c 28 79 29  arameter((x),(y)
0820: 29 0a 23 64 65 66 69 6e 65 20 51 50 28 78 29 20  ).#define QP(x) 
0830: 20 20 20 20 20 20 71 75 6f 74 61 62 6c 65 5f 73        quotable_s
0840: 74 72 69 6e 67 28 63 67 69 5f 70 61 72 61 6d 65  tring(cgi_parame
0850: 74 65 72 28 28 78 29 2c 30 29 29 0a 23 64 65 66  ter((x),0)).#def
0860: 69 6e 65 20 51 50 44 28 78 2c 79 29 20 20 20 20  ine QPD(x,y)    
0870: 71 75 6f 74 61 62 6c 65 5f 73 74 72 69 6e 67 28  quotable_string(
0880: 63 67 69 5f 70 61 72 61 6d 65 74 65 72 28 28 78  cgi_parameter((x
0890: 29 2c 28 79 29 29 29 0a 0a 0a 2f 2a 0a 2a 2a 20  ),(y))).../*.** 
08a0: 44 65 73 74 69 6e 61 74 69 6f 6e 73 20 66 6f 72  Destinations for
08b0: 20 6f 75 74 70 75 74 20 74 65 78 74 2e 0a 2a 2f   output text..*/
08c0: 0a 23 64 65 66 69 6e 65 20 43 47 49 5f 48 45 41  .#define CGI_HEA
08d0: 44 45 52 20 20 20 30 0a 23 64 65 66 69 6e 65 20  DER   0.#define 
08e0: 43 47 49 5f 42 4f 44 59 20 20 20 20 20 31 0a 0a  CGI_BODY     1..
08f0: 23 65 6e 64 69 66 20 2f 2a 20 49 4e 54 45 52 46  #endif /* INTERF
0900: 41 43 45 20 2a 2f 0a 0a 2f 2a 0a 2a 2a 20 50 72  ACE */../*.** Pr
0910: 6f 76 69 64 65 20 61 20 72 65 6c 69 61 62 6c 65  ovide a reliable
0920: 20 69 6d 70 6c 65 6d 65 6e 74 61 74 69 6f 6e 20   implementation 
0930: 6f 66 20 61 20 63 61 73 65 6c 65 73 73 20 73 74  of a caseless st
0940: 72 69 6e 67 20 63 6f 6d 70 61 72 69 73 6f 6e 0a  ring comparison.
0950: 2a 2a 20 66 75 6e 63 74 69 6f 6e 2e 0a 2a 2f 0a  ** function..*/.
0960: 23 64 65 66 69 6e 65 20 73 74 72 69 63 6d 70 20  #define stricmp 
0970: 73 71 6c 69 74 65 33 53 74 72 49 43 6d 70 0a 65  sqlite3StrICmp.e
0980: 78 74 65 72 6e 20 69 6e 74 20 73 71 6c 69 74 65  xtern int sqlite
0990: 33 53 74 72 49 43 6d 70 28 63 6f 6e 73 74 20 63  3StrICmp(const c
09a0: 68 61 72 2a 2c 20 63 6f 6e 73 74 20 63 68 61 72  har*, const char
09b0: 2a 29 3b 0a 0a 2f 2a 0a 2a 2a 20 54 68 65 20 48  *);../*.** The H
09c0: 54 54 50 20 72 65 70 6c 79 20 69 73 20 67 65 6e  TTP reply is gen
09d0: 65 72 61 74 65 64 20 69 6e 20 74 77 6f 20 70 69  erated in two pi
09e0: 65 63 65 73 3a 20 74 68 65 20 68 65 61 64 65 72  eces: the header
09f0: 20 61 6e 64 20 74 68 65 20 62 6f 64 79 2e 0a 2a   and the body..*
0a00: 2a 20 54 68 65 73 65 20 70 69 65 63 65 73 20 61  * These pieces a
0a10: 72 65 20 67 65 6e 65 72 61 74 65 64 20 73 65 70  re generated sep
0a20: 61 72 61 74 65 6c 79 20 62 65 63 61 75 73 65 20  arately because 
0a30: 74 68 65 79 20 61 72 65 20 6e 6f 74 20 6e 65 63  they are not nec
0a40: 65 73 73 61 72 79 0a 2a 2a 20 70 72 6f 64 75 63  essary.** produc
0a50: 65 64 20 69 6e 20 6f 72 64 65 72 2e 20 20 50 61  ed in order.  Pa
0a60: 72 74 73 20 6f 66 20 74 68 65 20 68 65 61 64 65  rts of the heade
0a70: 72 20 6d 69 67 68 74 20 62 65 20 62 75 69 6c 74  r might be built
0a80: 20 61 66 74 65 72 20 61 6c 6c 20 6f 72 0a 2a 2a   after all or.**
0a90: 20 70 61 72 74 20 6f 66 20 74 68 65 20 62 6f 64   part of the bod
0aa0: 79 2e 20 20 54 68 65 20 68 65 61 64 65 72 20 61  y.  The header a
0ab0: 6e 64 20 62 6f 64 79 20 61 72 65 20 61 63 63 75  nd body are accu
0ac0: 6d 75 6c 61 74 65 64 20 69 6e 20 73 65 70 61 72  mulated in separ
0ad0: 61 74 65 0a 2a 2a 20 42 6c 6f 62 20 73 74 72 75  ate.** Blob stru
0ae0: 63 74 75 72 65 73 20 74 68 65 6e 20 6f 75 74 70  ctures then outp
0af0: 75 74 20 73 65 71 75 65 6e 74 69 61 6c 6c 79 20  ut sequentially 
0b00: 6f 6e 63 65 20 65 76 65 72 79 74 68 69 6e 67 20  once everything 
0b10: 68 61 73 20 62 65 65 6e 0a 2a 2a 20 62 75 69 6c  has been.** buil
0b20: 74 2e 0a 2a 2a 0a 2a 2a 20 54 68 65 20 63 67 69  t..**.** The cgi
0b30: 5f 64 65 73 74 69 6e 61 74 69 6f 6e 28 29 20 69  _destination() i
0b40: 6e 74 65 72 66 61 63 65 20 73 77 69 74 63 68 20  nterface switch 
0b50: 62 65 74 77 65 65 6e 20 74 68 65 20 62 75 66 66  between the buff
0b60: 65 72 73 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 42  ers..*/.static B
0b70: 6c 6f 62 20 63 67 69 43 6f 6e 74 65 6e 74 5b 32  lob cgiContent[2
0b80: 5d 20 3d 20 7b 20 42 4c 4f 42 5f 49 4e 49 54 49  ] = { BLOB_INITI
0b90: 41 4c 49 5a 45 52 2c 20 42 4c 4f 42 5f 49 4e 49  ALIZER, BLOB_INI
0ba0: 54 49 41 4c 49 5a 45 52 20 7d 3b 0a 73 74 61 74  TIALIZER };.stat
0bb0: 69 63 20 42 6c 6f 62 20 2a 70 43 6f 6e 74 65 6e  ic Blob *pConten
0bc0: 74 20 3d 20 26 63 67 69 43 6f 6e 74 65 6e 74 5b  t = &cgiContent[
0bd0: 30 5d 3b 0a 0a 2f 2a 0a 2a 2a 20 53 65 74 20 74  0];../*.** Set t
0be0: 68 65 20 64 65 73 74 69 6e 61 74 69 6f 6e 20 62  he destination b
0bf0: 75 66 66 65 72 20 69 6e 74 6f 20 77 68 69 63 68  uffer into which
0c00: 20 74 6f 20 61 63 63 75 6d 75 6c 61 74 65 20 43   to accumulate C
0c10: 47 49 20 63 6f 6e 74 65 6e 74 2e 0a 2a 2f 0a 76  GI content..*/.v
0c20: 6f 69 64 20 63 67 69 5f 64 65 73 74 69 6e 61 74  oid cgi_destinat
0c30: 69 6f 6e 28 69 6e 74 20 64 65 73 74 29 7b 0a 20  ion(int dest){. 
0c40: 20 73 77 69 74 63 68 28 20 64 65 73 74 20 29 7b   switch( dest ){
0c50: 0a 20 20 20 20 63 61 73 65 20 43 47 49 5f 48 45  .    case CGI_HE
0c60: 41 44 45 52 3a 20 7b 0a 20 20 20 20 20 20 70 43  ADER: {.      pC
0c70: 6f 6e 74 65 6e 74 20 3d 20 26 63 67 69 43 6f 6e  ontent = &cgiCon
0c80: 74 65 6e 74 5b 30 5d 3b 0a 20 20 20 20 20 20 62  tent[0];.      b
0c90: 72 65 61 6b 3b 0a 20 20 20 20 7d 0a 20 20 20 20  reak;.    }.    
0ca0: 63 61 73 65 20 43 47 49 5f 42 4f 44 59 3a 20 7b  case CGI_BODY: {
0cb0: 0a 20 20 20 20 20 20 70 43 6f 6e 74 65 6e 74 20  .      pContent 
0cc0: 3d 20 26 63 67 69 43 6f 6e 74 65 6e 74 5b 31 5d  = &cgiContent[1]
0cd0: 3b 0a 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20  ;.      break;. 
0ce0: 20 20 20 7d 0a 20 20 20 20 64 65 66 61 75 6c 74     }.    default
0cf0: 3a 20 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 61  : {.      cgi_pa
0d00: 6e 69 63 28 22 62 61 64 20 64 65 73 74 69 6e 61  nic("bad destina
0d10: 74 69 6f 6e 22 29 3b 0a 20 20 20 20 7d 0a 20 20  tion");.    }.  
0d20: 7d 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 41 70 70 65 6e  }.}../*.** Appen
0d30: 64 20 72 65 70 6c 79 20 63 6f 6e 74 65 6e 74 20  d reply content 
0d40: 74 6f 20 77 68 61 74 20 61 6c 72 65 61 64 79 20  to what already 
0d50: 65 78 69 73 74 73 2e 0a 2a 2f 0a 76 6f 69 64 20  exists..*/.void 
0d60: 63 67 69 5f 61 70 70 65 6e 64 5f 63 6f 6e 74 65  cgi_append_conte
0d70: 6e 74 28 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a  nt(const char *z
0d80: 44 61 74 61 2c 20 69 6e 74 20 6e 41 6d 74 29 7b  Data, int nAmt){
0d90: 0a 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 28 70  .  blob_append(p
0da0: 43 6f 6e 74 65 6e 74 2c 20 7a 44 61 74 61 2c 20  Content, zData, 
0db0: 6e 41 6d 74 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  nAmt);.}../*.** 
0dc0: 52 65 73 65 74 20 74 68 65 20 48 54 54 50 20 72  Reset the HTTP r
0dd0: 65 70 6c 79 20 74 65 78 74 20 74 6f 20 62 65 20  eply text to be 
0de0: 61 6e 20 65 6d 70 74 79 20 73 74 72 69 6e 67 2e  an empty string.
0df0: 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 72 65 73  .*/.void cgi_res
0e00: 65 74 5f 63 6f 6e 74 65 6e 74 28 76 6f 69 64 29  et_content(void)
0e10: 7b 0a 20 20 62 6c 6f 62 5f 72 65 73 65 74 28 26  {.  blob_reset(&
0e20: 63 67 69 43 6f 6e 74 65 6e 74 5b 30 5d 29 3b 0a  cgiContent[0]);.
0e30: 20 20 62 6c 6f 62 5f 72 65 73 65 74 28 26 63 67    blob_reset(&cg
0e40: 69 43 6f 6e 74 65 6e 74 5b 31 5d 29 3b 0a 7d 0a  iContent[1]);.}.
0e50: 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 61 20  ./*.** Return a 
0e60: 70 6f 69 6e 74 65 72 20 74 6f 20 74 68 65 20 43  pointer to the C
0e70: 47 49 20 6f 75 74 70 75 74 20 62 6c 6f 62 2e 0a  GI output blob..
0e80: 2a 2f 0a 42 6c 6f 62 20 2a 63 67 69 5f 6f 75 74  */.Blob *cgi_out
0e90: 70 75 74 5f 62 6c 6f 62 28 76 6f 69 64 29 7b 0a  put_blob(void){.
0ea0: 20 20 72 65 74 75 72 6e 20 70 43 6f 6e 74 65 6e    return pConten
0eb0: 74 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 43 6f 6d 62  t;.}../*.** Comb
0ec0: 69 6e 65 20 74 68 65 20 68 65 61 64 65 72 20 61  ine the header a
0ed0: 6e 64 20 62 6f 64 79 20 6f 66 20 74 68 65 20 43  nd body of the C
0ee0: 47 49 20 69 6e 74 6f 20 61 20 73 69 6e 67 6c 65  GI into a single
0ef0: 20 73 74 72 69 6e 67 2e 0a 2a 2f 0a 73 74 61 74   string..*/.stat
0f00: 69 63 20 76 6f 69 64 20 63 67 69 5f 63 6f 6d 62  ic void cgi_comb
0f10: 69 6e 65 5f 68 65 61 64 65 72 5f 61 6e 64 5f 62  ine_header_and_b
0f20: 6f 64 79 28 76 6f 69 64 29 7b 0a 20 20 69 6e 74  ody(void){.  int
0f30: 20 73 69 7a 65 20 3d 20 62 6c 6f 62 5f 73 69 7a   size = blob_siz
0f40: 65 28 26 63 67 69 43 6f 6e 74 65 6e 74 5b 31 5d  e(&cgiContent[1]
0f50: 29 3b 0a 20 20 69 66 28 20 73 69 7a 65 3e 30 20  );.  if( size>0 
0f60: 29 7b 0a 20 20 20 20 62 6c 6f 62 5f 61 70 70 65  ){.    blob_appe
0f70: 6e 64 28 26 63 67 69 43 6f 6e 74 65 6e 74 5b 30  nd(&cgiContent[0
0f80: 5d 2c 20 62 6c 6f 62 5f 62 75 66 66 65 72 28 26  ], blob_buffer(&
0f90: 63 67 69 43 6f 6e 74 65 6e 74 5b 31 5d 29 2c 20  cgiContent[1]), 
0fa0: 73 69 7a 65 29 3b 0a 20 20 20 20 62 6c 6f 62 5f  size);.    blob_
0fb0: 72 65 73 65 74 28 26 63 67 69 43 6f 6e 74 65 6e  reset(&cgiConten
0fc0: 74 5b 31 5d 29 3b 0a 20 20 7d 0a 7d 0a 0a 2f 2a  t[1]);.  }.}../*
0fd0: 0a 2a 2a 20 52 65 74 75 72 6e 20 61 20 70 6f 69  .** Return a poi
0fe0: 6e 74 65 72 20 74 6f 20 74 68 65 20 48 54 54 50  nter to the HTTP
0ff0: 20 72 65 70 6c 79 20 74 65 78 74 2e 0a 2a 2f 0a   reply text..*/.
1000: 63 68 61 72 20 2a 63 67 69 5f 65 78 74 72 61 63  char *cgi_extrac
1010: 74 5f 63 6f 6e 74 65 6e 74 28 69 6e 74 20 2a 70  t_content(int *p
1020: 6e 41 6d 74 29 7b 0a 20 20 63 67 69 5f 63 6f 6d  nAmt){.  cgi_com
1030: 62 69 6e 65 5f 68 65 61 64 65 72 5f 61 6e 64 5f  bine_header_and_
1040: 62 6f 64 79 28 29 3b 0a 20 20 72 65 74 75 72 6e  body();.  return
1050: 20 62 6c 6f 62 5f 62 75 66 66 65 72 28 26 63 67   blob_buffer(&cg
1060: 69 43 6f 6e 74 65 6e 74 5b 30 5d 29 3b 0a 7d 0a  iContent[0]);.}.
1070: 0a 2f 2a 0a 2a 2a 20 41 64 64 69 74 69 6f 6e 61  ./*.** Additiona
1080: 6c 20 69 6e 66 6f 72 6d 61 74 69 6f 6e 20 75 73  l information us
1090: 65 64 20 74 6f 20 66 6f 72 6d 20 74 68 65 20 48  ed to form the H
10a0: 54 54 50 20 72 65 70 6c 79 0a 2a 2f 0a 73 74 61  TTP reply.*/.sta
10b0: 74 69 63 20 63 68 61 72 20 2a 7a 43 6f 6e 74 65  tic char *zConte
10c0: 6e 74 54 79 70 65 20 3d 20 22 74 65 78 74 2f 68  ntType = "text/h
10d0: 74 6d 6c 22 3b 20 20 20 20 20 2f 2a 20 43 6f 6e  tml";     /* Con
10e0: 74 65 6e 74 20 74 79 70 65 20 6f 66 20 74 68 65  tent type of the
10f0: 20 72 65 70 6c 79 20 2a 2f 0a 73 74 61 74 69 63   reply */.static
1100: 20 63 68 61 72 20 2a 7a 52 65 70 6c 79 53 74 61   char *zReplySta
1110: 74 75 73 20 3d 20 22 4f 4b 22 3b 20 20 20 20 20  tus = "OK";     
1120: 20 20 20 20 20 20 20 2f 2a 20 52 65 70 6c 79 20         /* Reply 
1130: 73 74 61 74 75 73 20 64 65 73 63 72 69 70 74 69  status descripti
1140: 6f 6e 20 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74  on */.static int
1150: 20 69 52 65 70 6c 79 53 74 61 74 75 73 20 3d 20   iReplyStatus = 
1160: 32 30 30 3b 20 20 20 20 20 20 20 20 20 20 20 20  200;            
1170: 20 20 20 2f 2a 20 52 65 70 6c 79 20 73 74 61 74     /* Reply stat
1180: 75 73 20 63 6f 64 65 20 2a 2f 0a 73 74 61 74 69  us code */.stati
1190: 63 20 42 6c 6f 62 20 65 78 74 72 61 48 65 61 64  c Blob extraHead
11a0: 65 72 20 3d 20 42 4c 4f 42 5f 49 4e 49 54 49 41  er = BLOB_INITIA
11b0: 4c 49 5a 45 52 3b 20 20 2f 2a 20 45 78 74 72 61  LIZER;  /* Extra
11c0: 20 68 65 61 64 65 72 20 74 65 78 74 20 2a 2f 0a   header text */.
11d0: 73 74 61 74 69 63 20 69 6e 74 20 66 75 6c 6c 48  static int fullH
11e0: 74 74 70 52 65 70 6c 79 20 3d 20 30 3b 20 20 20  ttpReply = 0;   
11f0: 20 20 20 2f 2a 20 54 72 75 65 20 66 6f 72 20 61     /* True for a
1200: 20 66 75 6c 6c 2d 62 6c 6f 77 6e 20 48 54 54 50   full-blown HTTP
1210: 20 68 65 61 64 65 72 20 2a 2f 0a 0a 2f 2a 0a 2a   header */../*.*
1220: 2a 20 53 65 74 20 74 68 65 20 72 65 70 6c 79 20  * Set the reply 
1230: 63 6f 6e 74 65 6e 74 20 74 79 70 65 0a 2a 2f 0a  content type.*/.
1240: 76 6f 69 64 20 63 67 69 5f 73 65 74 5f 63 6f 6e  void cgi_set_con
1250: 74 65 6e 74 5f 74 79 70 65 28 63 6f 6e 73 74 20  tent_type(const 
1260: 63 68 61 72 20 2a 7a 54 79 70 65 29 7b 0a 20 20  char *zType){.  
1270: 7a 43 6f 6e 74 65 6e 74 54 79 70 65 20 3d 20 6d  zContentType = m
1280: 70 72 69 6e 74 66 28 22 25 73 22 2c 20 7a 54 79  printf("%s", zTy
1290: 70 65 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 65  pe);.}../*.** Se
12a0: 74 20 74 68 65 20 72 65 70 6c 79 20 63 6f 6e 74  t the reply cont
12b0: 65 6e 74 20 74 6f 20 74 68 65 20 73 70 65 63 69  ent to the speci
12c0: 66 69 65 64 20 42 4c 4f 42 2e 0a 2a 2f 0a 76 6f  fied BLOB..*/.vo
12d0: 69 64 20 63 67 69 5f 73 65 74 5f 63 6f 6e 74 65  id cgi_set_conte
12e0: 6e 74 28 42 6c 6f 62 20 2a 70 4e 65 77 43 6f 6e  nt(Blob *pNewCon
12f0: 74 65 6e 74 29 7b 0a 20 20 63 67 69 5f 72 65 73  tent){.  cgi_res
1300: 65 74 5f 63 6f 6e 74 65 6e 74 28 29 3b 0a 20 20  et_content();.  
1310: 63 67 69 5f 64 65 73 74 69 6e 61 74 69 6f 6e 28  cgi_destination(
1320: 43 47 49 5f 48 45 41 44 45 52 29 3b 0a 20 20 63  CGI_HEADER);.  c
1330: 67 69 43 6f 6e 74 65 6e 74 5b 30 5d 20 3d 20 2a  giContent[0] = *
1340: 70 4e 65 77 43 6f 6e 74 65 6e 74 3b 0a 20 20 62  pNewContent;.  b
1350: 6c 6f 62 5f 7a 65 72 6f 28 70 4e 65 77 43 6f 6e  lob_zero(pNewCon
1360: 74 65 6e 74 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  tent);.}../*.** 
1370: 53 65 74 20 74 68 65 20 72 65 70 6c 79 20 73 74  Set the reply st
1380: 61 74 75 73 20 63 6f 64 65 0a 2a 2f 0a 76 6f 69  atus code.*/.voi
1390: 64 20 63 67 69 5f 73 65 74 5f 73 74 61 74 75 73  d cgi_set_status
13a0: 28 69 6e 74 20 69 53 74 61 74 2c 20 63 6f 6e 73  (int iStat, cons
13b0: 74 20 63 68 61 72 20 2a 7a 53 74 61 74 29 7b 0a  t char *zStat){.
13c0: 20 20 7a 52 65 70 6c 79 53 74 61 74 75 73 20 3d    zReplyStatus =
13d0: 20 6d 70 72 69 6e 74 66 28 22 25 73 22 2c 20 7a   mprintf("%s", z
13e0: 53 74 61 74 29 3b 0a 20 20 69 52 65 70 6c 79 53  Stat);.  iReplyS
13f0: 74 61 74 75 73 20 3d 20 69 53 74 61 74 3b 0a 7d  tatus = iStat;.}
1400: 0a 0a 2f 2a 0a 2a 2a 20 41 70 70 65 6e 64 20 74  ../*.** Append t
1410: 65 78 74 20 74 6f 20 74 68 65 20 68 65 61 64 65  ext to the heade
1420: 72 20 6f 66 20 61 6e 20 48 54 54 50 20 72 65 70  r of an HTTP rep
1430: 6c 79 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 61  ly.*/.void cgi_a
1440: 70 70 65 6e 64 5f 68 65 61 64 65 72 28 63 6f 6e  ppend_header(con
1450: 73 74 20 63 68 61 72 20 2a 7a 4c 69 6e 65 29 7b  st char *zLine){
1460: 0a 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 28 26  .  blob_append(&
1470: 65 78 74 72 61 48 65 61 64 65 72 2c 20 7a 4c 69  extraHeader, zLi
1480: 6e 65 2c 20 2d 31 29 3b 0a 7d 0a 0a 2f 2a 0a 2a  ne, -1);.}../*.*
1490: 2a 20 53 65 74 20 61 20 63 6f 6f 6b 69 65 2e 0a  * Set a cookie..
14a0: 2a 2a 0a 2a 2a 20 5a 65 72 6f 20 6c 69 66 65 74  **.** Zero lifet
14b0: 69 6d 65 20 69 6d 70 6c 69 65 73 20 61 20 73 65  ime implies a se
14c0: 73 73 69 6f 6e 20 63 6f 6f 6b 69 65 2e 0a 2a 2f  ssion cookie..*/
14d0: 0a 76 6f 69 64 20 63 67 69 5f 73 65 74 5f 63 6f  .void cgi_set_co
14e0: 6f 6b 69 65 28 0a 20 20 63 6f 6e 73 74 20 63 68  okie(.  const ch
14f0: 61 72 20 2a 7a 4e 61 6d 65 2c 20 20 20 20 2f 2a  ar *zName,    /*
1500: 20 4e 61 6d 65 20 6f 66 20 74 68 65 20 63 6f 6f   Name of the coo
1510: 6b 69 65 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63  kie */.  const c
1520: 68 61 72 20 2a 7a 56 61 6c 75 65 2c 20 20 20 2f  har *zValue,   /
1530: 2a 20 56 61 6c 75 65 20 6f 66 20 74 68 65 20 63  * Value of the c
1540: 6f 6f 6b 69 65 2e 20 20 41 75 74 6f 6d 61 74 69  ookie.  Automati
1550: 63 61 6c 6c 79 20 65 73 63 61 70 65 64 20 2a 2f  cally escaped */
1560: 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a  .  const char *z
1570: 50 61 74 68 2c 20 20 20 20 2f 2a 20 50 61 74 68  Path,    /* Path
1580: 20 63 6f 6f 6b 69 65 20 61 70 70 6c 69 65 73 20   cookie applies 
1590: 74 6f 2e 20 20 4e 55 4c 4c 20 6d 65 61 6e 73 20  to.  NULL means 
15a0: 22 2f 22 20 2a 2f 0a 20 20 69 6e 74 20 6c 69 66  "/" */.  int lif
15b0: 65 74 69 6d 65 20 20 20 20 20 20 20 20 20 20 2f  etime          /
15c0: 2a 20 45 78 70 69 72 61 74 69 6f 6e 20 6f 66 20  * Expiration of 
15d0: 74 68 65 20 63 6f 6f 6b 69 65 20 69 6e 20 73 65  the cookie in se
15e0: 63 6f 6e 64 73 20 66 72 6f 6d 20 6e 6f 77 20 2a  conds from now *
15f0: 2f 0a 29 7b 0a 20 20 69 66 28 20 7a 50 61 74 68  /.){.  if( zPath
1600: 3d 3d 30 20 29 20 7a 50 61 74 68 20 3d 20 67 2e  ==0 ) zPath = g.
1610: 7a 54 6f 70 3b 0a 20 20 69 66 28 20 6c 69 66 65  zTop;.  if( life
1620: 74 69 6d 65 3e 30 20 29 7b 0a 20 20 20 20 6c 69  time>0 ){.    li
1630: 66 65 74 69 6d 65 20 2b 3d 20 28 69 6e 74 29 74  fetime += (int)t
1640: 69 6d 65 28 30 29 3b 0a 20 20 20 20 62 6c 6f 62  ime(0);.    blob
1650: 5f 61 70 70 65 6e 64 66 28 26 65 78 74 72 61 48  _appendf(&extraH
1660: 65 61 64 65 72 2c 0a 20 20 20 20 20 20 20 22 53  eader,.       "S
1670: 65 74 2d 43 6f 6f 6b 69 65 3a 20 25 73 3d 25 74  et-Cookie: %s=%t
1680: 3b 20 50 61 74 68 3d 25 73 3b 20 65 78 70 69 72  ; Path=%s; expir
1690: 65 73 3d 25 73 3b 20 56 65 72 73 69 6f 6e 3d 31  es=%s; Version=1
16a0: 5c 72 5c 6e 22 2c 0a 20 20 20 20 20 20 20 20 7a  \r\n",.        z
16b0: 4e 61 6d 65 2c 20 7a 56 61 6c 75 65 2c 20 7a 50  Name, zValue, zP
16c0: 61 74 68 2c 20 63 67 69 5f 72 66 63 38 32 32 5f  ath, cgi_rfc822_
16d0: 64 61 74 65 73 74 61 6d 70 28 6c 69 66 65 74 69  datestamp(lifeti
16e0: 6d 65 29 29 3b 0a 20 20 7d 65 6c 73 65 7b 0a 20  me));.  }else{. 
16f0: 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66 28     blob_appendf(
1700: 26 65 78 74 72 61 48 65 61 64 65 72 2c 0a 20 20  &extraHeader,.  
1710: 20 20 20 20 20 22 53 65 74 2d 43 6f 6f 6b 69 65       "Set-Cookie
1720: 3a 20 25 73 3d 25 74 3b 20 50 61 74 68 3d 25 73  : %s=%t; Path=%s
1730: 3b 20 56 65 72 73 69 6f 6e 3d 31 5c 72 5c 6e 22  ; Version=1\r\n"
1740: 2c 0a 20 20 20 20 20 20 20 7a 4e 61 6d 65 2c 20  ,.       zName, 
1750: 7a 56 61 6c 75 65 2c 20 7a 50 61 74 68 29 3b 0a  zValue, zPath);.
1760: 20 20 7d 0a 7d 0a 0a 23 69 66 20 30 0a 2f 2a 0a    }.}..#if 0./*.
1770: 2a 2a 20 41 64 64 20 61 6e 20 45 54 61 67 20 68  ** Add an ETag h
1780: 65 61 64 65 72 20 6c 69 6e 65 0a 2a 2f 0a 73 74  eader line.*/.st
1790: 61 74 69 63 20 63 68 61 72 20 2a 63 67 69 5f 61  atic char *cgi_a
17a0: 64 64 5f 65 74 61 67 28 63 68 61 72 20 2a 7a 54  dd_etag(char *zT
17b0: 78 74 2c 20 69 6e 74 20 6e 4c 65 6e 29 7b 0a 20  xt, int nLen){. 
17c0: 20 4d 44 35 43 6f 6e 74 65 78 74 20 63 74 78 3b   MD5Context ctx;
17d0: 0a 20 20 75 6e 73 69 67 6e 65 64 20 63 68 61 72  .  unsigned char
17e0: 20 64 69 67 65 73 74 5b 31 36 5d 3b 0a 20 20 69   digest[16];.  i
17f0: 6e 74 20 69 2c 20 6a 3b 0a 20 20 63 68 61 72 20  nt i, j;.  char 
1800: 7a 45 54 61 67 5b 36 34 5d 3b 0a 0a 20 20 4d 44  zETag[64];..  MD
1810: 35 49 6e 69 74 28 26 63 74 78 29 3b 0a 20 20 4d  5Init(&ctx);.  M
1820: 44 35 55 70 64 61 74 65 28 26 63 74 78 2c 7a 54  D5Update(&ctx,zT
1830: 78 74 2c 6e 4c 65 6e 29 3b 0a 20 20 4d 44 35 46  xt,nLen);.  MD5F
1840: 69 6e 61 6c 28 64 69 67 65 73 74 2c 26 63 74 78  inal(digest,&ctx
1850: 29 3b 0a 20 20 66 6f 72 28 6a 3d 69 3d 30 3b 20  );.  for(j=i=0; 
1860: 69 3c 31 36 3b 20 69 2b 2b 2c 6a 2b 3d 32 29 7b  i<16; i++,j+=2){
1870: 0a 20 20 20 20 62 70 72 69 6e 74 66 28 26 7a 45  .    bprintf(&zE
1880: 54 61 67 5b 6a 5d 2c 73 69 7a 65 6f 66 28 7a 45  Tag[j],sizeof(zE
1890: 54 61 67 29 2d 6a 2c 22 25 30 32 78 22 2c 28 69  Tag)-j,"%02x",(i
18a0: 6e 74 29 64 69 67 65 73 74 5b 69 5d 29 3b 0a 20  nt)digest[i]);. 
18b0: 20 7d 0a 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64   }.  blob_append
18c0: 66 28 26 65 78 74 72 61 48 65 61 64 65 72 2c 20  f(&extraHeader, 
18d0: 22 45 54 61 67 3a 20 25 73 5c 72 5c 6e 22 2c 20  "ETag: %s\r\n", 
18e0: 7a 45 54 61 67 29 3b 0a 20 20 72 65 74 75 72 6e  zETag);.  return
18f0: 20 73 74 72 64 75 70 28 7a 45 54 61 67 29 3b 0a   strdup(zETag);.
1900: 7d 0a 0a 2f 2a 0a 2a 2a 20 44 6f 20 73 6f 6d 65  }../*.** Do some
1910: 20 63 61 63 68 65 20 63 6f 6e 74 72 6f 6c 20 73   cache control s
1920: 74 75 66 66 2e 20 46 69 72 73 74 2c 20 77 65 20  tuff. First, we 
1930: 67 65 6e 65 72 61 74 65 20 61 6e 20 45 54 61 67  generate an ETag
1940: 20 61 6e 64 20 69 6e 63 6c 75 64 65 20 69 74 20   and include it 
1950: 69 6e 0a 2a 2a 20 74 68 65 20 72 65 73 70 6f 6e  in.** the respon
1960: 73 65 20 68 65 61 64 65 72 73 2e 20 53 65 63 6f  se headers. Seco
1970: 6e 64 2c 20 77 65 20 64 6f 20 77 68 61 74 65 76  nd, we do whatev
1980: 65 72 20 69 73 20 6e 65 63 65 73 73 61 72 79 20  er is necessary 
1990: 74 6f 20 64 65 74 65 72 6d 69 6e 65 20 69 66 0a  to determine if.
19a0: 2a 2a 20 74 68 65 20 72 65 71 75 65 73 74 20 77  ** the request w
19b0: 61 73 20 61 73 6b 69 6e 67 20 61 62 6f 75 74 20  as asking about 
19c0: 63 61 63 68 69 6e 67 20 61 6e 64 20 77 68 65 74  caching and whet
19d0: 68 65 72 20 77 65 20 6e 65 65 64 20 74 6f 20 73  her we need to s
19e0: 65 6e 64 20 62 61 63 6b 20 74 68 65 0a 2a 2a 20  end back the.** 
19f0: 72 65 73 70 6f 6e 73 65 20 62 6f 64 79 2e 20 49  response body. I
1a00: 66 20 77 65 20 73 68 6f 75 6c 64 6e 27 74 20 73  f we shouldn't s
1a10: 65 6e 64 20 61 20 62 6f 64 79 2c 20 72 65 74 75  end a body, retu
1a20: 72 6e 20 6e 6f 6e 2d 7a 65 72 6f 2e 0a 2a 2a 0a  rn non-zero..**.
1a30: 2a 2a 20 43 75 72 72 65 6e 74 6c 79 2c 20 77 65  ** Currently, we
1a40: 20 6a 75 73 74 20 63 68 65 63 6b 20 74 68 65 20   just check the 
1a50: 45 54 61 67 20 61 67 61 69 6e 73 74 20 61 6e 79  ETag against any
1a60: 20 49 66 2d 4e 6f 6e 65 2d 4d 61 74 63 68 20 68   If-None-Match h
1a70: 65 61 64 65 72 2e 0a 2a 2a 0a 2a 2a 20 46 49 58  eader..**.** FIX
1a80: 4d 45 3a 20 49 6e 20 73 6f 6d 65 20 63 61 73 65  ME: In some case
1a90: 73 20 28 61 74 74 61 63 68 6d 65 6e 74 73 2c 20  s (attachments, 
1aa0: 66 69 6c 65 20 63 6f 6e 74 65 6e 74 73 29 20 77  file contents) w
1ab0: 65 20 63 6f 75 6c 64 20 63 68 65 63 6b 0a 2a 2a  e could check.**
1ac0: 20 49 66 2d 4d 6f 64 69 66 69 65 64 2d 53 69 6e   If-Modified-Sin
1ad0: 63 65 20 68 65 61 64 65 72 73 20 61 6e 64 20 61  ce headers and a
1ae0: 6c 77 61 79 73 20 69 6e 63 6c 75 64 65 20 4c 61  lways include La
1af0: 73 74 2d 4d 6f 64 69 66 69 65 64 20 69 6e 20 72  st-Modified in r
1b00: 65 73 70 6f 6e 73 65 73 2e 0a 2a 2f 0a 73 74 61  esponses..*/.sta
1b10: 74 69 63 20 69 6e 74 20 63 68 65 63 6b 5f 63 61  tic int check_ca
1b20: 63 68 65 5f 63 6f 6e 74 72 6f 6c 28 76 6f 69 64  che_control(void
1b30: 29 7b 0a 20 20 2f 2a 20 46 49 58 4d 45 3a 20 74  ){.  /* FIXME: t
1b40: 68 65 72 65 27 73 20 73 6f 6d 65 20 67 6f 74 63  here's some gotc
1b50: 68 61 73 20 77 74 68 20 63 6f 6f 6b 69 65 73 20  has wth cookies 
1b60: 61 6e 64 20 73 6f 6d 65 20 68 65 61 64 65 72 73  and some headers
1b70: 2e 20 2a 2f 0a 20 20 63 68 61 72 20 2a 7a 45 54  . */.  char *zET
1b80: 61 67 20 3d 20 63 67 69 5f 61 64 64 5f 65 74 61  ag = cgi_add_eta
1b90: 67 28 62 6c 6f 62 5f 62 75 66 66 65 72 28 26 63  g(blob_buffer(&c
1ba0: 67 69 43 6f 6e 74 65 6e 74 29 2c 62 6c 6f 62 5f  giContent),blob_
1bb0: 73 69 7a 65 28 26 63 67 69 43 6f 6e 74 65 6e 74  size(&cgiContent
1bc0: 29 29 3b 0a 20 20 63 68 61 72 20 2a 7a 4d 61 74  ));.  char *zMat
1bd0: 63 68 20 3d 20 50 28 22 48 54 54 50 5f 49 46 5f  ch = P("HTTP_IF_
1be0: 4e 4f 4e 45 5f 4d 41 54 43 48 22 29 3b 0a 0a 20  NONE_MATCH");.. 
1bf0: 20 69 66 28 20 7a 45 54 61 67 21 3d 30 20 26 26   if( zETag!=0 &&
1c00: 20 7a 4d 61 74 63 68 21 3d 30 20 29 20 7b 0a 20   zMatch!=0 ) {. 
1c10: 20 20 20 63 68 61 72 20 2a 7a 42 75 66 20 3d 20     char *zBuf = 
1c20: 73 74 72 64 75 70 28 7a 4d 61 74 63 68 29 3b 0a  strdup(zMatch);.
1c30: 20 20 20 20 69 66 28 20 7a 42 75 66 21 3d 30 20      if( zBuf!=0 
1c40: 29 7b 0a 20 20 20 20 20 20 63 68 61 72 20 2a 7a  ){.      char *z
1c50: 54 6f 6b 20 3d 20 30 3b 0a 20 20 20 20 20 20 63  Tok = 0;.      c
1c60: 68 61 72 20 2a 7a 50 6f 73 3b 0a 20 20 20 20 20  har *zPos;.     
1c70: 20 66 6f 72 28 20 7a 54 6f 6b 20 3d 20 73 74 72   for( zTok = str
1c80: 74 6f 6b 5f 72 28 7a 42 75 66 2c 20 22 2c 5c 22  tok_r(zBuf, ",\"
1c90: 22 2c 26 7a 50 6f 73 29 3b 0a 20 20 20 20 20 20  ",&zPos);.      
1ca0: 20 20 20 20 20 7a 54 6f 6b 20 26 26 20 73 74 72       zTok && str
1cb0: 63 61 73 65 63 6d 70 28 7a 54 6f 6b 2c 7a 45 54  casecmp(zTok,zET
1cc0: 61 67 29 3b 0a 20 20 20 20 20 20 20 20 20 20 20  ag);.           
1cd0: 7a 54 6f 6b 20 3d 20 20 73 74 72 74 6f 6b 5f 72  zTok =  strtok_r
1ce0: 28 30 2c 20 22 2c 5c 22 22 2c 26 7a 50 6f 73 29  (0, ",\"",&zPos)
1cf0: 29 7b 7d 0a 20 20 20 20 20 20 66 72 65 65 28 7a  ){}.      free(z
1d00: 42 75 66 29 3b 0a 20 20 20 20 20 20 69 66 28 7a  Buf);.      if(z
1d10: 54 6f 6b 29 20 72 65 74 75 72 6e 20 31 3b 0a 20  Tok) return 1;. 
1d20: 20 20 20 7d 0a 20 20 7d 0a 20 20 0a 20 20 72 65     }.  }.  .  re
1d30: 74 75 72 6e 20 30 3b 0a 7d 0a 23 65 6e 64 69 66  turn 0;.}.#endif
1d40: 0a 0a 2f 2a 0a 2a 2a 20 44 6f 20 61 20 6e 6f 72  ../*.** Do a nor
1d50: 6d 61 6c 20 48 54 54 50 20 72 65 70 6c 79 0a 2a  mal HTTP reply.*
1d60: 2f 0a 76 6f 69 64 20 63 67 69 5f 72 65 70 6c 79  /.void cgi_reply
1d70: 28 76 6f 69 64 29 7b 0a 20 20 69 6e 74 20 74 6f  (void){.  int to
1d80: 74 61 6c 5f 73 69 7a 65 20 3d 20 30 3b 0a 20 20  tal_size = 0;.  
1d90: 69 66 28 20 69 52 65 70 6c 79 53 74 61 74 75 73  if( iReplyStatus
1da0: 3c 3d 30 20 29 7b 0a 20 20 20 20 69 52 65 70 6c  <=0 ){.    iRepl
1db0: 79 53 74 61 74 75 73 20 3d 20 32 30 30 3b 0a 20  yStatus = 200;. 
1dc0: 20 20 20 7a 52 65 70 6c 79 53 74 61 74 75 73 20     zReplyStatus 
1dd0: 3d 20 22 4f 4b 22 3b 0a 20 20 7d 0a 0a 23 69 66  = "OK";.  }..#if
1de0: 20 30 0a 20 20 69 66 28 20 69 52 65 70 6c 79 53   0.  if( iReplyS
1df0: 74 61 74 75 73 3d 3d 32 30 30 20 26 26 20 63 68  tatus==200 && ch
1e00: 65 63 6b 5f 63 61 63 68 65 5f 63 6f 6e 74 72 6f  eck_cache_contro
1e10: 6c 28 29 20 29 20 7b 0a 20 20 20 20 2f 2a 20 63  l() ) {.    /* c
1e20: 68 61 6e 67 65 20 74 68 65 20 73 74 61 74 75 73  hange the status
1e30: 20 74 6f 20 22 75 6e 63 68 61 6e 67 65 64 22 20   to "unchanged" 
1e40: 61 6e 64 20 77 65 20 63 61 6e 20 73 6b 69 70 20  and we can skip 
1e50: 73 65 6e 64 69 6e 67 20 74 68 65 0a 20 20 20 20  sending the.    
1e60: 2a 2a 20 61 63 74 75 61 6c 20 72 65 73 70 6f 6e  ** actual respon
1e70: 73 65 20 62 6f 64 79 2e 20 4f 62 76 69 6f 75 73  se body. Obvious
1e80: 6c 79 20 77 65 20 6f 6e 6c 79 20 64 6f 20 74 68  ly we only do th
1e90: 69 73 20 77 68 65 6e 20 77 65 20 5f 68 61 76 65  is when we _have
1ea0: 5f 20 61 0a 20 20 20 20 2a 2a 20 62 6f 64 79 20  _ a.    ** body 
1eb0: 28 63 6f 64 65 20 32 30 30 29 2e 0a 20 20 20 20  (code 200)..    
1ec0: 2a 2f 0a 20 20 20 20 69 52 65 70 6c 79 53 74 61  */.    iReplySta
1ed0: 74 75 73 20 3d 20 33 30 34 3b 0a 20 20 20 20 7a  tus = 304;.    z
1ee0: 52 65 70 6c 79 53 74 61 74 75 73 20 3d 20 22 4e  ReplyStatus = "N
1ef0: 6f 74 20 4d 6f 64 69 66 69 65 64 22 3b 0a 20 20  ot Modified";.  
1f00: 7d 0a 23 65 6e 64 69 66 0a 0a 20 20 69 66 28 20  }.#endif..  if( 
1f10: 66 75 6c 6c 48 74 74 70 52 65 70 6c 79 20 29 7b  fullHttpReply ){
1f20: 0a 20 20 20 20 66 70 72 69 6e 74 66 28 67 2e 68  .    fprintf(g.h
1f30: 74 74 70 4f 75 74 2c 20 22 48 54 54 50 2f 31 2e  ttpOut, "HTTP/1.
1f40: 30 20 25 64 20 25 73 5c 72 5c 6e 22 2c 20 69 52  0 %d %s\r\n", iR
1f50: 65 70 6c 79 53 74 61 74 75 73 2c 20 7a 52 65 70  eplyStatus, zRep
1f60: 6c 79 53 74 61 74 75 73 29 3b 0a 20 20 20 20 66  lyStatus);.    f
1f70: 70 72 69 6e 74 66 28 67 2e 68 74 74 70 4f 75 74  printf(g.httpOut
1f80: 2c 20 22 44 61 74 65 3a 20 25 73 5c 72 5c 6e 22  , "Date: %s\r\n"
1f90: 2c 20 63 67 69 5f 72 66 63 38 32 32 5f 64 61 74  , cgi_rfc822_dat
1fa0: 65 73 74 61 6d 70 28 74 69 6d 65 28 30 29 29 29  estamp(time(0)))
1fb0: 3b 0a 20 20 20 20 66 70 72 69 6e 74 66 28 67 2e  ;.    fprintf(g.
1fc0: 68 74 74 70 4f 75 74 2c 20 22 43 6f 6e 6e 65 63  httpOut, "Connec
1fd0: 74 69 6f 6e 3a 20 63 6c 6f 73 65 5c 72 5c 6e 22  tion: close\r\n"
1fe0: 29 3b 0a 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20  );.  }else{.    
1ff0: 66 70 72 69 6e 74 66 28 67 2e 68 74 74 70 4f 75  fprintf(g.httpOu
2000: 74 2c 20 22 53 74 61 74 75 73 3a 20 25 64 20 25  t, "Status: %d %
2010: 73 5c 72 5c 6e 22 2c 20 69 52 65 70 6c 79 53 74  s\r\n", iReplySt
2020: 61 74 75 73 2c 20 7a 52 65 70 6c 79 53 74 61 74  atus, zReplyStat
2030: 75 73 29 3b 0a 20 20 7d 0a 0a 20 20 69 66 28 20  us);.  }..  if( 
2040: 62 6c 6f 62 5f 73 69 7a 65 28 26 65 78 74 72 61  blob_size(&extra
2050: 48 65 61 64 65 72 29 3e 30 20 29 7b 0a 20 20 20  Header)>0 ){.   
2060: 20 66 70 72 69 6e 74 66 28 67 2e 68 74 74 70 4f   fprintf(g.httpO
2070: 75 74 2c 20 22 25 73 22 2c 20 62 6c 6f 62 5f 62  ut, "%s", blob_b
2080: 75 66 66 65 72 28 26 65 78 74 72 61 48 65 61 64  uffer(&extraHead
2090: 65 72 29 29 3b 0a 20 20 7d 0a 0a 20 20 69 66 28  er));.  }..  if(
20a0: 20 67 2e 69 73 43 6f 6e 73 74 20 29 7b 0a 20 20   g.isConst ){.  
20b0: 20 20 2f 2a 20 63 6f 6e 73 74 61 6e 74 20 6d 65    /* constant me
20c0: 61 6e 73 20 74 68 61 74 20 74 68 65 20 69 6e 70  ans that the inp
20d0: 75 74 20 55 52 4c 20 77 69 6c 6c 20 5f 6e 65 76  ut URL will _nev
20e0: 65 72 5f 20 67 65 6e 65 72 61 74 65 20 61 6e 79  er_ generate any
20f0: 74 68 69 6e 67 0a 20 20 20 20 2a 2a 20 65 6c 73  thing.    ** els
2100: 65 2e 20 49 6e 20 74 68 65 20 63 61 73 65 20 6f  e. In the case o
2110: 66 20 61 74 74 61 63 68 6d 65 6e 74 73 2c 20 74  f attachments, t
2120: 68 65 20 63 6f 6e 74 65 6e 74 73 20 77 6f 6e 27  he contents won'
2130: 74 20 63 68 61 6e 67 65 20 62 65 63 61 75 73 65  t change because
2140: 0a 20 20 20 20 2a 2a 20 61 6e 20 61 74 74 65 6d  .    ** an attem
2150: 70 74 20 74 6f 20 63 68 61 6e 67 65 20 74 68 65  pt to change the
2160: 6d 20 67 65 6e 65 72 61 74 65 73 20 61 20 6e 65  m generates a ne
2170: 77 20 61 74 74 61 63 68 6d 65 6e 74 20 6e 75 6d  w attachment num
2180: 62 65 72 2e 20 49 6e 20 74 68 65 0a 20 20 20 20  ber. In the.    
2190: 2a 2a 20 63 61 73 65 20 6f 66 20 6d 6f 73 74 20  ** case of most 
21a0: 2f 67 65 74 66 69 6c 65 20 63 61 6c 6c 73 20 66  /getfile calls f
21b0: 6f 72 20 73 70 65 63 69 66 69 63 20 76 65 72 73  or specific vers
21c0: 69 6f 6e 73 2c 20 74 68 65 20 6f 6e 6c 79 20 77  ions, the only w
21d0: 61 79 20 74 68 65 0a 20 20 20 20 2a 2a 20 63 6f  ay the.    ** co
21e0: 6e 74 65 6e 74 20 63 68 61 6e 67 65 73 20 69 73  ntent changes is
21f0: 20 69 66 20 73 6f 6d 65 6f 6e 65 20 62 72 65 61   if someone brea
2200: 6b 73 20 74 68 65 20 53 43 4d 2e 20 41 6e 64 20  ks the SCM. And 
2210: 69 66 20 74 68 61 74 20 68 61 70 70 65 6e 73 2c  if that happens,
2220: 20 61 0a 20 20 20 20 2a 2a 20 73 74 61 6c 65 20   a.    ** stale 
2230: 63 61 63 68 65 20 69 73 20 74 68 65 20 6c 65 61  cache is the lea
2240: 73 74 20 6f 66 20 74 68 65 20 70 72 6f 62 6c 65  st of the proble
2250: 6d 2e 20 53 6f 20 77 65 20 70 72 6f 76 69 64 65  m. So we provide
2260: 20 61 6e 20 45 78 70 69 72 65 73 0a 20 20 20 20   an Expires.    
2270: 2a 2a 20 68 65 61 64 65 72 20 73 65 74 20 74 6f  ** header set to
2280: 20 61 20 72 65 61 73 6f 6e 61 62 6c 65 20 70 65   a reasonable pe
2290: 72 69 6f 64 20 28 64 65 66 61 75 6c 74 3a 20 6f  riod (default: o
22a0: 6e 65 20 77 65 65 6b 29 2e 0a 20 20 20 20 2a 2f  ne week)..    */
22b0: 0a 20 20 20 20 2f 2a 74 69 6d 65 5f 74 20 65 78  .    /*time_t ex
22c0: 70 69 72 65 73 20 3d 20 74 69 6d 65 28 30 29 20  pires = time(0) 
22d0: 2b 20 61 74 6f 69 28 64 62 5f 63 6f 6e 66 69 67  + atoi(db_config
22e0: 28 22 63 6f 6e 73 74 61 6e 74 5f 65 78 70 69 72  ("constant_expir
22f0: 65 73 22 2c 22 36 30 34 38 30 30 22 29 29 3b 2a  es","604800"));*
2300: 2f 0a 20 20 20 20 74 69 6d 65 5f 74 20 65 78 70  /.    time_t exp
2310: 69 72 65 73 20 3d 20 74 69 6d 65 28 30 29 20 2b  ires = time(0) +
2320: 20 36 30 34 38 30 30 3b 0a 20 20 20 20 66 70 72   604800;.    fpr
2330: 69 6e 74 66 28 67 2e 68 74 74 70 4f 75 74 2c 20  intf(g.httpOut, 
2340: 22 45 78 70 69 72 65 73 3a 20 25 73 5c 72 5c 6e  "Expires: %s\r\n
2350: 22 2c 20 63 67 69 5f 72 66 63 38 32 32 5f 64 61  ", cgi_rfc822_da
2360: 74 65 73 74 61 6d 70 28 65 78 70 69 72 65 73 29  testamp(expires)
2370: 29 3b 0a 20 20 7d 0a 0a 20 20 2f 2a 20 43 6f 6e  );.  }..  /* Con
2380: 74 65 6e 74 20 69 6e 74 65 6e 64 65 64 20 66 6f  tent intended fo
2390: 72 20 6c 6f 67 67 65 64 20 69 6e 20 75 73 65 72  r logged in user
23a0: 73 20 73 68 6f 75 6c 64 20 6f 6e 6c 79 20 62 65  s should only be
23b0: 20 63 61 63 68 65 64 20 69 6e 0a 20 20 2a 2a 20   cached in.  ** 
23c0: 74 68 65 20 62 72 6f 77 73 65 72 2c 20 6e 6f 74  the browser, not
23d0: 20 73 6f 6d 65 20 73 68 61 72 65 64 20 6c 6f 63   some shared loc
23e0: 61 74 69 6f 6e 2e 0a 20 20 2a 2f 0a 20 20 66 70  ation..  */.  fp
23f0: 72 69 6e 74 66 28 67 2e 68 74 74 70 4f 75 74 2c  rintf(g.httpOut,
2400: 20 22 43 61 63 68 65 2d 63 6f 6e 74 72 6f 6c 3a   "Cache-control:
2410: 20 70 72 69 76 61 74 65 5c 72 5c 6e 22 29 3b 0a   private\r\n");.
2420: 0a 23 69 66 20 46 4f 53 53 49 4c 5f 49 31 38 4e  .#if FOSSIL_I18N
2430: 0a 20 20 66 70 72 69 6e 74 66 28 67 2e 68 74 74  .  fprintf(g.htt
2440: 70 4f 75 74 2c 0a 20 20 20 20 20 22 43 6f 6e 74  pOut,.     "Cont
2450: 65 6e 74 2d 54 79 70 65 3a 20 25 73 3b 20 63 68  ent-Type: %s; ch
2460: 61 72 73 65 74 3d 25 73 5c 72 5c 6e 22 2c 20 7a  arset=%s\r\n", z
2470: 43 6f 6e 74 65 6e 74 54 79 70 65 2c 20 6e 6c 5f  ContentType, nl_
2480: 6c 61 6e 67 69 6e 66 6f 28 43 4f 44 45 53 45 54  langinfo(CODESET
2490: 29 29 3b 0a 23 65 6c 73 65 0a 20 20 66 70 72 69  ));.#else.  fpri
24a0: 6e 74 66 28 67 2e 68 74 74 70 4f 75 74 2c 0a 20  ntf(g.httpOut,. 
24b0: 20 20 20 20 22 43 6f 6e 74 65 6e 74 2d 54 79 70      "Content-Typ
24c0: 65 3a 20 25 73 3b 20 63 68 61 72 73 65 74 3d 49  e: %s; charset=I
24d0: 53 4f 2d 38 38 35 39 2d 31 5c 72 5c 6e 22 2c 20  SO-8859-1\r\n", 
24e0: 7a 43 6f 6e 74 65 6e 74 54 79 70 65 29 3b 0a 23  zContentType);.#
24f0: 65 6e 64 69 66 0a 20 20 69 66 28 20 73 74 72 63  endif.  if( strc
2500: 6d 70 28 7a 43 6f 6e 74 65 6e 74 54 79 70 65 2c  mp(zContentType,
2510: 22 61 70 70 6c 69 63 61 74 69 6f 6e 2f 78 2d 66  "application/x-f
2520: 6f 73 73 69 6c 22 29 3d 3d 30 20 29 7b 0a 20 20  ossil")==0 ){.  
2530: 20 20 63 67 69 5f 63 6f 6d 62 69 6e 65 5f 68 65    cgi_combine_he
2540: 61 64 65 72 5f 61 6e 64 5f 62 6f 64 79 28 29 3b  ader_and_body();
2550: 0a 20 20 20 20 62 6c 6f 62 5f 63 6f 6d 70 72 65  .    blob_compre
2560: 73 73 28 26 63 67 69 43 6f 6e 74 65 6e 74 5b 30  ss(&cgiContent[0
2570: 5d 2c 20 26 63 67 69 43 6f 6e 74 65 6e 74 5b 30  ], &cgiContent[0
2580: 5d 29 3b 0a 20 20 7d 0a 0a 20 20 69 66 28 20 69  ]);.  }..  if( i
2590: 52 65 70 6c 79 53 74 61 74 75 73 20 21 3d 20 33  ReplyStatus != 3
25a0: 30 34 20 29 20 7b 0a 20 20 20 20 74 6f 74 61 6c  04 ) {.    total
25b0: 5f 73 69 7a 65 20 3d 20 62 6c 6f 62 5f 73 69 7a  _size = blob_siz
25c0: 65 28 26 63 67 69 43 6f 6e 74 65 6e 74 5b 30 5d  e(&cgiContent[0]
25d0: 29 20 2b 20 62 6c 6f 62 5f 73 69 7a 65 28 26 63  ) + blob_size(&c
25e0: 67 69 43 6f 6e 74 65 6e 74 5b 31 5d 29 3b 0a 20  giContent[1]);. 
25f0: 20 20 20 66 70 72 69 6e 74 66 28 67 2e 68 74 74     fprintf(g.htt
2600: 70 4f 75 74 2c 20 22 43 6f 6e 74 65 6e 74 2d 4c  pOut, "Content-L
2610: 65 6e 67 74 68 3a 20 25 64 5c 72 5c 6e 22 2c 20  ength: %d\r\n", 
2620: 74 6f 74 61 6c 5f 73 69 7a 65 29 3b 0a 20 20 7d  total_size);.  }
2630: 0a 20 20 66 70 72 69 6e 74 66 28 67 2e 68 74 74  .  fprintf(g.htt
2640: 70 4f 75 74 2c 20 22 5c 72 5c 6e 22 29 3b 0a 20  pOut, "\r\n");. 
2650: 20 69 66 28 20 74 6f 74 61 6c 5f 73 69 7a 65 3e   if( total_size>
2660: 30 20 26 26 20 69 52 65 70 6c 79 53 74 61 74 75  0 && iReplyStatu
2670: 73 20 21 3d 20 33 30 34 20 29 7b 0a 20 20 20 20  s != 304 ){.    
2680: 69 6e 74 20 69 2c 20 73 69 7a 65 3b 0a 20 20 20  int i, size;.   
2690: 20 66 6f 72 28 69 3d 30 3b 20 69 3c 32 3b 20 69   for(i=0; i<2; i
26a0: 2b 2b 29 7b 0a 20 20 20 20 20 20 73 69 7a 65 20  ++){.      size 
26b0: 3d 20 62 6c 6f 62 5f 73 69 7a 65 28 26 63 67 69  = blob_size(&cgi
26c0: 43 6f 6e 74 65 6e 74 5b 69 5d 29 3b 0a 20 20 20  Content[i]);.   
26d0: 20 20 20 69 66 28 20 73 69 7a 65 3e 30 20 29 7b     if( size>0 ){
26e0: 0a 20 20 20 20 20 20 20 20 66 77 72 69 74 65 28  .        fwrite(
26f0: 62 6c 6f 62 5f 62 75 66 66 65 72 28 26 63 67 69  blob_buffer(&cgi
2700: 43 6f 6e 74 65 6e 74 5b 69 5d 29 2c 20 31 2c 20  Content[i]), 1, 
2710: 73 69 7a 65 2c 20 67 2e 68 74 74 70 4f 75 74 29  size, g.httpOut)
2720: 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20 7d 0a  ;.      }.    }.
2730: 20 20 7d 0a 20 20 43 47 49 44 45 42 55 47 28 28    }.  CGIDEBUG((
2740: 22 44 4f 4e 45 5c 6e 22 29 29 3b 0a 7d 0a 0a 2f  "DONE\n"));.}../
2750: 2a 0a 2a 2a 20 44 6f 20 61 20 72 65 64 69 72 65  *.** Do a redire
2760: 63 74 20 72 65 71 75 65 73 74 20 74 6f 20 74 68  ct request to th
2770: 65 20 55 52 4c 20 67 69 76 65 6e 20 69 6e 20 74  e URL given in t
2780: 68 65 20 61 72 67 75 6d 65 6e 74 2e 0a 2a 2a 0a  he argument..**.
2790: 2a 2a 20 54 68 65 20 55 52 4c 20 6d 75 73 74 20  ** The URL must 
27a0: 62 65 20 72 65 6c 61 74 69 76 65 20 74 6f 20 74  be relative to t
27b0: 68 65 20 62 61 73 65 20 6f 66 20 74 68 65 20 66  he base of the f
27c0: 6f 73 73 69 6c 20 73 65 72 76 65 72 2e 0a 2a 2f  ossil server..*/
27d0: 0a 76 6f 69 64 20 63 67 69 5f 72 65 64 69 72 65  .void cgi_redire
27e0: 63 74 28 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a  ct(const char *z
27f0: 55 52 4c 29 7b 0a 20 20 63 68 61 72 20 2a 7a 4c  URL){.  char *zL
2800: 6f 63 61 74 69 6f 6e 3b 0a 20 20 43 47 49 44 45  ocation;.  CGIDE
2810: 42 55 47 28 28 22 72 65 64 69 72 65 63 74 20 74  BUG(("redirect t
2820: 6f 20 25 73 5c 6e 22 2c 20 7a 55 52 4c 29 29 3b  o %s\n", zURL));
2830: 0a 20 20 69 66 28 20 73 74 72 6e 63 6d 70 28 7a  .  if( strncmp(z
2840: 55 52 4c 2c 22 68 74 74 70 3a 22 2c 35 29 3d 3d  URL,"http:",5)==
2850: 30 20 7c 7c 20 73 74 72 6e 63 6d 70 28 7a 55 52  0 || strncmp(zUR
2860: 4c 2c 22 68 74 74 70 73 3a 22 2c 36 29 3d 3d 30  L,"https:",6)==0
2870: 20 7c 7c 20 2a 7a 55 52 4c 3d 3d 27 2f 27 20 29   || *zURL=='/' )
2880: 7b 0a 20 20 20 20 7a 4c 6f 63 61 74 69 6f 6e 20  {.    zLocation 
2890: 3d 20 6d 70 72 69 6e 74 66 28 22 4c 6f 63 61 74  = mprintf("Locat
28a0: 69 6f 6e 3a 20 25 73 5c 72 5c 6e 22 2c 20 7a 55  ion: %s\r\n", zU
28b0: 52 4c 29 3b 0a 20 20 7d 65 6c 73 65 7b 0a 20 20  RL);.  }else{.  
28c0: 20 20 7a 4c 6f 63 61 74 69 6f 6e 20 3d 20 6d 70    zLocation = mp
28d0: 72 69 6e 74 66 28 22 4c 6f 63 61 74 69 6f 6e 3a  rintf("Location:
28e0: 20 25 73 2f 25 73 5c 72 5c 6e 22 2c 20 67 2e 7a   %s/%s\r\n", g.z
28f0: 42 61 73 65 55 52 4c 2c 20 7a 55 52 4c 29 3b 0a  BaseURL, zURL);.
2900: 20 20 7d 0a 20 20 63 67 69 5f 61 70 70 65 6e 64    }.  cgi_append
2910: 5f 68 65 61 64 65 72 28 7a 4c 6f 63 61 74 69 6f  _header(zLocatio
2920: 6e 29 3b 0a 20 20 63 67 69 5f 72 65 73 65 74 5f  n);.  cgi_reset_
2930: 63 6f 6e 74 65 6e 74 28 29 3b 0a 20 20 63 67 69  content();.  cgi
2940: 5f 70 72 69 6e 74 66 28 22 3c 68 74 6d 6c 3e 5c  _printf("<html>\
2950: 6e 3c 70 3e 52 65 64 69 72 65 63 74 20 74 6f 20  n<p>Redirect to 
2960: 25 68 3c 2f 70 3e 5c 6e 3c 2f 68 74 6d 6c 3e 5c  %h</p>\n</html>\
2970: 6e 22 2c 20 7a 55 52 4c 29 3b 0a 20 20 63 67 69  n", zURL);.  cgi
2980: 5f 73 65 74 5f 73 74 61 74 75 73 28 33 30 32 2c  _set_status(302,
2990: 20 22 4d 6f 76 65 64 20 54 65 6d 70 6f 72 61 72   "Moved Temporar
29a0: 69 6c 79 22 29 3b 0a 20 20 66 72 65 65 28 7a 4c  ily");.  free(zL
29b0: 6f 63 61 74 69 6f 6e 29 3b 0a 20 20 63 67 69 5f  ocation);.  cgi_
29c0: 72 65 70 6c 79 28 29 3b 0a 20 20 65 78 69 74 28  reply();.  exit(
29d0: 30 29 3b 0a 7d 0a 76 6f 69 64 20 63 67 69 5f 72  0);.}.void cgi_r
29e0: 65 64 69 72 65 63 74 66 28 63 6f 6e 73 74 20 63  edirectf(const c
29f0: 68 61 72 20 2a 7a 46 6f 72 6d 61 74 2c 20 2e 2e  har *zFormat, ..
2a00: 2e 29 7b 0a 20 20 76 61 5f 6c 69 73 74 20 61 70  .){.  va_list ap
2a10: 3b 0a 20 20 76 61 5f 73 74 61 72 74 28 61 70 2c  ;.  va_start(ap,
2a20: 20 7a 46 6f 72 6d 61 74 29 3b 0a 20 20 63 67 69   zFormat);.  cgi
2a30: 5f 72 65 64 69 72 65 63 74 28 76 6d 70 72 69 6e  _redirect(vmprin
2a40: 74 66 28 7a 46 6f 72 6d 61 74 2c 20 61 70 29 29  tf(zFormat, ap))
2a50: 3b 0a 20 20 76 61 5f 65 6e 64 28 61 70 29 3b 0a  ;.  va_end(ap);.
2a60: 7d 0a 0a 2f 2a 0a 2a 2a 20 49 6e 66 6f 72 6d 61  }../*.** Informa
2a70: 74 69 6f 6e 20 61 62 6f 75 74 20 61 6c 6c 20 71  tion about all q
2a80: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 73 20  uery parameters 
2a90: 61 6e 64 20 63 6f 6f 6b 69 65 73 20 61 72 65 20  and cookies are 
2aa0: 73 74 6f 72 65 64 0a 2a 2a 20 69 6e 20 74 68 65  stored.** in the
2ab0: 73 65 20 76 61 72 69 61 62 6c 65 73 2e 0a 2a 2f  se variables..*/
2ac0: 0a 73 74 61 74 69 63 20 69 6e 74 20 6e 41 6c 6c  .static int nAll
2ad0: 6f 63 51 50 20 3d 20 30 3b 20 2f 2a 20 53 70 61  ocQP = 0; /* Spa
2ae0: 63 65 20 61 6c 6c 6f 63 61 74 65 64 20 66 6f 72  ce allocated for
2af0: 20 61 50 61 72 61 6d 51 50 5b 5d 20 2a 2f 0a 73   aParamQP[] */.s
2b00: 74 61 74 69 63 20 69 6e 74 20 6e 55 73 65 64 51  tatic int nUsedQ
2b10: 50 20 3d 20 30 3b 20 20 2f 2a 20 53 70 61 63 65  P = 0;  /* Space
2b20: 20 61 63 74 75 61 6c 6c 79 20 75 73 65 64 20 69   actually used i
2b30: 6e 20 61 50 61 72 61 6d 51 50 5b 5d 20 2a 2f 0a  n aParamQP[] */.
2b40: 73 74 61 74 69 63 20 69 6e 74 20 73 6f 72 74 51  static int sortQ
2b50: 50 20 3d 20 30 3b 20 20 20 2f 2a 20 54 72 75 65  P = 0;   /* True
2b60: 20 69 66 20 61 50 61 72 61 6d 51 50 5b 5d 20 6e   if aParamQP[] n
2b70: 65 65 64 73 20 73 6f 72 74 69 6e 67 20 2a 2f 0a  eeds sorting */.
2b80: 73 74 61 74 69 63 20 69 6e 74 20 73 65 71 51 50  static int seqQP
2b90: 20 3d 20 30 3b 20 20 20 20 2f 2a 20 53 65 71 75   = 0;    /* Sequ
2ba0: 65 6e 63 65 20 6e 75 6d 62 65 72 73 20 2a 2f 0a  ence numbers */.
2bb0: 73 74 61 74 69 63 20 73 74 72 75 63 74 20 51 50  static struct QP
2bc0: 61 72 61 6d 20 7b 20 20 20 2f 2a 20 4f 6e 65 20  aram {   /* One 
2bd0: 65 6e 74 72 79 20 66 6f 72 20 65 61 63 68 20 71  entry for each q
2be0: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 20 6f  uery parameter o
2bf0: 72 20 63 6f 6f 6b 69 65 20 2a 2f 0a 20 20 63 6f  r cookie */.  co
2c00: 6e 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65 3b  nst char *zName;
2c10: 20 20 20 20 20 20 20 20 2f 2a 20 50 61 72 61 6d          /* Param
2c20: 65 74 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 6e  eter or cookie n
2c30: 61 6d 65 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63  ame */.  const c
2c40: 68 61 72 20 2a 7a 56 61 6c 75 65 3b 20 20 20 20  har *zValue;    
2c50: 20 20 20 2f 2a 20 56 61 6c 75 65 20 6f 66 20 74     /* Value of t
2c60: 68 65 20 71 75 65 72 79 20 70 61 72 61 6d 65 74  he query paramet
2c70: 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 2a 2f 0a  er or cookie */.
2c80: 20 20 69 6e 74 20 73 65 71 3b 20 20 20 20 20 20    int seq;      
2c90: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4f              /* O
2ca0: 72 64 65 72 20 6f 66 20 69 6e 73 65 72 74 69 6f  rder of insertio
2cb0: 6e 20 2a 2f 0a 7d 20 2a 61 50 61 72 61 6d 51 50  n */.} *aParamQP
2cc0: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a  ;             /*
2cd0: 20 41 6e 20 61 72 72 61 79 20 6f 66 20 61 6c 6c   An array of all
2ce0: 20 70 61 72 61 6d 65 74 65 72 73 20 61 6e 64 20   parameters and 
2cf0: 63 6f 6f 6b 69 65 73 20 2a 2f 0a 0a 2f 2a 0a 2a  cookies */../*.*
2d00: 2a 20 41 64 64 20 61 6e 6f 74 68 65 72 20 71 75  * Add another qu
2d10: 65 72 79 20 70 61 72 61 6d 65 74 65 72 20 6f 72  ery parameter or
2d20: 20 63 6f 6f 6b 69 65 20 74 6f 20 74 68 65 20 70   cookie to the p
2d30: 61 72 61 6d 65 74 65 72 20 73 65 74 2e 0a 2a 2a  arameter set..**
2d40: 20 7a 4e 61 6d 65 20 69 73 20 74 68 65 20 6e 61   zName is the na
2d50: 6d 65 20 6f 66 20 74 68 65 20 71 75 65 72 79 20  me of the query 
2d60: 70 61 72 61 6d 65 74 65 72 20 6f 72 20 63 6f 6f  parameter or coo
2d70: 6b 69 65 20 61 6e 64 20 7a 56 61 6c 75 65 0a 2a  kie and zValue.*
2d80: 2a 20 69 73 20 69 74 73 20 66 75 6c 6c 79 20 64  * is its fully d
2d90: 65 63 6f 64 65 64 20 76 61 6c 75 65 2e 0a 2a 2a  ecoded value..**
2da0: 0a 2a 2a 20 7a 4e 61 6d 65 20 61 6e 64 20 7a 56  .** zName and zV
2db0: 61 6c 75 65 20 61 72 65 20 6e 6f 74 20 63 6f 70  alue are not cop
2dc0: 69 65 64 20 61 6e 64 20 6d 75 73 74 20 6e 6f 74  ied and must not
2dd0: 20 63 68 61 6e 67 65 20 6f 72 20 62 65 0a 2a 2a   change or be.**
2de0: 20 64 65 61 6c 6c 6f 63 61 74 65 64 20 61 66 74   deallocated aft
2df0: 65 72 20 74 68 69 73 20 72 6f 75 74 69 6e 65 20  er this routine 
2e00: 72 65 74 75 72 6e 73 2e 0a 2a 2f 0a 76 6f 69 64  returns..*/.void
2e10: 20 63 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 74   cgi_set_paramet
2e20: 65 72 5f 6e 6f 63 6f 70 79 28 63 6f 6e 73 74 20  er_nocopy(const 
2e30: 63 68 61 72 20 2a 7a 4e 61 6d 65 2c 20 63 6f 6e  char *zName, con
2e40: 73 74 20 63 68 61 72 20 2a 7a 56 61 6c 75 65 29  st char *zValue)
2e50: 7b 0a 20 20 69 66 28 20 6e 41 6c 6c 6f 63 51 50  {.  if( nAllocQP
2e60: 3c 3d 6e 55 73 65 64 51 50 20 29 7b 0a 20 20 20  <=nUsedQP ){.   
2e70: 20 6e 41 6c 6c 6f 63 51 50 20 3d 20 6e 41 6c 6c   nAllocQP = nAll
2e80: 6f 63 51 50 2a 32 20 2b 20 31 30 3b 0a 20 20 20  ocQP*2 + 10;.   
2e90: 20 61 50 61 72 61 6d 51 50 20 3d 20 72 65 61 6c   aParamQP = real
2ea0: 6c 6f 63 28 20 61 50 61 72 61 6d 51 50 2c 20 6e  loc( aParamQP, n
2eb0: 41 6c 6c 6f 63 51 50 2a 73 69 7a 65 6f 66 28 61  AllocQP*sizeof(a
2ec0: 50 61 72 61 6d 51 50 5b 30 5d 29 20 29 3b 0a 20  ParamQP[0]) );. 
2ed0: 20 20 20 69 66 28 20 61 50 61 72 61 6d 51 50 3d     if( aParamQP=
2ee0: 3d 30 20 29 20 65 78 69 74 28 31 29 3b 0a 20 20  =0 ) exit(1);.  
2ef0: 7d 0a 20 20 61 50 61 72 61 6d 51 50 5b 6e 55 73  }.  aParamQP[nUs
2f00: 65 64 51 50 5d 2e 7a 4e 61 6d 65 20 3d 20 7a 4e  edQP].zName = zN
2f10: 61 6d 65 3b 0a 20 20 61 50 61 72 61 6d 51 50 5b  ame;.  aParamQP[
2f20: 6e 55 73 65 64 51 50 5d 2e 7a 56 61 6c 75 65 20  nUsedQP].zValue 
2f30: 3d 20 7a 56 61 6c 75 65 3b 0a 20 20 61 50 61 72  = zValue;.  aPar
2f40: 61 6d 51 50 5b 6e 55 73 65 64 51 50 5d 2e 73 65  amQP[nUsedQP].se
2f50: 71 20 3d 20 73 65 71 51 50 2b 2b 3b 0a 20 20 6e  q = seqQP++;.  n
2f60: 55 73 65 64 51 50 2b 2b 3b 0a 20 20 73 6f 72 74  UsedQP++;.  sort
2f70: 51 50 20 3d 20 31 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a  QP = 1;.}../*.**
2f80: 20 41 64 64 20 61 6e 6f 74 68 65 72 20 71 75 65   Add another que
2f90: 72 79 20 70 61 72 61 6d 65 74 65 72 20 6f 72 20  ry parameter or 
2fa0: 63 6f 6f 6b 69 65 20 74 6f 20 74 68 65 20 70 61  cookie to the pa
2fb0: 72 61 6d 65 74 65 72 20 73 65 74 2e 0a 2a 2a 20  rameter set..** 
2fc0: 7a 4e 61 6d 65 20 69 73 20 74 68 65 20 6e 61 6d  zName is the nam
2fd0: 65 20 6f 66 20 74 68 65 20 71 75 65 72 79 20 70  e of the query p
2fe0: 61 72 61 6d 65 74 65 72 20 6f 72 20 63 6f 6f 6b  arameter or cook
2ff0: 69 65 20 61 6e 64 20 7a 56 61 6c 75 65 0a 2a 2a  ie and zValue.**
3000: 20 69 73 20 69 74 73 20 66 75 6c 6c 79 20 64 65   is its fully de
3010: 63 6f 64 65 64 20 76 61 6c 75 65 2e 0a 2a 2a 0a  coded value..**.
3020: 2a 2a 20 43 6f 70 69 65 73 20 61 72 65 20 6d 61  ** Copies are ma
3030: 64 65 20 6f 66 20 62 6f 74 68 20 74 68 65 20 7a  de of both the z
3040: 4e 61 6d 65 20 61 6e 64 20 7a 56 61 6c 75 65 20  Name and zValue 
3050: 70 61 72 61 6d 65 74 65 72 73 2e 0a 2a 2f 0a 76  parameters..*/.v
3060: 6f 69 64 20 63 67 69 5f 73 65 74 5f 70 61 72 61  oid cgi_set_para
3070: 6d 65 74 65 72 28 63 6f 6e 73 74 20 63 68 61 72  meter(const char
3080: 20 2a 7a 4e 61 6d 65 2c 20 63 6f 6e 73 74 20 63   *zName, const c
3090: 68 61 72 20 2a 7a 56 61 6c 75 65 29 7b 0a 20 20  har *zValue){.  
30a0: 63 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 74 65  cgi_set_paramete
30b0: 72 5f 6e 6f 63 6f 70 79 28 6d 70 72 69 6e 74 66  r_nocopy(mprintf
30c0: 28 22 25 73 22 2c 7a 4e 61 6d 65 29 2c 20 6d 70  ("%s",zName), mp
30d0: 72 69 6e 74 66 28 22 25 73 22 2c 7a 56 61 6c 75  rintf("%s",zValu
30e0: 65 29 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 52 65  e));.}../*.** Re
30f0: 70 6c 61 63 65 20 61 20 70 61 72 61 6d 65 74 65  place a paramete
3100: 72 20 77 69 74 68 20 61 20 6e 65 77 20 76 61 6c  r with a new val
3110: 75 65 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f  ue..*/.void cgi_
3120: 72 65 70 6c 61 63 65 5f 70 61 72 61 6d 65 74 65  replace_paramete
3130: 72 28 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e  r(const char *zN
3140: 61 6d 65 2c 20 63 6f 6e 73 74 20 63 68 61 72 20  ame, const char 
3150: 2a 7a 56 61 6c 75 65 29 7b 0a 20 20 69 6e 74 20  *zValue){.  int 
3160: 69 3b 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69 3c  i;.  for(i=0; i<
3170: 6e 55 73 65 64 51 50 3b 20 69 2b 2b 29 7b 0a 20  nUsedQP; i++){. 
3180: 20 20 20 69 66 28 20 73 74 72 63 6d 70 28 61 50     if( strcmp(aP
3190: 61 72 61 6d 51 50 5b 69 5d 2e 7a 4e 61 6d 65 2c  aramQP[i].zName,
31a0: 7a 4e 61 6d 65 29 3d 3d 30 20 29 7b 0a 20 20 20  zName)==0 ){.   
31b0: 20 20 20 61 50 61 72 61 6d 51 50 5b 69 5d 2e 7a     aParamQP[i].z
31c0: 56 61 6c 75 65 20 3d 20 7a 56 61 6c 75 65 3b 0a  Value = zValue;.
31d0: 20 20 20 20 7d 0a 20 20 7d 0a 7d 0a 0a 2f 2a 0a      }.  }.}../*.
31e0: 2a 2a 20 41 64 64 20 61 20 71 75 65 72 79 20 70  ** Add a query p
31f0: 61 72 61 6d 65 74 65 72 2e 20 20 54 68 65 20 7a  arameter.  The z
3200: 4e 61 6d 65 20 70 6f 72 74 69 6f 6e 20 69 73 20  Name portion is 
3210: 66 69 78 65 64 20 62 75 74 20 61 20 63 6f 70 79  fixed but a copy
3220: 0a 2a 2a 20 6d 75 73 74 20 62 65 20 6d 61 64 65  .** must be made
3230: 20 6f 66 20 7a 56 61 6c 75 65 2e 0a 2a 2f 0a 76   of zValue..*/.v
3240: 6f 69 64 20 63 67 69 5f 73 65 74 65 6e 76 28 63  oid cgi_setenv(c
3250: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65  onst char *zName
3260: 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 56  , const char *zV
3270: 61 6c 75 65 29 7b 0a 20 20 63 67 69 5f 73 65 74  alue){.  cgi_set
3280: 5f 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70  _parameter_nocop
3290: 79 28 7a 4e 61 6d 65 2c 20 6d 70 72 69 6e 74 66  y(zName, mprintf
32a0: 28 22 25 73 22 2c 7a 56 61 6c 75 65 29 29 3b 0a  ("%s",zValue));.
32b0: 7d 0a 20 0a 0a 2f 2a 0a 2a 2a 20 41 64 64 20 61  }. ../*.** Add a
32c0: 20 6c 69 73 74 20 6f 66 20 71 75 65 72 79 20 70   list of query p
32d0: 61 72 61 6d 65 74 65 72 73 20 6f 72 20 63 6f 6f  arameters or coo
32e0: 6b 69 65 73 20 74 6f 20 74 68 65 20 70 61 72 61  kies to the para
32f0: 6d 65 74 65 72 20 73 65 74 2e 0a 2a 2a 0a 2a 2a  meter set..**.**
3300: 20 45 61 63 68 20 70 61 72 61 6d 65 74 65 72 20   Each parameter 
3310: 69 73 20 6f 66 20 74 68 65 20 66 6f 72 6d 20 4e  is of the form N
3320: 41 4d 45 3d 56 41 4c 55 45 2e 20 20 42 6f 74 68  AME=VALUE.  Both
3330: 20 74 68 65 20 4e 41 4d 45 20 61 6e 64 20 74 68   the NAME and th
3340: 65 0a 2a 2a 20 56 41 4c 55 45 20 6d 61 79 20 62  e.** VALUE may b
3350: 65 20 75 72 6c 2d 65 6e 63 6f 64 65 64 20 28 22  e url-encoded ("
3360: 2b 22 20 66 6f 72 20 73 70 61 63 65 2c 20 22 25  +" for space, "%
3370: 48 48 22 20 66 6f 72 20 6f 74 68 65 72 20 73 70  HH" for other sp
3380: 65 63 69 61 6c 0a 2a 2a 20 63 68 61 72 61 63 74  ecial.** charact
3390: 65 72 73 29 2e 20 20 42 75 74 20 74 68 69 73 20  ers).  But this 
33a0: 72 6f 75 74 69 6e 65 20 61 73 73 75 6d 65 73 20  routine assumes 
33b0: 74 68 61 74 20 4e 41 4d 45 20 63 6f 6e 74 61 69  that NAME contai
33c0: 6e 73 20 6e 6f 0a 2a 2a 20 73 70 65 63 69 61 6c  ns no.** special
33d0: 20 63 68 61 72 61 63 74 65 72 20 61 6e 64 20 74   character and t
33e0: 68 65 72 65 66 6f 72 65 20 64 6f 65 73 20 6e 6f  herefore does no
33f0: 74 20 64 65 63 6f 64 65 20 69 74 2e 0a 2a 2a 0a  t decode it..**.
3400: 2a 2a 20 49 66 20 4e 41 4d 45 20 62 65 67 69 6e  ** If NAME begin
3410: 73 20 77 69 74 68 20 61 6e 6f 74 68 65 72 20 6f  s with another o
3420: 74 68 65 72 20 74 68 61 6e 20 61 20 6c 6f 77 65  ther than a lowe
3430: 72 2d 63 61 73 65 20 6c 65 74 74 65 72 20 74 68  r-case letter th
3440: 65 6e 0a 2a 2a 20 74 68 65 20 65 6e 74 69 72 65  en.** the entire
3450: 20 4e 41 4d 45 3d 56 41 4c 55 45 20 74 65 72 6d   NAME=VALUE term
3460: 20 69 73 20 69 67 6e 6f 72 65 64 2e 20 20 48 65   is ignored.  He
3470: 6e 63 65 3a 0a 2a 2a 0a 2a 2a 20 20 20 20 20 20  nce:.**.**      
3480: 2a 20 20 63 6f 6f 6b 69 65 73 20 61 6e 64 20 71  *  cookies and q
3490: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 73 20  uery parameters 
34a0: 74 68 61 74 20 68 61 76 65 20 75 70 70 65 72 63  that have upperc
34b0: 61 73 65 20 6e 61 6d 65 73 0a 2a 2a 20 20 20 20  ase names.**    
34c0: 20 20 20 20 20 61 72 65 20 69 67 6e 6f 72 65 64       are ignored
34d0: 2e 0a 2a 2a 0a 2a 2a 20 20 20 20 20 20 2a 20 20  ..**.**      *  
34e0: 69 74 20 69 73 20 69 6d 70 6f 73 73 69 62 6c 65  it is impossible
34f0: 20 66 6f 72 20 61 20 63 6f 6f 6b 69 65 20 6f 72   for a cookie or
3500: 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72   query parameter
3510: 20 74 6f 0a 2a 2a 20 20 20 20 20 20 20 20 20 6f   to.**         o
3520: 76 65 72 72 69 64 65 20 74 68 65 20 76 61 6c 75  verride the valu
3530: 65 20 6f 66 20 61 6e 20 65 6e 76 69 72 6f 6e 6d  e of an environm
3540: 65 6e 74 20 76 61 72 69 61 62 6c 65 20 73 69 6e  ent variable sin
3550: 63 65 0a 2a 2a 20 20 20 20 20 20 20 20 20 65 6e  ce.**         en
3560: 76 69 72 6f 6e 6d 65 6e 74 20 76 61 72 69 61 62  vironment variab
3570: 6c 65 73 20 61 6c 77 61 79 73 20 68 61 76 65 20  les always have 
3580: 75 70 70 65 72 63 61 73 65 20 6e 61 6d 65 73 2e  uppercase names.
3590: 0a 2a 2a 0a 2a 2a 20 50 61 72 61 6d 65 74 65 72  .**.** Parameter
35a0: 73 20 61 72 65 20 73 65 70 61 72 61 74 65 64 20  s are separated 
35b0: 62 79 20 74 68 65 20 22 74 65 72 6d 69 6e 61 74  by the "terminat
35c0: 6f 72 22 20 63 68 61 72 61 63 74 65 72 2e 20 20  or" character.  
35d0: 57 68 69 74 65 73 70 61 63 65 0a 2a 2a 20 62 65  Whitespace.** be
35e0: 66 6f 72 65 20 74 68 65 20 4e 41 4d 45 20 69 73  fore the NAME is
35f0: 20 69 67 6e 6f 72 65 64 2e 0a 2a 2a 0a 2a 2a 20   ignored..**.** 
3600: 54 68 65 20 69 6e 70 75 74 20 73 74 72 69 6e 67  The input string
3610: 20 22 7a 22 20 69 73 20 6d 6f 64 69 66 69 65 64   "z" is modified
3620: 20 62 75 74 20 6e 6f 20 63 6f 70 69 65 73 20 69   but no copies i
3630: 73 20 6d 61 64 65 2e 20 20 22 7a 22 0a 2a 2a 20  s made.  "z".** 
3640: 73 68 6f 75 6c 64 20 6e 6f 74 20 62 65 20 64 65  should not be de
3650: 61 6c 6c 6f 63 61 74 65 64 20 6f 72 20 63 68 61  allocated or cha
3660: 6e 67 65 64 20 61 67 61 69 6e 20 61 66 74 65 72  nged again after
3670: 20 74 68 69 73 20 72 6f 75 74 69 6e 65 0a 2a 2a   this routine.**
3680: 20 72 65 74 75 72 6e 73 20 6f 72 20 69 74 20 77   returns or it w
3690: 69 6c 6c 20 63 6f 72 72 75 70 74 20 74 68 65 20  ill corrupt the 
36a0: 70 61 72 61 6d 65 74 65 72 20 74 61 62 6c 65 2e  parameter table.
36b0: 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20  .*/.static void 
36c0: 61 64 64 5f 70 61 72 61 6d 5f 6c 69 73 74 28 63  add_param_list(c
36d0: 68 61 72 20 2a 7a 2c 20 69 6e 74 20 74 65 72 6d  har *z, int term
36e0: 69 6e 61 74 6f 72 29 7b 0a 20 20 77 68 69 6c 65  inator){.  while
36f0: 28 20 2a 7a 20 29 7b 0a 20 20 20 20 63 68 61 72  ( *z ){.    char
3700: 20 2a 7a 4e 61 6d 65 3b 0a 20 20 20 20 63 68 61   *zName;.    cha
3710: 72 20 2a 7a 56 61 6c 75 65 3b 0a 20 20 20 20 77  r *zValue;.    w
3720: 68 69 6c 65 28 20 69 73 73 70 61 63 65 28 2a 7a  hile( isspace(*z
3730: 29 20 29 7b 20 7a 2b 2b 3b 20 7d 0a 20 20 20 20  ) ){ z++; }.    
3740: 7a 4e 61 6d 65 20 3d 20 7a 3b 0a 20 20 20 20 77  zName = z;.    w
3750: 68 69 6c 65 28 20 2a 7a 20 26 26 20 2a 7a 21 3d  hile( *z && *z!=
3760: 27 3d 27 20 26 26 20 2a 7a 21 3d 74 65 72 6d 69  '=' && *z!=termi
3770: 6e 61 74 6f 72 20 29 7b 20 7a 2b 2b 3b 20 7d 0a  nator ){ z++; }.
3780: 20 20 20 20 69 66 28 20 2a 7a 3d 3d 27 3d 27 20      if( *z=='=' 
3790: 29 7b 0a 20 20 20 20 20 20 2a 7a 20 3d 20 30 3b  ){.      *z = 0;
37a0: 0a 20 20 20 20 20 20 7a 2b 2b 3b 0a 20 20 20 20  .      z++;.    
37b0: 20 20 7a 56 61 6c 75 65 20 3d 20 7a 3b 0a 20 20    zValue = z;.  
37c0: 20 20 20 20 77 68 69 6c 65 28 20 2a 7a 20 26 26      while( *z &&
37d0: 20 2a 7a 21 3d 74 65 72 6d 69 6e 61 74 6f 72 20   *z!=terminator 
37e0: 29 7b 20 7a 2b 2b 3b 20 7d 0a 20 20 20 20 20 20  ){ z++; }.      
37f0: 69 66 28 20 2a 7a 20 29 7b 0a 20 20 20 20 20 20  if( *z ){.      
3800: 20 20 2a 7a 20 3d 20 30 3b 0a 20 20 20 20 20 20    *z = 0;.      
3810: 20 20 7a 2b 2b 3b 0a 20 20 20 20 20 20 7d 0a 20    z++;.      }. 
3820: 20 20 20 20 20 64 65 68 74 74 70 69 7a 65 28 7a       dehttpize(z
3830: 56 61 6c 75 65 29 3b 0a 20 20 20 20 7d 65 6c 73  Value);.    }els
3840: 65 7b 0a 20 20 20 20 20 20 69 66 28 20 2a 7a 20  e{.      if( *z 
3850: 29 7b 20 2a 7a 2b 2b 20 3d 20 30 3b 20 7d 0a 20  ){ *z++ = 0; }. 
3860: 20 20 20 20 20 7a 56 61 6c 75 65 20 3d 20 22 22       zValue = ""
3870: 3b 0a 20 20 20 20 7d 0a 20 20 20 20 69 66 28 20  ;.    }.    if( 
3880: 69 73 6c 6f 77 65 72 28 7a 4e 61 6d 65 5b 30 5d  islower(zName[0]
3890: 29 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73  ) ){.      cgi_s
38a0: 65 74 5f 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63  et_parameter_noc
38b0: 6f 70 79 28 7a 4e 61 6d 65 2c 20 7a 56 61 6c 75  opy(zName, zValu
38c0: 65 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 7d 0a  e);.    }.  }.}.
38d0: 0a 2f 2a 0a 2a 2a 20 2a 70 7a 20 69 73 20 61 20  ./*.** *pz is a 
38e0: 73 74 72 69 6e 67 20 74 68 61 74 20 63 6f 6e 73  string that cons
38f0: 69 73 74 73 20 6f 66 20 6d 75 6c 74 69 70 6c 65  ists of multiple
3900: 20 6c 69 6e 65 73 20 6f 66 20 74 65 78 74 2e 20   lines of text. 
3910: 20 54 68 69 73 0a 2a 2a 20 72 6f 75 74 69 6e 65   This.** routine
3920: 20 66 69 6e 64 73 20 74 68 65 20 65 6e 64 20 6f   finds the end o
3930: 66 20 74 68 65 20 63 75 72 72 65 6e 74 20 6c 69  f the current li
3940: 6e 65 20 6f 66 20 74 65 78 74 20 61 6e 64 20 63  ne of text and c
3950: 6f 6e 76 65 72 74 73 0a 2a 2a 20 74 68 65 20 22  onverts.** the "
3960: 5c 6e 22 20 6f 72 20 22 5c 72 5c 6e 22 20 74 68  \n" or "\r\n" th
3970: 61 74 20 65 6e 64 73 20 74 68 61 74 20 6c 69 6e  at ends that lin
3980: 65 20 69 6e 74 6f 20 61 20 22 5c 30 30 30 22 2e  e into a "\000".
3990: 20 20 49 74 20 74 68 65 6e 0a 2a 2a 20 61 64 76    It then.** adv
39a0: 61 6e 63 65 73 20 2a 70 7a 20 74 6f 20 74 68 65  ances *pz to the
39b0: 20 62 65 67 69 6e 6e 69 6e 67 20 6f 66 20 74 68   beginning of th
39c0: 65 20 6e 65 78 74 20 6c 69 6e 65 20 61 6e 64 20  e next line and 
39d0: 72 65 74 75 72 6e 73 20 74 68 65 0a 2a 2a 20 70  returns the.** p
39e0: 72 65 76 69 6f 75 73 20 76 61 6c 75 65 20 6f 66  revious value of
39f0: 20 2a 70 7a 20 28 77 68 69 63 68 20 69 73 20 74   *pz (which is t
3a00: 68 65 20 73 74 61 72 74 20 6f 66 20 74 68 65 20  he start of the 
3a10: 63 75 72 72 65 6e 74 20 6c 69 6e 65 2e 29 0a 2a  current line.).*
3a20: 2f 0a 73 74 61 74 69 63 20 63 68 61 72 20 2a 67  /.static char *g
3a30: 65 74 5f 6c 69 6e 65 5f 66 72 6f 6d 5f 73 74 72  et_line_from_str
3a40: 69 6e 67 28 63 68 61 72 20 2a 2a 70 7a 2c 20 69  ing(char **pz, i
3a50: 6e 74 20 2a 70 4c 65 6e 29 7b 0a 20 20 63 68 61  nt *pLen){.  cha
3a60: 72 20 2a 7a 20 3d 20 2a 70 7a 3b 0a 20 20 69 6e  r *z = *pz;.  in
3a70: 74 20 69 3b 0a 20 20 69 66 28 20 7a 5b 30 5d 3d  t i;.  if( z[0]=
3a80: 3d 30 20 29 20 72 65 74 75 72 6e 20 30 3b 0a 20  =0 ) return 0;. 
3a90: 20 66 6f 72 28 69 3d 30 3b 20 7a 5b 69 5d 3b 20   for(i=0; z[i]; 
3aa0: 69 2b 2b 29 7b 0a 20 20 20 20 69 66 28 20 7a 5b  i++){.    if( z[
3ab0: 69 5d 3d 3d 27 5c 6e 27 20 29 7b 0a 20 20 20 20  i]=='\n' ){.    
3ac0: 20 20 69 66 28 20 69 3e 30 20 26 26 20 7a 5b 69    if( i>0 && z[i
3ad0: 2d 31 5d 3d 3d 27 5c 72 27 20 29 7b 0a 20 20 20  -1]=='\r' ){.   
3ae0: 20 20 20 20 20 7a 5b 69 2d 31 5d 20 3d 20 30 3b       z[i-1] = 0;
3af0: 0a 20 20 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20  .      }else{.  
3b00: 20 20 20 20 20 20 7a 5b 69 5d 20 3d 20 30 3b 0a        z[i] = 0;.
3b10: 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 69 2b        }.      i+
3b20: 2b 3b 0a 20 20 20 20 20 20 62 72 65 61 6b 3b 0a  +;.      break;.
3b30: 20 20 20 20 7d 0a 20 20 7d 0a 20 20 2a 70 7a 20      }.  }.  *pz 
3b40: 3d 20 26 7a 5b 69 5d 3b 0a 20 20 2a 70 4c 65 6e  = &z[i];.  *pLen
3b50: 20 2d 3d 20 69 3b 0a 20 20 72 65 74 75 72 6e 20   -= i;.  return 
3b60: 7a 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 65 20  z;.}../*.** The 
3b70: 69 6e 70 75 74 20 2a 70 7a 20 70 6f 69 6e 74 73  input *pz points
3b80: 20 74 6f 20 63 6f 6e 74 65 6e 74 20 74 68 61 74   to content that
3b90: 20 69 73 20 74 65 72 6d 69 6e 61 74 65 64 20 62   is terminated b
3ba0: 79 20 61 20 22 5c 72 5c 6e 22 0a 2a 2a 20 66 6f  y a "\r\n".** fo
3bb0: 6c 6c 6f 77 65 64 20 62 79 20 74 68 65 20 62 6f  llowed by the bo
3bc0: 75 6e 64 72 79 20 6d 61 72 6b 65 72 20 7a 42 6f  undry marker zBo
3bd0: 75 6e 64 72 79 2e 20 20 41 6e 20 65 78 74 72 61  undry.  An extra
3be0: 20 22 2d 2d 22 20 6d 61 79 20 6f 72 0a 2a 2a 20   "--" may or.** 
3bf0: 6d 61 79 20 6e 6f 74 20 62 65 20 61 70 70 65 6e  may not be appen
3c00: 64 65 64 20 74 6f 20 74 68 65 20 62 6f 75 6e 64  ded to the bound
3c10: 72 79 20 6d 61 72 6b 65 72 2e 20 20 54 68 65 72  ry marker.  Ther
3c20: 65 20 61 72 65 20 2a 70 4c 65 6e 20 63 68 61 72  e are *pLen char
3c30: 61 63 74 65 72 73 0a 2a 2a 20 69 6e 20 2a 70 7a  acters.** in *pz
3c40: 2e 0a 2a 2a 0a 2a 2a 20 54 68 69 73 20 72 6f 75  ..**.** This rou
3c50: 74 69 6e 65 20 61 64 64 73 20 61 20 22 5c 30 30  tine adds a "\00
3c60: 30 22 20 74 6f 20 74 68 65 20 65 6e 64 20 6f 66  0" to the end of
3c70: 20 74 68 65 20 63 6f 6e 74 65 6e 74 20 28 6f 76   the content (ov
3c80: 65 72 77 72 69 74 69 6e 67 0a 2a 2a 20 74 68 65  erwriting.** the
3c90: 20 22 5c 72 5c 6e 22 29 20 61 6e 64 20 72 65 74   "\r\n") and ret
3ca0: 75 72 6e 73 20 61 20 70 6f 69 6e 74 65 72 20 74  urns a pointer t
3cb0: 6f 20 74 68 65 20 63 6f 6e 74 65 6e 74 2e 20 20  o the content.  
3cc0: 54 68 65 20 2a 70 7a 20 69 6e 70 75 74 0a 2a 2a  The *pz input.**
3cd0: 20 69 73 20 61 64 6a 75 73 74 65 64 20 74 6f 20   is adjusted to 
3ce0: 70 6f 69 6e 74 20 74 6f 20 74 68 65 20 66 69 72  point to the fir
3cf0: 73 74 20 6c 69 6e 65 20 66 6f 6c 6c 6f 77 69 6e  st line followin
3d00: 67 20 74 68 65 20 62 6f 75 6e 64 72 79 2e 0a 2a  g the boundry..*
3d10: 2a 20 54 68 65 20 6c 65 6e 67 74 68 20 6f 66 20  * The length of 
3d20: 74 68 65 20 63 6f 6e 74 65 6e 74 20 69 73 20 73  the content is s
3d30: 74 6f 72 65 64 20 69 6e 20 2a 70 6e 43 6f 6e 74  tored in *pnCont
3d40: 65 6e 74 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 63  ent..*/.static c
3d50: 68 61 72 20 2a 67 65 74 5f 62 6f 75 6e 64 65 64  har *get_bounded
3d60: 5f 63 6f 6e 74 65 6e 74 28 0a 20 20 63 68 61 72  _content(.  char
3d70: 20 2a 2a 70 7a 2c 20 20 20 20 20 20 20 20 20 2f   **pz,         /
3d80: 2a 20 43 6f 6e 74 65 6e 74 20 74 61 6b 65 6e 20  * Content taken 
3d90: 66 72 6f 6d 20 68 65 72 65 20 2a 2f 0a 20 20 69  from here */.  i
3da0: 6e 74 20 2a 70 4c 65 6e 2c 20 20 20 20 20 20 20  nt *pLen,       
3db0: 20 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 62    /* Number of b
3dc0: 79 74 65 73 20 6f 66 20 64 61 74 61 20 69 6e 20  ytes of data in 
3dd0: 28 2a 70 7a 29 5b 5d 20 2a 2f 0a 20 20 63 68 61  (*pz)[] */.  cha
3de0: 72 20 2a 7a 42 6f 75 6e 64 72 79 2c 20 20 20 20  r *zBoundry,    
3df0: 2f 2a 20 42 6f 75 6e 64 72 79 20 74 65 78 74 20  /* Boundry text 
3e00: 6d 61 72 6b 69 6e 67 20 74 68 65 20 65 6e 64 20  marking the end 
3e10: 6f 66 20 63 6f 6e 74 65 6e 74 20 2a 2f 0a 20 20  of content */.  
3e20: 69 6e 74 20 2a 70 6e 43 6f 6e 74 65 6e 74 20 20  int *pnContent  
3e30: 20 20 20 2f 2a 20 57 72 69 74 65 20 74 68 65 20     /* Write the 
3e40: 73 69 7a 65 20 6f 66 20 74 68 65 20 63 6f 6e 74  size of the cont
3e50: 65 6e 74 20 68 65 72 65 20 2a 2f 0a 29 7b 0a 20  ent here */.){. 
3e60: 20 63 68 61 72 20 2a 7a 20 3d 20 2a 70 7a 3b 0a   char *z = *pz;.
3e70: 20 20 69 6e 74 20 6c 65 6e 20 3d 20 2a 70 4c 65    int len = *pLe
3e80: 6e 3b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 69 6e  n;.  int i;.  in
3e90: 74 20 6e 42 6f 75 6e 64 72 79 20 3d 20 73 74 72  t nBoundry = str
3ea0: 6c 65 6e 28 7a 42 6f 75 6e 64 72 79 29 3b 0a 20  len(zBoundry);. 
3eb0: 20 2a 70 6e 43 6f 6e 74 65 6e 74 20 3d 20 6c 65   *pnContent = le
3ec0: 6e 3b 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69 3c  n;.  for(i=0; i<
3ed0: 6c 65 6e 3b 20 69 2b 2b 29 7b 0a 20 20 20 20 69  len; i++){.    i
3ee0: 66 28 20 7a 5b 69 5d 3d 3d 27 5c 6e 27 20 26 26  f( z[i]=='\n' &&
3ef0: 20 73 74 72 6e 63 6d 70 28 7a 42 6f 75 6e 64 72   strncmp(zBoundr
3f00: 79 2c 20 26 7a 5b 69 2b 31 5d 2c 20 6e 42 6f 75  y, &z[i+1], nBou
3f10: 6e 64 72 79 29 3d 3d 30 20 29 7b 0a 20 20 20 20  ndry)==0 ){.    
3f20: 20 20 69 66 28 20 69 3e 30 20 26 26 20 7a 5b 69    if( i>0 && z[i
3f30: 2d 31 5d 3d 3d 27 5c 72 27 20 29 20 69 2d 2d 3b  -1]=='\r' ) i--;
3f40: 0a 20 20 20 20 20 20 7a 5b 69 5d 20 3d 20 30 3b  .      z[i] = 0;
3f50: 0a 20 20 20 20 20 20 2a 70 6e 43 6f 6e 74 65 6e  .      *pnConten
3f60: 74 20 3d 20 69 3b 0a 20 20 20 20 20 20 69 20 2b  t = i;.      i +
3f70: 3d 20 6e 42 6f 75 6e 64 72 79 3b 0a 20 20 20 20  = nBoundry;.    
3f80: 20 20 62 72 65 61 6b 3b 0a 20 20 20 20 7d 0a 20    break;.    }. 
3f90: 20 7d 0a 20 20 2a 70 7a 20 3d 20 26 7a 5b 69 5d   }.  *pz = &z[i]
3fa0: 3b 0a 20 20 67 65 74 5f 6c 69 6e 65 5f 66 72 6f  ;.  get_line_fro
3fb0: 6d 5f 73 74 72 69 6e 67 28 70 7a 2c 20 70 4c 65  m_string(pz, pLe
3fc0: 6e 29 3b 0a 20 20 72 65 74 75 72 6e 20 7a 3b 20  n);.  return z; 
3fd0: 20 20 20 20 20 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54       .}../*.** T
3fe0: 6f 6b 65 6e 69 7a 65 20 61 20 6c 69 6e 65 20 6f  okenize a line o
3ff0: 66 20 74 65 78 74 20 69 6e 74 6f 20 61 73 20 6d  f text into as m
4000: 61 6e 79 20 61 73 20 6e 41 72 67 20 74 6f 6b 65  any as nArg toke
4010: 6e 73 2e 20 20 4d 61 6b 65 0a 2a 2a 20 61 7a 41  ns.  Make.** azA
4020: 72 67 5b 5d 20 70 6f 69 6e 74 20 74 6f 20 74 68  rg[] point to th
4030: 65 20 73 74 61 72 74 20 6f 66 20 65 61 63 68 20  e start of each 
4040: 74 6f 6b 65 6e 2e 0a 2a 2a 0a 2a 2a 20 54 6f 6b  token..**.** Tok
4050: 65 6e 73 20 63 6f 6e 73 69 73 74 20 6f 66 20 73  ens consist of s
4060: 70 61 63 65 20 6f 72 20 73 65 6d 69 2d 63 6f 6c  pace or semi-col
4070: 6f 6e 20 64 65 6c 69 6d 69 74 65 64 20 77 6f 72  on delimited wor
4080: 64 73 20 6f 72 0a 2a 2a 20 73 74 72 69 6e 67 73  ds or.** strings
4090: 20 69 6e 73 69 64 65 20 64 6f 75 62 6c 65 2d 71   inside double-q
40a0: 75 6f 74 65 73 2e 20 20 45 78 61 6d 70 6c 65 3a  uotes.  Example:
40b0: 0a 2a 2a 0a 2a 2a 20 20 20 20 63 6f 6e 74 65 6e  .**.**    conten
40c0: 74 2d 64 69 73 70 6f 73 69 74 69 6f 6e 3a 20 66  t-disposition: f
40d0: 6f 72 6d 2d 64 61 74 61 3b 20 6e 61 6d 65 3d 22  orm-data; name="
40e0: 66 6e 22 3b 20 66 69 6c 65 6e 61 6d 65 3d 22 69  fn"; filename="i
40f0: 6e 64 65 78 2e 68 74 6d 6c 22 0a 2a 2a 0a 2a 2a  ndex.html".**.**
4100: 20 54 68 65 20 6c 69 6e 65 20 61 62 6f 76 65 20   The line above 
4110: 69 73 20 74 6f 6b 65 6e 69 7a 65 64 20 61 73 20  is tokenized as 
4120: 66 6f 6c 6c 6f 77 73 3a 0a 2a 2a 0a 2a 2a 20 20  follows:.**.**  
4130: 20 20 61 7a 41 72 67 5b 30 5d 20 3d 20 22 63 6f    azArg[0] = "co
4140: 6e 74 65 6e 74 2d 64 69 73 70 6f 73 69 74 69 6f  ntent-dispositio
4150: 6e 3a 22 0a 2a 2a 20 20 20 20 61 7a 41 72 67 5b  n:".**    azArg[
4160: 31 5d 20 3d 20 22 66 6f 72 6d 2d 64 61 74 61 22  1] = "form-data"
4170: 0a 2a 2a 20 20 20 20 61 7a 41 72 67 5b 32 5d 20  .**    azArg[2] 
4180: 3d 20 22 6e 61 6d 65 3d 22 0a 2a 2a 20 20 20 20  = "name=".**    
4190: 61 7a 41 72 67 5b 33 5d 20 3d 20 22 66 6e 22 0a  azArg[3] = "fn".
41a0: 2a 2a 20 20 20 20 61 7a 41 72 67 5b 34 5d 20 3d  **    azArg[4] =
41b0: 20 22 66 69 6c 65 6e 61 6d 65 3d 22 0a 2a 2a 20   "filename=".** 
41c0: 20 20 20 61 7a 41 72 67 5b 35 5d 20 3d 20 22 69     azArg[5] = "i
41d0: 6e 64 65 78 2e 68 74 6d 6c 22 0a 2a 2a 20 20 20  ndex.html".**   
41e0: 20 61 7a 41 72 67 5b 36 5d 20 3d 20 30 3b 0a 2a   azArg[6] = 0;.*
41f0: 2a 0a 2a 2a 20 27 5c 30 30 30 27 20 63 68 61 72  *.** '\000' char
4200: 61 63 74 65 72 73 20 61 72 65 20 69 6e 73 65 72  acters are inser
4210: 74 65 64 20 69 6e 20 7a 5b 5d 20 61 74 20 74 68  ted in z[] at th
4220: 65 20 65 6e 64 20 6f 66 20 65 61 63 68 20 74 6f  e end of each to
4230: 6b 65 6e 2e 0a 2a 2a 20 54 68 69 73 20 72 6f 75  ken..** This rou
4240: 74 69 6e 65 20 72 65 74 75 72 6e 73 20 74 68 65  tine returns the
4250: 20 74 6f 74 61 6c 20 6e 75 6d 62 65 72 20 6f 66   total number of
4260: 20 74 6f 6b 65 6e 73 20 6f 6e 20 74 68 65 20 6c   tokens on the l
4270: 69 6e 65 2c 20 36 0a 2a 2a 20 69 6e 20 74 68 65  ine, 6.** in the
4280: 20 65 78 61 6d 70 6c 65 20 61 62 6f 76 65 2e 0a   example above..
4290: 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 74 6f  */.static int to
42a0: 6b 65 6e 69 7a 65 5f 6c 69 6e 65 28 63 68 61 72  kenize_line(char
42b0: 20 2a 7a 2c 20 69 6e 74 20 6d 78 41 72 67 2c 20   *z, int mxArg, 
42c0: 63 68 61 72 20 2a 2a 61 7a 41 72 67 29 7b 0a 20  char **azArg){. 
42d0: 20 69 6e 74 20 69 20 3d 20 30 3b 0a 20 20 77 68   int i = 0;.  wh
42e0: 69 6c 65 28 20 2a 7a 20 29 7b 0a 20 20 20 20 77  ile( *z ){.    w
42f0: 68 69 6c 65 28 20 69 73 73 70 61 63 65 28 2a 7a  hile( isspace(*z
4300: 29 20 7c 7c 20 2a 7a 3d 3d 27 3b 27 20 29 7b 20  ) || *z==';' ){ 
4310: 7a 2b 2b 3b 20 7d 0a 20 20 20 20 69 66 28 20 2a  z++; }.    if( *
4320: 7a 3d 3d 27 22 27 20 26 26 20 7a 5b 31 5d 20 29  z=='"' && z[1] )
4330: 7b 0a 20 20 20 20 20 20 2a 7a 20 3d 20 30 3b 0a  {.      *z = 0;.
4340: 20 20 20 20 20 20 7a 2b 2b 3b 0a 20 20 20 20 20        z++;.     
4350: 20 69 66 28 20 69 3c 6d 78 41 72 67 2d 31 20 29   if( i<mxArg-1 )
4360: 7b 20 61 7a 41 72 67 5b 69 2b 2b 5d 20 3d 20 7a  { azArg[i++] = z
4370: 3b 20 7d 0a 20 20 20 20 20 20 77 68 69 6c 65 28  ; }.      while(
4380: 20 2a 7a 20 26 26 20 2a 7a 21 3d 27 22 27 20 29   *z && *z!='"' )
4390: 7b 20 7a 2b 2b 3b 20 7d 0a 20 20 20 20 20 20 69  { z++; }.      i
43a0: 66 28 20 2a 7a 3d 3d 30 20 29 20 62 72 65 61 6b  f( *z==0 ) break
43b0: 3b 0a 20 20 20 20 20 20 2a 7a 20 3d 20 30 3b 0a  ;.      *z = 0;.
43c0: 20 20 20 20 20 20 7a 2b 2b 3b 0a 20 20 20 20 7d        z++;.    }
43d0: 65 6c 73 65 7b 0a 20 20 20 20 20 20 69 66 28 20  else{.      if( 
43e0: 69 3c 6d 78 41 72 67 2d 31 20 29 7b 20 61 7a 41  i<mxArg-1 ){ azA
43f0: 72 67 5b 69 2b 2b 5d 20 3d 20 7a 3b 20 7d 0a 20  rg[i++] = z; }. 
4400: 20 20 20 20 20 77 68 69 6c 65 28 20 2a 7a 20 26       while( *z &
4410: 26 20 21 69 73 73 70 61 63 65 28 2a 7a 29 20 26  & !isspace(*z) &
4420: 26 20 2a 7a 21 3d 27 3b 27 20 26 26 20 2a 7a 21  & *z!=';' && *z!
4430: 3d 27 22 27 20 29 7b 20 7a 2b 2b 3b 20 7d 0a 20  ='"' ){ z++; }. 
4440: 20 20 20 20 20 69 66 28 20 2a 7a 20 26 26 20 2a       if( *z && *
4450: 7a 21 3d 27 22 27 20 29 7b 0a 20 20 20 20 20 20  z!='"' ){.      
4460: 20 20 2a 7a 20 3d 20 30 3b 0a 20 20 20 20 20 20    *z = 0;.      
4470: 20 20 7a 2b 2b 3b 0a 20 20 20 20 20 20 7d 0a 20    z++;.      }. 
4480: 20 20 20 7d 0a 20 20 7d 0a 20 20 61 7a 41 72 67     }.  }.  azArg
4490: 5b 69 5d 20 3d 20 30 3b 0a 20 20 72 65 74 75 72  [i] = 0;.  retur
44a0: 6e 20 69 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 63  n i;.}../*.** Sc
44b0: 61 6e 20 74 68 65 20 6d 75 6c 74 69 70 61 72 74  an the multipart
44c0: 2d 66 6f 72 6d 20 63 6f 6e 74 65 6e 74 20 61 6e  -form content an
44d0: 64 20 6d 61 6b 65 20 61 70 70 72 6f 70 72 69 61  d make appropria
44e0: 74 65 20 65 6e 74 72 69 65 73 0a 2a 2a 20 69 6e  te entries.** in
44f0: 74 6f 20 74 68 65 20 70 61 72 61 6d 65 74 65 72  to the parameter
4500: 20 74 61 62 6c 65 2e 0a 2a 2a 0a 2a 2a 20 54 68   table..**.** Th
4510: 65 20 63 6f 6e 74 65 6e 74 20 73 74 72 69 6e 67  e content string
4520: 20 22 7a 22 20 69 73 20 6d 6f 64 69 66 69 65 64   "z" is modified
4530: 20 62 79 20 74 68 69 73 20 72 6f 75 74 69 6e 65   by this routine
4540: 20 62 75 74 20 69 74 20 69 73 0a 2a 2a 20 6e 6f   but it is.** no
4550: 74 20 63 6f 70 69 65 64 2e 20 20 54 68 65 20 63  t copied.  The c
4560: 61 6c 6c 69 6e 67 20 66 75 6e 63 74 69 6f 6e 20  alling function 
4570: 6d 75 73 74 20 6e 6f 74 20 64 65 61 6c 6c 6f 63  must not dealloc
4580: 61 74 65 20 6f 72 20 6d 6f 64 69 66 79 0a 2a 2a  ate or modify.**
4590: 20 22 7a 22 20 61 66 74 65 72 20 74 68 69 73 20   "z" after this 
45a0: 72 6f 75 74 69 6e 65 20 66 69 6e 69 73 68 65 73  routine finishes
45b0: 20 6f 72 20 69 74 20 63 6f 75 6c 64 20 63 6f 72   or it could cor
45c0: 72 75 70 74 20 74 68 65 20 70 61 72 61 6d 65 74  rupt the paramet
45d0: 65 72 0a 2a 2a 20 74 61 62 6c 65 2e 0a 2a 2f 0a  er.** table..*/.
45e0: 73 74 61 74 69 63 20 76 6f 69 64 20 70 72 6f 63  static void proc
45f0: 65 73 73 5f 6d 75 6c 74 69 70 61 72 74 5f 66 6f  ess_multipart_fo
4600: 72 6d 5f 64 61 74 61 28 63 68 61 72 20 2a 7a 2c  rm_data(char *z,
4610: 20 69 6e 74 20 6c 65 6e 29 7b 0a 20 20 63 68 61   int len){.  cha
4620: 72 20 2a 7a 4c 69 6e 65 3b 0a 20 20 69 6e 74 20  r *zLine;.  int 
4630: 6e 41 72 67 2c 20 69 3b 0a 20 20 63 68 61 72 20  nArg, i;.  char 
4640: 2a 7a 42 6f 75 6e 64 72 79 3b 0a 20 20 63 68 61  *zBoundry;.  cha
4650: 72 20 2a 7a 56 61 6c 75 65 3b 0a 20 20 63 68 61  r *zValue;.  cha
4660: 72 20 2a 7a 4e 61 6d 65 20 3d 20 30 3b 0a 20 20  r *zName = 0;.  
4670: 69 6e 74 20 73 68 6f 77 42 79 74 65 73 20 3d 20  int showBytes = 
4680: 30 3b 0a 20 20 63 68 61 72 20 2a 61 7a 41 72 67  0;.  char *azArg
4690: 5b 35 30 5d 3b 0a 0a 20 20 7a 42 6f 75 6e 64 72  [50];..  zBoundr
46a0: 79 20 3d 20 67 65 74 5f 6c 69 6e 65 5f 66 72 6f  y = get_line_fro
46b0: 6d 5f 73 74 72 69 6e 67 28 26 7a 2c 20 26 6c 65  m_string(&z, &le
46c0: 6e 29 3b 0a 20 20 69 66 28 20 7a 42 6f 75 6e 64  n);.  if( zBound
46d0: 72 79 3d 3d 30 20 29 20 72 65 74 75 72 6e 3b 0a  ry==0 ) return;.
46e0: 20 20 77 68 69 6c 65 28 20 28 7a 4c 69 6e 65 20    while( (zLine 
46f0: 3d 20 67 65 74 5f 6c 69 6e 65 5f 66 72 6f 6d 5f  = get_line_from_
4700: 73 74 72 69 6e 67 28 26 7a 2c 20 26 6c 65 6e 29  string(&z, &len)
4710: 29 21 3d 30 20 29 7b 0a 20 20 20 20 69 66 28 20  )!=0 ){.    if( 
4720: 7a 4c 69 6e 65 5b 30 5d 3d 3d 30 20 29 7b 0a 20  zLine[0]==0 ){. 
4730: 20 20 20 20 20 69 6e 74 20 6e 43 6f 6e 74 65 6e       int nConten
4740: 74 20 3d 20 30 3b 0a 20 20 20 20 20 20 7a 56 61  t = 0;.      zVa
4750: 6c 75 65 20 3d 20 67 65 74 5f 62 6f 75 6e 64 65  lue = get_bounde
4760: 64 5f 63 6f 6e 74 65 6e 74 28 26 7a 2c 20 26 6c  d_content(&z, &l
4770: 65 6e 2c 20 7a 42 6f 75 6e 64 72 79 2c 20 26 6e  en, zBoundry, &n
4780: 43 6f 6e 74 65 6e 74 29 3b 0a 20 20 20 20 20 20  Content);.      
4790: 69 66 28 20 7a 4e 61 6d 65 20 26 26 20 7a 56 61  if( zName && zVa
47a0: 6c 75 65 20 26 26 20 69 73 6c 6f 77 65 72 28 7a  lue && islower(z
47b0: 4e 61 6d 65 5b 30 5d 29 20 29 7b 0a 20 20 20 20  Name[0]) ){.    
47c0: 20 20 20 20 63 67 69 5f 73 65 74 5f 70 61 72 61      cgi_set_para
47d0: 6d 65 74 65 72 5f 6e 6f 63 6f 70 79 28 7a 4e 61  meter_nocopy(zNa
47e0: 6d 65 2c 20 7a 56 61 6c 75 65 29 3b 0a 20 20 20  me, zValue);.   
47f0: 20 20 20 20 20 69 66 28 20 73 68 6f 77 42 79 74       if( showByt
4800: 65 73 20 29 7b 0a 20 20 20 20 20 20 20 20 20 20  es ){.          
4810: 63 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 74 65  cgi_set_paramete
4820: 72 5f 6e 6f 63 6f 70 79 28 6d 70 72 69 6e 74 66  r_nocopy(mprintf
4830: 28 22 25 73 3a 62 79 74 65 73 22 2c 20 7a 4e 61  ("%s:bytes", zNa
4840: 6d 65 29 2c 0a 20 20 20 20 20 20 20 20 20 20 20  me),.           
4850: 20 20 20 20 6d 70 72 69 6e 74 66 28 22 25 64 22      mprintf("%d"
4860: 2c 6e 43 6f 6e 74 65 6e 74 29 29 3b 0a 20 20 20  ,nContent));.   
4870: 20 20 20 20 20 7d 0a 20 20 20 20 20 20 7d 0a 20       }.      }. 
4880: 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 30 3b 0a       zName = 0;.
4890: 20 20 20 20 20 20 73 68 6f 77 42 79 74 65 73 20        showBytes 
48a0: 3d 20 30 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a  = 0;.    }else{.
48b0: 20 20 20 20 20 20 6e 41 72 67 20 3d 20 74 6f 6b        nArg = tok
48c0: 65 6e 69 7a 65 5f 6c 69 6e 65 28 7a 4c 69 6e 65  enize_line(zLine
48d0: 2c 20 73 69 7a 65 6f 66 28 61 7a 41 72 67 29 2f  , sizeof(azArg)/
48e0: 73 69 7a 65 6f 66 28 61 7a 41 72 67 5b 30 5d 29  sizeof(azArg[0])
48f0: 2c 20 61 7a 41 72 67 29 3b 0a 20 20 20 20 20 20  , azArg);.      
4900: 66 6f 72 28 69 3d 30 3b 20 69 3c 6e 41 72 67 3b  for(i=0; i<nArg;
4910: 20 69 2b 2b 29 7b 0a 20 20 20 20 20 20 20 20 69   i++){.        i
4920: 6e 74 20 63 20 3d 20 74 6f 6c 6f 77 65 72 28 61  nt c = tolower(a
4930: 7a 41 72 67 5b 69 5d 5b 30 5d 29 3b 0a 20 20 20  zArg[i][0]);.   
4940: 20 20 20 20 20 69 66 28 20 63 3d 3d 27 63 27 20       if( c=='c' 
4950: 26 26 20 73 74 72 69 63 6d 70 28 61 7a 41 72 67  && stricmp(azArg
4960: 5b 69 5d 2c 22 63 6f 6e 74 65 6e 74 2d 64 69 73  [i],"content-dis
4970: 70 6f 73 69 74 69 6f 6e 3a 22 29 3d 3d 30 20 29  position:")==0 )
4980: 7b 0a 20 20 20 20 20 20 20 20 20 20 69 2b 2b 3b  {.          i++;
4990: 0a 20 20 20 20 20 20 20 20 7d 65 6c 73 65 20 69  .        }else i
49a0: 66 28 20 63 3d 3d 27 6e 27 20 26 26 20 73 74 72  f( c=='n' && str
49b0: 69 63 6d 70 28 61 7a 41 72 67 5b 69 5d 2c 22 6e  icmp(azArg[i],"n
49c0: 61 6d 65 3d 22 29 3d 3d 30 20 29 7b 0a 20 20 20  ame=")==0 ){.   
49d0: 20 20 20 20 20 20 20 7a 4e 61 6d 65 20 3d 20 61         zName = a
49e0: 7a 41 72 67 5b 2b 2b 69 5d 3b 0a 20 20 20 20 20  zArg[++i];.     
49f0: 20 20 20 7d 65 6c 73 65 20 69 66 28 20 63 3d 3d     }else if( c==
4a00: 27 66 27 20 26 26 20 73 74 72 69 63 6d 70 28 61  'f' && stricmp(a
4a10: 7a 41 72 67 5b 69 5d 2c 22 66 69 6c 65 6e 61 6d  zArg[i],"filenam
4a20: 65 3d 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20  e=")==0 ){.     
4a30: 20 20 20 20 20 63 68 61 72 20 2a 7a 20 3d 20 61       char *z = a
4a40: 7a 41 72 67 5b 2b 2b 69 5d 3b 0a 20 20 20 20 20  zArg[++i];.     
4a50: 20 20 20 20 20 69 66 28 20 7a 4e 61 6d 65 20 26       if( zName &
4a60: 26 20 7a 20 26 26 20 69 73 6c 6f 77 65 72 28 7a  & z && islower(z
4a70: 4e 61 6d 65 5b 30 5d 29 20 29 7b 0a 20 20 20 20  Name[0]) ){.    
4a80: 20 20 20 20 20 20 20 20 63 67 69 5f 73 65 74 5f          cgi_set_
4a90: 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79  parameter_nocopy
4aa0: 28 6d 70 72 69 6e 74 66 28 22 25 73 3a 66 69 6c  (mprintf("%s:fil
4ab0: 65 6e 61 6d 65 22 2c 7a 4e 61 6d 65 29 2c 20 7a  ename",zName), z
4ac0: 29 3b 0a 20 20 20 20 20 20 20 20 20 20 7d 0a 20  );.          }. 
4ad0: 20 20 20 20 20 20 20 20 20 73 68 6f 77 42 79 74           showByt
4ae0: 65 73 20 3d 20 31 3b 0a 20 20 20 20 20 20 20 20  es = 1;.        
4af0: 7d 65 6c 73 65 20 69 66 28 20 63 3d 3d 27 63 27  }else if( c=='c'
4b00: 20 26 26 20 73 74 72 69 63 6d 70 28 61 7a 41 72   && stricmp(azAr
4b10: 67 5b 69 5d 2c 22 63 6f 6e 74 65 6e 74 2d 74 79  g[i],"content-ty
4b20: 70 65 3a 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20  pe:")==0 ){.    
4b30: 20 20 20 20 20 20 63 68 61 72 20 2a 7a 20 3d 20        char *z = 
4b40: 61 7a 41 72 67 5b 2b 2b 69 5d 3b 0a 20 20 20 20  azArg[++i];.    
4b50: 20 20 20 20 20 20 69 66 28 20 7a 4e 61 6d 65 20        if( zName 
4b60: 26 26 20 7a 20 26 26 20 69 73 6c 6f 77 65 72 28  && z && islower(
4b70: 7a 4e 61 6d 65 5b 30 5d 29 20 29 7b 0a 20 20 20  zName[0]) ){.   
4b80: 20 20 20 20 20 20 20 20 20 63 67 69 5f 73 65 74           cgi_set
4b90: 5f 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70  _parameter_nocop
4ba0: 79 28 6d 70 72 69 6e 74 66 28 22 25 73 3a 6d 69  y(mprintf("%s:mi
4bb0: 6d 65 74 79 70 65 22 2c 7a 4e 61 6d 65 29 2c 20  metype",zName), 
4bc0: 7a 29 3b 0a 20 20 20 20 20 20 20 20 20 20 7d 0a  z);.          }.
4bd0: 20 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20          }.      
4be0: 7d 0a 20 20 20 20 7d 0a 20 20 7d 20 20 20 20 20  }.    }.  }     
4bf0: 20 20 20 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 49 6e 69     .}../*.** Ini
4c00: 74 69 61 6c 69 7a 65 20 74 68 65 20 71 75 65 72  tialize the quer
4c10: 79 20 70 61 72 61 6d 65 74 65 72 20 64 61 74 61  y parameter data
4c20: 62 61 73 65 2e 20 20 49 6e 66 6f 72 6d 61 74 69  base.  Informati
4c30: 6f 6e 20 69 73 20 70 75 6c 6c 65 64 20 66 72 6f  on is pulled fro
4c40: 6d 0a 2a 2a 20 74 68 65 20 51 55 45 52 59 5f 53  m.** the QUERY_S
4c50: 54 52 49 4e 47 20 65 6e 76 69 72 6f 6e 6d 65 6e  TRING environmen
4c60: 74 20 76 61 72 69 61 62 6c 65 20 28 69 66 20 69  t variable (if i
4c70: 74 20 65 78 69 73 74 73 29 2c 20 66 72 6f 6d 20  t exists), from 
4c80: 73 74 61 6e 64 61 72 64 0a 2a 2a 20 69 6e 70 75  standard.** inpu
4c90: 74 20 69 66 20 74 68 65 72 65 20 69 73 20 50 4f  t if there is PO
4ca0: 53 54 20 64 61 74 61 2c 20 61 6e 64 20 66 72 6f  ST data, and fro
4cb0: 6d 20 48 54 54 50 5f 43 4f 4f 4b 49 45 2e 0a 2a  m HTTP_COOKIE..*
4cc0: 2f 0a 76 6f 69 64 20 63 67 69 5f 69 6e 69 74 28  /.void cgi_init(
4cd0: 76 6f 69 64 29 7b 0a 20 20 63 68 61 72 20 2a 7a  void){.  char *z
4ce0: 3b 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a  ;.  const char *
4cf0: 7a 54 79 70 65 3b 0a 20 20 69 6e 74 20 6c 65 6e  zType;.  int len
4d00: 3b 0a 20 20 63 67 69 5f 64 65 73 74 69 6e 61 74  ;.  cgi_destinat
4d10: 69 6f 6e 28 43 47 49 5f 42 4f 44 59 29 3b 0a 20  ion(CGI_BODY);. 
4d20: 20 7a 20 3d 20 28 63 68 61 72 2a 29 50 28 22 51   z = (char*)P("Q
4d30: 55 45 52 59 5f 53 54 52 49 4e 47 22 29 3b 0a 20  UERY_STRING");. 
4d40: 20 69 66 28 20 7a 20 29 7b 0a 20 20 20 20 7a 20   if( z ){.    z 
4d50: 3d 20 6d 70 72 69 6e 74 66 28 22 25 73 22 2c 7a  = mprintf("%s",z
4d60: 29 3b 0a 20 20 20 20 61 64 64 5f 70 61 72 61 6d  );.    add_param
4d70: 5f 6c 69 73 74 28 7a 2c 20 27 26 27 29 3b 0a 20  _list(z, '&');. 
4d80: 20 7d 0a 0a 20 20 6c 65 6e 20 3d 20 61 74 6f 69   }..  len = atoi
4d90: 28 50 44 28 22 43 4f 4e 54 45 4e 54 5f 4c 45 4e  (PD("CONTENT_LEN
4da0: 47 54 48 22 2c 20 22 30 22 29 29 3b 0a 20 20 67  GTH", "0"));.  g
4db0: 2e 7a 43 6f 6e 74 65 6e 74 54 79 70 65 20 3d 20  .zContentType = 
4dc0: 7a 54 79 70 65 20 3d 20 50 28 22 43 4f 4e 54 45  zType = P("CONTE
4dd0: 4e 54 5f 54 59 50 45 22 29 3b 0a 20 20 69 66 28  NT_TYPE");.  if(
4de0: 20 6c 65 6e 3e 30 20 26 26 20 7a 54 79 70 65 20   len>0 && zType 
4df0: 29 7b 0a 20 20 20 20 62 6c 6f 62 5f 7a 65 72 6f  ){.    blob_zero
4e00: 28 26 67 2e 63 67 69 49 6e 29 3b 0a 20 20 20 20  (&g.cgiIn);.    
4e10: 69 66 28 20 73 74 72 63 6d 70 28 7a 54 79 70 65  if( strcmp(zType
4e20: 2c 22 61 70 70 6c 69 63 61 74 69 6f 6e 2f 78 2d  ,"application/x-
4e30: 77 77 77 2d 66 6f 72 6d 2d 75 72 6c 65 6e 63 6f  www-form-urlenco
4e40: 64 65 64 22 29 3d 3d 30 20 0a 20 20 20 20 20 20  ded")==0 .      
4e50: 20 20 20 7c 7c 20 73 74 72 6e 63 6d 70 28 7a 54     || strncmp(zT
4e60: 79 70 65 2c 22 6d 75 6c 74 69 70 61 72 74 2f 66  ype,"multipart/f
4e70: 6f 72 6d 2d 64 61 74 61 22 2c 31 39 29 3d 3d 30  orm-data",19)==0
4e80: 20 29 7b 0a 20 20 20 20 20 20 7a 20 3d 20 6d 61   ){.      z = ma
4e90: 6c 6c 6f 63 28 20 6c 65 6e 2b 31 20 29 3b 0a 20  lloc( len+1 );. 
4ea0: 20 20 20 20 20 69 66 28 20 7a 3d 3d 30 20 29 20       if( z==0 ) 
4eb0: 65 78 69 74 28 31 29 3b 0a 20 20 20 20 20 20 6c  exit(1);.      l
4ec0: 65 6e 20 3d 20 66 72 65 61 64 28 7a 2c 20 31 2c  en = fread(z, 1,
4ed0: 20 6c 65 6e 2c 20 67 2e 68 74 74 70 49 6e 29 3b   len, g.httpIn);
4ee0: 0a 20 20 20 20 20 20 7a 5b 6c 65 6e 5d 20 3d 20  .      z[len] = 
4ef0: 30 3b 0a 20 20 20 20 20 20 69 66 28 20 7a 54 79  0;.      if( zTy
4f00: 70 65 5b 30 5d 3d 3d 27 61 27 20 29 7b 0a 20 20  pe[0]=='a' ){.  
4f10: 20 20 20 20 20 20 61 64 64 5f 70 61 72 61 6d 5f        add_param_
4f20: 6c 69 73 74 28 7a 2c 20 27 26 27 29 3b 0a 20 20  list(z, '&');.  
4f30: 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20      }else{.     
4f40: 20 20 20 70 72 6f 63 65 73 73 5f 6d 75 6c 74 69     process_multi
4f50: 70 61 72 74 5f 66 6f 72 6d 5f 64 61 74 61 28 7a  part_form_data(z
4f60: 2c 20 6c 65 6e 29 3b 0a 20 20 20 20 20 20 7d 0a  , len);.      }.
4f70: 20 20 20 20 7d 65 6c 73 65 20 69 66 28 20 73 74      }else if( st
4f80: 72 63 6d 70 28 7a 54 79 70 65 2c 20 22 61 70 70  rcmp(zType, "app
4f90: 6c 69 63 61 74 69 6f 6e 2f 78 2d 66 6f 73 73 69  lication/x-fossi
4fa0: 6c 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20  l")==0 ){.      
4fb0: 62 6c 6f 62 5f 72 65 61 64 5f 66 72 6f 6d 5f 63  blob_read_from_c
4fc0: 68 61 6e 6e 65 6c 28 26 67 2e 63 67 69 49 6e 2c  hannel(&g.cgiIn,
4fd0: 20 67 2e 68 74 74 70 49 6e 2c 20 6c 65 6e 29 3b   g.httpIn, len);
4fe0: 0a 20 20 20 20 20 20 62 6c 6f 62 5f 75 6e 63 6f  .      blob_unco
4ff0: 6d 70 72 65 73 73 28 26 67 2e 63 67 69 49 6e 2c  mpress(&g.cgiIn,
5000: 20 26 67 2e 63 67 69 49 6e 29 3b 0a 20 20 20 20   &g.cgiIn);.    
5010: 7d 65 6c 73 65 20 69 66 28 20 73 74 72 63 6d 70  }else if( strcmp
5020: 28 7a 54 79 70 65 2c 20 22 61 70 70 6c 69 63 61  (zType, "applica
5030: 74 69 6f 6e 2f 78 2d 66 6f 73 73 69 6c 2d 64 65  tion/x-fossil-de
5040: 62 75 67 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20  bug")==0 ){.    
5050: 20 20 62 6c 6f 62 5f 72 65 61 64 5f 66 72 6f 6d    blob_read_from
5060: 5f 63 68 61 6e 6e 65 6c 28 26 67 2e 63 67 69 49  _channel(&g.cgiI
5070: 6e 2c 20 67 2e 68 74 74 70 49 6e 2c 20 6c 65 6e  n, g.httpIn, len
5080: 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 0a 20 20  );.    }.  }..  
5090: 7a 20 3d 20 28 63 68 61 72 2a 29 50 28 22 48 54  z = (char*)P("HT
50a0: 54 50 5f 43 4f 4f 4b 49 45 22 29 3b 0a 20 20 69  TP_COOKIE");.  i
50b0: 66 28 20 7a 20 29 7b 0a 20 20 20 20 7a 20 3d 20  f( z ){.    z = 
50c0: 6d 70 72 69 6e 74 66 28 22 25 73 22 2c 7a 29 3b  mprintf("%s",z);
50d0: 0a 20 20 20 20 61 64 64 5f 70 61 72 61 6d 5f 6c  .    add_param_l
50e0: 69 73 74 28 7a 2c 20 27 3b 27 29 3b 0a 20 20 7d  ist(z, ';');.  }
50f0: 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 69 73 20 69  .}../*.** This i
5100: 73 20 74 68 65 20 63 6f 6d 70 61 72 69 73 6f 6e  s the comparison
5110: 20 66 75 6e 63 74 69 6f 6e 20 75 73 65 64 20 74   function used t
5120: 6f 20 73 6f 72 74 20 74 68 65 20 61 50 61 72 61  o sort the aPara
5130: 6d 51 50 5b 5d 20 61 72 72 61 79 20 6f 66 0a 2a  mQP[] array of.*
5140: 2a 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65  * query paramete
5150: 72 73 20 61 6e 64 20 63 6f 6f 6b 69 65 73 2e 0a  rs and cookies..
5160: 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 71 70  */.static int qp
5170: 61 72 61 6d 5f 63 6f 6d 70 61 72 65 28 63 6f 6e  aram_compare(con
5180: 73 74 20 76 6f 69 64 20 2a 61 2c 20 63 6f 6e 73  st void *a, cons
5190: 74 20 76 6f 69 64 20 2a 62 29 7b 0a 20 20 73 74  t void *b){.  st
51a0: 72 75 63 74 20 51 50 61 72 61 6d 20 2a 70 41 20  ruct QParam *pA 
51b0: 3d 20 28 73 74 72 75 63 74 20 51 50 61 72 61 6d  = (struct QParam
51c0: 2a 29 61 3b 0a 20 20 73 74 72 75 63 74 20 51 50  *)a;.  struct QP
51d0: 61 72 61 6d 20 2a 70 42 20 3d 20 28 73 74 72 75  aram *pB = (stru
51e0: 63 74 20 51 50 61 72 61 6d 2a 29 62 3b 0a 20 20  ct QParam*)b;.  
51f0: 69 6e 74 20 63 3b 0a 20 20 63 20 3d 20 73 74 72  int c;.  c = str
5200: 63 6d 70 28 70 41 2d 3e 7a 4e 61 6d 65 2c 20 70  cmp(pA->zName, p
5210: 42 2d 3e 7a 4e 61 6d 65 29 3b 0a 20 20 69 66 28  B->zName);.  if(
5220: 20 63 3d 3d 30 20 29 7b 0a 20 20 20 20 63 20 3d   c==0 ){.    c =
5230: 20 70 41 2d 3e 73 65 71 20 2d 20 70 42 2d 3e 73   pA->seq - pB->s
5240: 65 71 3b 0a 20 20 7d 0a 20 20 72 65 74 75 72 6e  eq;.  }.  return
5250: 20 63 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 52 65 74   c;.}../*.** Ret
5260: 75 72 6e 20 74 68 65 20 76 61 6c 75 65 20 6f 66  urn the value of
5270: 20 61 20 71 75 65 72 79 20 70 61 72 61 6d 65 74   a query paramet
5280: 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 77 68 6f  er or cookie who
5290: 73 65 20 6e 61 6d 65 20 69 73 20 7a 4e 61 6d 65  se name is zName
52a0: 2e 0a 2a 2a 20 49 66 20 74 68 65 72 65 20 69 73  ..** If there is
52b0: 20 6e 6f 20 71 75 65 72 79 20 70 61 72 61 6d 65   no query parame
52c0: 74 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 6e 61  ter or cookie na
52d0: 6d 65 64 20 7a 4e 61 6d 65 20 61 6e 64 20 74 68  med zName and th
52e0: 65 20 66 69 72 73 74 0a 2a 2a 20 63 68 61 72 61  e first.** chara
52f0: 63 74 65 72 20 6f 66 20 7a 4e 61 6d 65 20 69 73  cter of zName is
5300: 20 75 70 70 65 72 63 61 73 65 2c 20 74 68 65 6e   uppercase, then
5310: 20 63 68 65 63 6b 20 74 6f 20 73 65 65 20 69 66   check to see if
5320: 20 74 68 65 72 65 20 69 73 20 61 6e 0a 2a 2a 20   there is an.** 
5330: 65 6e 76 69 72 6f 6e 6d 65 6e 74 20 76 61 72 69  environment vari
5340: 61 62 6c 65 20 62 79 20 74 68 61 74 20 6e 61 6d  able by that nam
5350: 65 20 61 6e 64 20 72 65 74 75 72 6e 20 69 74 20  e and return it 
5360: 69 66 20 74 68 65 72 65 20 69 73 2e 20 20 41 73  if there is.  As
5370: 0a 2a 2a 20 61 20 6c 61 73 74 20 72 65 73 6f 72  .** a last resor
5380: 74 20 77 68 65 6e 20 6e 6f 74 68 69 6e 67 20 65  t when nothing e
5390: 6c 73 65 20 6d 61 74 63 68 65 73 2c 20 72 65 74  lse matches, ret
53a0: 75 72 6e 20 7a 44 65 66 61 75 6c 74 2e 0a 2a 2f  urn zDefault..*/
53b0: 0a 63 6f 6e 73 74 20 63 68 61 72 20 2a 63 67 69  .const char *cgi
53c0: 5f 70 61 72 61 6d 65 74 65 72 28 63 6f 6e 73 74  _parameter(const
53d0: 20 63 68 61 72 20 2a 7a 4e 61 6d 65 2c 20 63 6f   char *zName, co
53e0: 6e 73 74 20 63 68 61 72 20 2a 7a 44 65 66 61 75  nst char *zDefau
53f0: 6c 74 29 7b 0a 20 20 69 6e 74 20 6c 6f 2c 20 68  lt){.  int lo, h
5400: 69 2c 20 6d 69 64 2c 20 63 3b 0a 0a 20 20 2f 2a  i, mid, c;..  /*
5410: 20 54 68 65 20 73 6f 72 74 51 50 20 66 6c 61 67   The sortQP flag
5420: 20 69 73 20 73 65 74 20 77 68 65 6e 65 76 65 72   is set whenever
5430: 20 61 20 6e 65 77 20 71 75 65 72 79 20 70 61 72   a new query par
5440: 61 6d 65 74 65 72 20 69 73 20 69 6e 73 65 72 74  ameter is insert
5450: 65 64 2e 0a 20 20 2a 2a 20 49 74 20 69 6e 64 69  ed..  ** It indi
5460: 63 61 74 65 73 20 74 68 61 74 20 77 65 20 6e 65  cates that we ne
5470: 65 64 20 74 6f 20 72 65 73 6f 72 74 20 74 68 65  ed to resort the
5480: 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72   query parameter
5490: 73 2e 0a 20 20 2a 2f 0a 20 20 69 66 28 20 73 6f  s..  */.  if( so
54a0: 72 74 51 50 20 29 7b 0a 20 20 20 20 69 6e 74 20  rtQP ){.    int 
54b0: 69 2c 20 6a 3b 0a 20 20 20 20 71 73 6f 72 74 28  i, j;.    qsort(
54c0: 61 50 61 72 61 6d 51 50 2c 20 6e 55 73 65 64 51  aParamQP, nUsedQ
54d0: 50 2c 20 73 69 7a 65 6f 66 28 61 50 61 72 61 6d  P, sizeof(aParam
54e0: 51 50 5b 30 5d 29 2c 20 71 70 61 72 61 6d 5f 63  QP[0]), qparam_c
54f0: 6f 6d 70 61 72 65 29 3b 0a 20 20 20 20 73 6f 72  ompare);.    sor
5500: 74 51 50 20 3d 20 30 3b 0a 20 20 20 20 2f 2a 20  tQP = 0;.    /* 
5510: 41 66 74 65 72 20 73 6f 72 74 69 6e 67 2c 20 72  After sorting, r
5520: 65 6d 6f 76 65 20 64 75 70 6c 69 63 61 74 65 20  emove duplicate 
5530: 70 61 72 61 6d 65 74 65 72 73 2e 20 20 54 68 65  parameters.  The
5540: 20 73 65 63 6f 6e 64 61 72 79 20 73 6f 72 74 0a   secondary sort.
5550: 20 20 20 20 2a 2a 20 6b 65 79 20 69 73 20 61 50      ** key is aP
5560: 61 72 61 6d 51 50 5b 5d 2e 73 65 71 20 61 6e 64  aramQP[].seq and
5570: 20 77 65 20 6b 65 65 70 20 74 68 65 20 66 69 72   we keep the fir
5580: 73 74 20 65 6e 74 72 79 2e 20 20 54 68 61 74 20  st entry.  That 
5590: 6d 65 61 6e 73 0a 20 20 20 20 2a 2a 20 77 69 74  means.    ** wit
55a0: 68 20 64 75 70 6c 69 63 61 74 65 20 63 61 6c 6c  h duplicate call
55b0: 73 20 74 6f 20 63 67 69 5f 73 65 74 5f 70 61 72  s to cgi_set_par
55c0: 61 6d 65 74 65 72 28 29 20 74 68 65 20 73 65 63  ameter() the sec
55d0: 6f 6e 64 20 61 6e 64 0a 20 20 20 20 2a 2a 20 73  ond and.    ** s
55e0: 75 62 73 65 71 75 65 6e 74 20 63 61 6c 6c 73 20  ubsequent calls 
55f0: 61 72 65 20 65 66 66 65 63 74 69 76 65 6c 79 20  are effectively 
5600: 6e 6f 2d 6f 70 73 2e 20 2a 2f 0a 20 20 20 20 66  no-ops. */.    f
5610: 6f 72 28 69 3d 6a 3d 31 3b 20 69 3c 6e 55 73 65  or(i=j=1; i<nUse
5620: 64 51 50 3b 20 69 2b 2b 29 7b 0a 20 20 20 20 20  dQP; i++){.     
5630: 20 69 66 28 20 73 74 72 63 6d 70 28 61 50 61 72   if( strcmp(aPar
5640: 61 6d 51 50 5b 69 5d 2e 7a 4e 61 6d 65 2c 61 50  amQP[i].zName,aP
5650: 61 72 61 6d 51 50 5b 69 2d 31 5d 2e 7a 4e 61 6d  aramQP[i-1].zNam
5660: 65 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 20  e)==0 ){.       
5670: 20 63 6f 6e 74 69 6e 75 65 3b 0a 20 20 20 20 20   continue;.     
5680: 20 7d 0a 20 20 20 20 20 20 69 66 28 20 6a 3c 69   }.      if( j<i
5690: 20 29 7b 0a 20 20 20 20 20 20 20 20 6d 65 6d 63   ){.        memc
56a0: 70 79 28 26 61 50 61 72 61 6d 51 50 5b 6a 5d 2c  py(&aParamQP[j],
56b0: 20 26 61 50 61 72 61 6d 51 50 5b 69 5d 2c 20 73   &aParamQP[i], s
56c0: 69 7a 65 6f 66 28 61 50 61 72 61 6d 51 50 5b 6a  izeof(aParamQP[j
56d0: 5d 29 29 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20  ]));.      }.   
56e0: 20 20 20 6a 2b 2b 3b 0a 20 20 20 20 7d 0a 20 20     j++;.    }.  
56f0: 20 20 6e 55 73 65 64 51 50 20 3d 20 6a 3b 0a 20    nUsedQP = j;. 
5700: 20 7d 0a 0a 20 20 2f 2a 20 44 6f 20 61 20 62 69   }..  /* Do a bi
5710: 6e 61 72 79 20 73 65 61 72 63 68 20 66 6f 72 20  nary search for 
5720: 61 20 6d 61 74 63 68 69 6e 67 20 71 75 65 72 79  a matching query
5730: 20 70 61 72 61 6d 65 74 65 72 20 2a 2f 0a 20 20   parameter */.  
5740: 6c 6f 20 3d 20 30 3b 0a 20 20 68 69 20 3d 20 6e  lo = 0;.  hi = n
5750: 55 73 65 64 51 50 2d 31 3b 0a 20 20 77 68 69 6c  UsedQP-1;.  whil
5760: 65 28 20 6c 6f 3c 3d 68 69 20 29 7b 0a 20 20 20  e( lo<=hi ){.   
5770: 20 6d 69 64 20 3d 20 28 6c 6f 2b 68 69 29 2f 32   mid = (lo+hi)/2
5780: 3b 0a 20 20 20 20 63 20 3d 20 73 74 72 63 6d 70  ;.    c = strcmp
5790: 28 61 50 61 72 61 6d 51 50 5b 6d 69 64 5d 2e 7a  (aParamQP[mid].z
57a0: 4e 61 6d 65 2c 20 7a 4e 61 6d 65 29 3b 0a 20 20  Name, zName);.  
57b0: 20 20 69 66 28 20 63 3d 3d 30 20 29 7b 0a 20 20    if( c==0 ){.  
57c0: 20 20 20 20 43 47 49 44 45 42 55 47 28 28 22 6d      CGIDEBUG(("m
57d0: 65 6d 2d 6d 61 74 63 68 20 5b 25 73 5d 20 3d 20  em-match [%s] = 
57e0: 5b 25 73 5d 5c 6e 22 2c 20 7a 4e 61 6d 65 2c 20  [%s]\n", zName, 
57f0: 61 50 61 72 61 6d 51 50 5b 6d 69 64 5d 2e 7a 56  aParamQP[mid].zV
5800: 61 6c 75 65 29 29 3b 0a 20 20 20 20 20 20 72 65  alue));.      re
5810: 74 75 72 6e 20 61 50 61 72 61 6d 51 50 5b 6d 69  turn aParamQP[mi
5820: 64 5d 2e 7a 56 61 6c 75 65 3b 0a 20 20 20 20 7d  d].zValue;.    }
5830: 65 6c 73 65 20 69 66 28 20 63 3e 30 20 29 7b 0a  else if( c>0 ){.
5840: 20 20 20 20 20 20 68 69 20 3d 20 6d 69 64 2d 31        hi = mid-1
5850: 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20  ;.    }else{.   
5860: 20 20 20 6c 6f 20 3d 20 6d 69 64 2b 31 3b 0a 20     lo = mid+1;. 
5870: 20 20 20 7d 0a 20 20 7d 0a 0a 20 20 2f 2a 20 49     }.  }..  /* I
5880: 66 20 6e 6f 20 6d 61 74 63 68 20 69 73 20 66 6f  f no match is fo
5890: 75 6e 64 20 61 6e 64 20 74 68 65 20 6e 61 6d 65  und and the name
58a0: 20 62 65 67 69 6e 73 20 77 69 74 68 20 61 6e 20   begins with an 
58b0: 75 70 70 65 72 2d 63 61 73 65 0a 20 20 2a 2a 20  upper-case.  ** 
58c0: 6c 65 74 74 65 72 2c 20 74 68 65 6e 20 63 68 65  letter, then che
58d0: 63 6b 20 74 6f 20 73 65 65 20 69 66 20 74 68 65  ck to see if the
58e0: 72 65 20 69 73 20 61 6e 20 65 6e 76 69 72 6f 6e  re is an environ
58f0: 6d 65 6e 74 20 76 61 72 69 61 62 6c 65 0a 20 20  ment variable.  
5900: 2a 2a 20 77 69 74 68 20 74 68 65 20 67 69 76 65  ** with the give
5910: 6e 20 6e 61 6d 65 2e 0a 20 20 2a 2f 0a 20 20 69  n name..  */.  i
5920: 66 28 20 69 73 75 70 70 65 72 28 7a 4e 61 6d 65  f( isupper(zName
5930: 5b 30 5d 29 20 29 7b 0a 20 20 20 20 63 6f 6e 73  [0]) ){.    cons
5940: 74 20 63 68 61 72 20 2a 7a 56 61 6c 75 65 20 3d  t char *zValue =
5950: 20 67 65 74 65 6e 76 28 7a 4e 61 6d 65 29 3b 0a   getenv(zName);.
5960: 20 20 20 20 69 66 28 20 7a 56 61 6c 75 65 20 29      if( zValue )
5970: 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 65 74 5f  {.      cgi_set_
5980: 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79  parameter_nocopy
5990: 28 7a 4e 61 6d 65 2c 20 7a 56 61 6c 75 65 29 3b  (zName, zValue);
59a0: 0a 20 20 20 20 20 20 43 47 49 44 45 42 55 47 28  .      CGIDEBUG(
59b0: 28 22 65 6e 76 2d 6d 61 74 63 68 20 5b 25 73 5d  ("env-match [%s]
59c0: 20 3d 20 5b 25 73 5d 5c 6e 22 2c 20 7a 4e 61 6d   = [%s]\n", zNam
59d0: 65 2c 20 7a 56 61 6c 75 65 29 29 3b 0a 20 20 20  e, zValue));.   
59e0: 20 20 20 72 65 74 75 72 6e 20 7a 56 61 6c 75 65     return zValue
59f0: 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 43 47  ;.    }.  }.  CG
5a00: 49 44 45 42 55 47 28 28 22 6e 6f 2d 6d 61 74 63  IDEBUG(("no-matc
5a10: 68 20 5b 25 73 5d 5c 6e 22 2c 20 7a 4e 61 6d 65  h [%s]\n", zName
5a20: 29 29 3b 0a 20 20 72 65 74 75 72 6e 20 7a 44 65  ));.  return zDe
5a30: 66 61 75 6c 74 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  fault;.}../*.** 
5a40: 52 65 74 75 72 6e 20 74 68 65 20 6e 61 6d 65 20  Return the name 
5a50: 6f 66 20 74 68 65 20 69 2d 74 68 20 43 47 49 20  of the i-th CGI 
5a60: 70 61 72 61 6d 65 74 65 72 2e 20 20 52 65 74 75  parameter.  Retu
5a70: 72 6e 20 4e 55 4c 4c 20 69 66 20 74 68 65 72 65  rn NULL if there
5a80: 0a 2a 2a 20 61 72 65 20 66 65 77 65 72 20 74 68  .** are fewer th
5a90: 61 6e 20 69 20 72 65 67 69 73 74 65 72 65 64 20  an i registered 
5aa0: 43 47 49 20 70 61 72 6d 61 65 74 65 72 73 2e 0a  CGI parmaeters..
5ab0: 2a 2f 0a 63 6f 6e 73 74 20 63 68 61 72 20 2a 63  */.const char *c
5ac0: 67 69 5f 70 61 72 61 6d 65 74 65 72 5f 6e 61 6d  gi_parameter_nam
5ad0: 65 28 69 6e 74 20 69 29 7b 0a 20 20 69 66 28 20  e(int i){.  if( 
5ae0: 69 3e 3d 30 20 26 26 20 69 3c 6e 55 73 65 64 51  i>=0 && i<nUsedQ
5af0: 50 20 29 7b 0a 20 20 20 20 72 65 74 75 72 6e 20  P ){.    return 
5b00: 61 50 61 72 61 6d 51 50 5b 69 5d 2e 7a 4e 61 6d  aParamQP[i].zNam
5b10: 65 3b 0a 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20  e;.  }else{.    
5b20: 72 65 74 75 72 6e 20 30 3b 0a 20 20 7d 0a 7d 0a  return 0;.  }.}.
5b30: 0a 2f 2a 0a 2a 2a 20 50 72 69 6e 74 20 43 47 49  ./*.** Print CGI
5b40: 20 64 65 62 75 67 67 69 6e 67 20 6d 65 73 73 61   debugging messa
5b50: 67 65 73 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69  ges..*/.void cgi
5b60: 5f 64 65 62 75 67 28 63 6f 6e 73 74 20 63 68 61  _debug(const cha
5b70: 72 20 2a 7a 46 6f 72 6d 61 74 2c 20 2e 2e 2e 29  r *zFormat, ...)
5b80: 7b 0a 20 20 76 61 5f 6c 69 73 74 20 61 70 3b 0a  {.  va_list ap;.
5b90: 20 20 69 66 28 20 67 2e 66 44 65 62 75 67 20 29    if( g.fDebug )
5ba0: 7b 0a 20 20 20 20 76 61 5f 73 74 61 72 74 28 61  {.    va_start(a
5bb0: 70 2c 20 7a 46 6f 72 6d 61 74 29 3b 0a 20 20 20  p, zFormat);.   
5bc0: 20 76 66 70 72 69 6e 74 66 28 67 2e 66 44 65 62   vfprintf(g.fDeb
5bd0: 75 67 2c 20 7a 46 6f 72 6d 61 74 2c 20 61 70 29  ug, zFormat, ap)
5be0: 3b 0a 20 20 20 20 76 61 5f 65 6e 64 28 61 70 29  ;.    va_end(ap)
5bf0: 3b 0a 20 20 20 20 66 66 6c 75 73 68 28 67 2e 66  ;.    fflush(g.f
5c00: 44 65 62 75 67 29 3b 0a 20 20 7d 0a 7d 0a 0a 2f  Debug);.  }.}../
5c10: 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 74 72 75 65  *.** Return true
5c20: 20 69 66 20 61 6e 79 20 6f 66 20 74 68 65 20 71   if any of the q
5c30: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 73 20  uery parameters 
5c40: 69 6e 20 74 68 65 20 61 72 67 75 6d 65 6e 74 0a  in the argument.
5c50: 2a 2a 20 6c 69 73 74 20 61 72 65 20 64 65 66 69  ** list are defi
5c60: 6e 65 64 2e 0a 2a 2f 0a 69 6e 74 20 63 67 69 5f  ned..*/.int cgi_
5c70: 61 6e 79 28 63 6f 6e 73 74 20 63 68 61 72 20 2a  any(const char *
5c80: 7a 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f 6c 69  z, ...){.  va_li
5c90: 73 74 20 61 70 3b 0a 20 20 63 68 61 72 20 2a 7a  st ap;.  char *z
5ca0: 32 3b 0a 20 20 69 66 28 20 63 67 69 5f 70 61 72  2;.  if( cgi_par
5cb0: 61 6d 65 74 65 72 28 7a 2c 30 29 21 3d 30 20 29  ameter(z,0)!=0 )
5cc0: 20 72 65 74 75 72 6e 20 31 3b 0a 20 20 76 61 5f   return 1;.  va_
5cd0: 73 74 61 72 74 28 61 70 2c 20 7a 29 3b 0a 20 20  start(ap, z);.  
5ce0: 77 68 69 6c 65 28 20 28 7a 32 20 3d 20 76 61 5f  while( (z2 = va_
5cf0: 61 72 67 28 61 70 2c 20 63 68 61 72 2a 29 29 21  arg(ap, char*))!
5d00: 3d 30 20 29 7b 0a 20 20 20 20 69 66 28 20 63 67  =0 ){.    if( cg
5d10: 69 5f 70 61 72 61 6d 65 74 65 72 28 7a 32 2c 30  i_parameter(z2,0
5d20: 29 21 3d 30 20 29 20 72 65 74 75 72 6e 20 31 3b  )!=0 ) return 1;
5d30: 0a 20 20 7d 0a 20 20 76 61 5f 65 6e 64 28 61 70  .  }.  va_end(ap
5d40: 29 3b 0a 20 20 72 65 74 75 72 6e 20 30 3b 0a 7d  );.  return 0;.}
5d50: 0a 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 74  ../*.** Return t
5d60: 72 75 65 20 69 66 20 61 6c 6c 20 6f 66 20 74 68  rue if all of th
5d70: 65 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65  e query paramete
5d80: 72 73 20 69 6e 20 74 68 65 20 61 72 67 75 6d 65  rs in the argume
5d90: 6e 74 20 6c 69 73 74 0a 2a 2a 20 61 72 65 20 64  nt list.** are d
5da0: 65 66 69 6e 65 64 2e 0a 2a 2f 0a 69 6e 74 20 63  efined..*/.int c
5db0: 67 69 5f 61 6c 6c 28 63 6f 6e 73 74 20 63 68 61  gi_all(const cha
5dc0: 72 20 2a 7a 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61  r *z, ...){.  va
5dd0: 5f 6c 69 73 74 20 61 70 3b 0a 20 20 63 68 61 72  _list ap;.  char
5de0: 20 2a 7a 32 3b 0a 20 20 69 66 28 20 63 67 69 5f   *z2;.  if( cgi_
5df0: 70 61 72 61 6d 65 74 65 72 28 7a 2c 30 29 3d 3d  parameter(z,0)==
5e00: 30 20 29 20 72 65 74 75 72 6e 20 30 3b 0a 20 20  0 ) return 0;.  
5e10: 76 61 5f 73 74 61 72 74 28 61 70 2c 20 7a 29 3b  va_start(ap, z);
5e20: 0a 20 20 77 68 69 6c 65 28 20 28 7a 32 20 3d 20  .  while( (z2 = 
5e30: 76 61 5f 61 72 67 28 61 70 2c 20 63 68 61 72 2a  va_arg(ap, char*
5e40: 29 29 3d 3d 30 20 29 7b 0a 20 20 20 20 69 66 28  ))==0 ){.    if(
5e50: 20 63 67 69 5f 70 61 72 61 6d 65 74 65 72 28 7a   cgi_parameter(z
5e60: 32 2c 30 29 3d 3d 30 20 29 20 72 65 74 75 72 6e  2,0)==0 ) return
5e70: 20 30 3b 0a 20 20 7d 0a 20 20 76 61 5f 65 6e 64   0;.  }.  va_end
5e80: 28 61 70 29 3b 0a 20 20 72 65 74 75 72 6e 20 31  (ap);.  return 1
5e90: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 50 72 69 6e 74  ;.}../*.** Print
5ea0: 20 61 6c 6c 20 71 75 65 72 79 20 70 61 72 61 6d   all query param
5eb0: 65 74 65 72 73 20 6f 6e 20 73 74 61 6e 64 61 72  eters on standar
5ec0: 64 20 6f 75 74 70 75 74 2e 20 20 46 6f 72 6d 61  d output.  Forma
5ed0: 74 20 74 68 65 0a 2a 2a 20 70 61 72 61 6d 65 74  t the.** paramet
5ee0: 65 72 73 20 61 73 20 48 54 4d 4c 2e 20 20 54 68  ers as HTML.  Th
5ef0: 69 73 20 69 73 20 75 73 65 64 20 66 6f 72 20 74  is is used for t
5f00: 65 73 74 69 6e 67 20 61 6e 64 20 64 65 62 75 67  esting and debug
5f10: 67 69 6e 67 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67  ging..*/.void cg
5f20: 69 5f 70 72 69 6e 74 5f 61 6c 6c 28 76 6f 69 64  i_print_all(void
5f30: 29 7b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 63 67  ){.  int i;.  cg
5f40: 69 5f 70 61 72 61 6d 65 74 65 72 28 22 22 2c 22  i_parameter("","
5f50: 22 29 3b 20 20 2f 2a 20 46 6f 72 63 65 20 74 68  ");  /* Force th
5f60: 65 20 70 61 72 61 6d 65 74 65 72 73 20 69 6e 74  e parameters int
5f70: 6f 20 73 6f 72 74 65 64 20 6f 72 64 65 72 20 2a  o sorted order *
5f80: 2f 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69 3c 6e  /.  for(i=0; i<n
5f90: 55 73 65 64 51 50 3b 20 69 2b 2b 29 7b 0a 20 20  UsedQP; i++){.  
5fa0: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 73    cgi_printf("%s
5fb0: 20 3d 20 25 73 20 20 3c 62 72 20 2f 3e 5c 6e 22   = %s  <br />\n"
5fc0: 2c 0a 20 20 20 20 20 20 20 68 74 6d 6c 69 7a 65  ,.       htmlize
5fd0: 28 61 50 61 72 61 6d 51 50 5b 69 5d 2e 7a 4e 61  (aParamQP[i].zNa
5fe0: 6d 65 2c 20 2d 31 29 2c 20 68 74 6d 6c 69 7a 65  me, -1), htmlize
5ff0: 28 61 50 61 72 61 6d 51 50 5b 69 5d 2e 7a 56 61  (aParamQP[i].zVa
6000: 6c 75 65 2c 20 2d 31 29 29 3b 0a 20 20 7d 0a 7d  lue, -1));.  }.}
6010: 0a 0a 2f 2a 0a 2a 2a 20 57 72 69 74 65 20 48 54  ../*.** Write HT
6020: 4d 4c 20 74 65 78 74 20 66 6f 72 20 61 6e 20 6f  ML text for an o
6030: 70 74 69 6f 6e 20 6d 65 6e 75 20 74 6f 20 73 74  ption menu to st
6040: 61 6e 64 61 72 64 20 6f 75 74 70 75 74 2e 20 20  andard output.  
6050: 7a 50 61 72 61 6d 0a 2a 2a 20 69 73 20 74 68 65  zParam.** is the
6060: 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72   query parameter
6070: 20 74 68 61 74 20 74 68 65 20 6f 70 74 69 6f 6e   that the option
6080: 20 6d 65 6e 75 20 73 65 74 73 2e 20 20 7a 44 66   menu sets.  zDf
6090: 6c 74 20 69 73 20 74 68 65 0a 2a 2a 20 69 6e 69  lt is the.** ini
60a0: 74 69 61 6c 20 76 61 6c 75 65 20 6f 66 20 74 68  tial value of th
60b0: 65 20 6f 70 74 69 6f 6e 20 6d 65 6e 75 2e 20 20  e option menu.  
60c0: 41 64 64 69 74 69 6f 6e 20 61 72 67 75 6d 65 6e  Addition argumen
60d0: 74 73 20 61 72 65 20 6e 61 6d 65 2f 76 61 6c 75  ts are name/valu
60e0: 65 0a 2a 2a 20 70 61 69 72 73 20 74 68 61 74 20  e.** pairs that 
60f0: 64 65 66 69 6e 65 20 76 61 6c 75 65 73 20 6f 6e  define values on
6100: 20 74 68 65 20 6d 65 6e 75 2e 20 20 54 68 65 20   the menu.  The 
6110: 6c 69 73 74 20 69 73 20 74 65 72 6d 69 6e 61 74  list is terminat
6120: 65 64 20 77 69 74 68 0a 2a 2a 20 61 20 73 69 6e  ed with.** a sin
6130: 67 6c 65 20 4e 55 4c 4c 20 61 72 67 75 6d 65 6e  gle NULL argumen
6140: 74 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 6f  t..*/.void cgi_o
6150: 70 74 69 6f 6e 6d 65 6e 75 28 69 6e 74 20 69 6e  ptionmenu(int in
6160: 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 50  , const char *zP
6170: 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 44  , const char *zD
6180: 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f 6c 69 73  , ...){.  va_lis
6190: 74 20 61 70 3b 0a 20 20 63 68 61 72 20 2a 7a 4e  t ap;.  char *zN
61a0: 61 6d 65 2c 20 2a 7a 56 61 6c 3b 0a 20 20 69 6e  ame, *zVal;.  in
61b0: 74 20 64 66 6c 74 53 65 65 6e 20 3d 20 30 3b 0a  t dfltSeen = 0;.
61c0: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a    cgi_printf("%*
61d0: 73 3c 73 65 6c 65 63 74 20 73 69 7a 65 3d 31 20  s<select size=1 
61e0: 6e 61 6d 65 3d 5c 22 25 73 5c 22 3e 5c 6e 22 2c  name=\"%s\">\n",
61f0: 20 69 6e 2c 20 22 22 2c 20 7a 50 29 3b 0a 20 20   in, "", zP);.  
6200: 76 61 5f 73 74 61 72 74 28 61 70 2c 20 7a 44 29  va_start(ap, zD)
6210: 3b 0a 20 20 77 68 69 6c 65 28 20 28 7a 4e 61 6d  ;.  while( (zNam
6220: 65 20 3d 20 76 61 5f 61 72 67 28 61 70 2c 20 63  e = va_arg(ap, c
6230: 68 61 72 2a 29 29 21 3d 30 20 26 26 20 28 7a 56  har*))!=0 && (zV
6240: 61 6c 20 3d 20 76 61 5f 61 72 67 28 61 70 2c 20  al = va_arg(ap, 
6250: 63 68 61 72 2a 29 29 21 3d 30 20 29 7b 0a 20 20  char*))!=0 ){.  
6260: 20 20 69 66 28 20 73 74 72 63 6d 70 28 7a 56 61    if( strcmp(zVa
6270: 6c 2c 7a 44 29 3d 3d 30 20 29 7b 20 64 66 6c 74  l,zD)==0 ){ dflt
6280: 53 65 65 6e 20 3d 20 31 3b 20 62 72 65 61 6b 3b  Seen = 1; break;
6290: 20 7d 0a 20 20 7d 0a 20 20 76 61 5f 65 6e 64 28   }.  }.  va_end(
62a0: 61 70 29 3b 0a 20 20 69 66 28 20 21 64 66 6c 74  ap);.  if( !dflt
62b0: 53 65 65 6e 20 29 7b 0a 20 20 20 20 69 66 28 20  Seen ){.    if( 
62c0: 7a 44 5b 30 5d 20 29 7b 0a 20 20 20 20 20 20 63  zD[0] ){.      c
62d0: 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f  gi_printf("%*s<o
62e0: 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c 22 25 68  ption value=\"%h
62f0: 5c 22 20 73 65 6c 65 63 74 65 64 3e 25 68 3c 2f  \" selected>%h</
6300: 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20  option>\n",.    
6310: 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c 20 7a 44      in+2, "", zD
6320: 2c 20 7a 44 29 3b 0a 20 20 20 20 7d 65 6c 73 65  , zD);.    }else
6330: 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69 6e  {.      cgi_prin
6340: 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20 76  tf("%*s<option v
6350: 61 6c 75 65 3d 5c 22 5c 22 20 73 65 6c 65 63 74  alue=\"\" select
6360: 65 64 3e 26 6e 62 73 70 3b 3c 2f 6f 70 74 69 6f  ed>&nbsp;</optio
6370: 6e 3e 5c 6e 22 2c 20 69 6e 2b 32 2c 20 22 22 29  n>\n", in+2, "")
6380: 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 76 61  ;.    }.  }.  va
6390: 5f 73 74 61 72 74 28 61 70 2c 20 7a 44 29 3b 0a  _start(ap, zD);.
63a0: 20 20 77 68 69 6c 65 28 20 28 7a 4e 61 6d 65 20    while( (zName 
63b0: 3d 20 76 61 5f 61 72 67 28 61 70 2c 20 63 68 61  = va_arg(ap, cha
63c0: 72 2a 29 29 21 3d 30 20 26 26 20 28 7a 56 61 6c  r*))!=0 && (zVal
63d0: 20 3d 20 76 61 5f 61 72 67 28 61 70 2c 20 63 68   = va_arg(ap, ch
63e0: 61 72 2a 29 29 21 3d 30 20 29 7b 0a 20 20 20 20  ar*))!=0 ){.    
63f0: 69 66 28 20 7a 4e 61 6d 65 5b 30 5d 20 29 7b 0a  if( zName[0] ){.
6400: 20 20 20 20 20 20 63 67 69 5f 70 72 69 6e 74 66        cgi_printf
6410: 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 6c  ("%*s<option val
6420: 75 65 3d 5c 22 25 68 5c 22 25 73 3e 25 68 3c 2f  ue=\"%h\"%s>%h</
6430: 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20  option>\n",.    
6440: 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c 0a 20 20      in+2, "",.  
6450: 20 20 20 20 20 20 7a 56 61 6c 2c 0a 20 20 20 20        zVal,.    
6460: 20 20 20 20 73 74 72 63 6d 70 28 7a 56 61 6c 2c      strcmp(zVal,
6470: 20 7a 44 29 20 3f 20 22 22 20 3a 20 22 20 73 65   zD) ? "" : " se
6480: 6c 65 63 74 65 64 22 2c 0a 20 20 20 20 20 20 20  lected",.       
6490: 20 7a 4e 61 6d 65 0a 20 20 20 20 20 20 29 3b 0a   zName.      );.
64a0: 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20      }else{.     
64b0: 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73   cgi_printf("%*s
64c0: 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c 22  <option value=\"
64d0: 5c 22 25 73 3e 26 6e 62 73 70 3b 3c 2f 6f 70 74  \"%s>&nbsp;</opt
64e0: 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 20 20 20  ion>\n",.       
64f0: 20 69 6e 2b 32 2c 20 22 22 2c 0a 20 20 20 20 20   in+2, "",.     
6500: 20 20 20 73 74 72 63 6d 70 28 7a 56 61 6c 2c 20     strcmp(zVal, 
6510: 7a 44 29 20 3f 20 22 22 20 3a 20 22 20 73 65 6c  zD) ? "" : " sel
6520: 65 63 74 65 64 22 0a 20 20 20 20 20 20 29 3b 0a  ected".      );.
6530: 20 20 20 20 7d 0a 20 20 7d 0a 20 20 76 61 5f 65      }.  }.  va_e
6540: 6e 64 28 61 70 29 3b 0a 20 20 63 67 69 5f 70 72  nd(ap);.  cgi_pr
6550: 69 6e 74 66 28 22 25 2a 73 3c 2f 73 65 6c 65 63  intf("%*s</selec
6560: 74 3e 5c 6e 22 2c 20 69 6e 2c 20 22 22 29 3b 0a  t>\n", in, "");.
6570: 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 69 73 20 72 6f  }../*.** This ro
6580: 75 74 69 6e 65 20 77 6f 72 6b 73 20 61 20 6c 6f  utine works a lo
6590: 74 20 6c 69 6b 65 20 63 67 69 5f 6f 70 74 69 6f  t like cgi_optio
65a0: 6e 6d 65 6e 75 28 29 20 65 78 63 65 70 74 20 74  nmenu() except t
65b0: 68 61 74 20 74 68 65 20 6c 69 73 74 20 6f 66 0a  hat the list of.
65c0: 2a 2a 20 76 61 6c 75 65 73 20 69 73 20 63 6f 6e  ** values is con
65d0: 74 61 69 6e 65 64 20 69 6e 20 61 6e 20 61 72 72  tained in an arr
65e0: 61 79 2e 20 20 41 6c 73 6f 2c 20 74 68 65 20 76  ay.  Also, the v
65f0: 61 6c 75 65 73 20 61 72 65 20 6a 75 73 74 20 76  alues are just v
6600: 61 6c 75 65 73 2c 20 6e 6f 74 0a 2a 2a 20 6e 61  alues, not.** na
6610: 6d 65 2f 76 61 6c 75 65 20 70 61 69 72 73 20 61  me/value pairs a
6620: 73 20 69 6e 20 63 67 69 5f 6f 70 74 69 6f 6e 6d  s in cgi_optionm
6630: 65 6e 75 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69  enu..*/.void cgi
6640: 5f 76 5f 6f 70 74 69 6f 6e 6d 65 6e 75 28 0a 20  _v_optionmenu(. 
6650: 20 69 6e 74 20 69 6e 2c 20 20 20 20 20 20 20 20   int in,        
6660: 20 20 20 20 20 20 2f 2a 20 49 6e 64 65 6e 74 20        /* Indent 
6670: 62 79 20 74 68 69 73 20 61 6d 6f 75 6e 74 20 2a  by this amount *
6680: 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a  /.  const char *
6690: 7a 50 2c 20 20 20 20 20 20 2f 2a 20 54 68 65 20  zP,      /* The 
66a0: 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 20  query parameter 
66b0: 6e 61 6d 65 20 2a 2f 0a 20 20 63 6f 6e 73 74 20  name */.  const 
66c0: 63 68 61 72 20 2a 7a 44 2c 20 20 20 20 20 20 2f  char *zD,      /
66d0: 2a 20 44 65 66 61 75 6c 74 20 76 61 6c 75 65 20  * Default value 
66e0: 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20  */.  const char 
66f0: 2a 2a 61 7a 20 20 20 20 20 20 2f 2a 20 4e 55 4c  **az      /* NUL
6700: 4c 2d 74 65 72 6d 69 6e 61 74 65 64 20 6c 69 73  L-terminated lis
6710: 74 20 6f 66 20 61 6c 6c 6f 77 65 64 20 76 61 6c  t of allowed val
6720: 75 65 73 20 2a 2f 0a 29 7b 0a 20 20 63 6f 6e 73  ues */.){.  cons
6730: 74 20 63 68 61 72 20 2a 7a 56 61 6c 3b 0a 20 20  t char *zVal;.  
6740: 69 6e 74 20 69 3b 0a 20 20 63 67 69 5f 70 72 69  int i;.  cgi_pri
6750: 6e 74 66 28 22 25 2a 73 3c 73 65 6c 65 63 74 20  ntf("%*s<select 
6760: 73 69 7a 65 3d 31 20 6e 61 6d 65 3d 5c 22 25 73  size=1 name=\"%s
6770: 5c 22 3e 5c 6e 22 2c 20 69 6e 2c 20 22 22 2c 20  \">\n", in, "", 
6780: 7a 50 29 3b 0a 20 20 66 6f 72 28 69 3d 30 3b 20  zP);.  for(i=0; 
6790: 61 7a 5b 69 5d 3b 20 69 2b 2b 29 7b 0a 20 20 20  az[i]; i++){.   
67a0: 20 69 66 28 20 73 74 72 63 6d 70 28 61 7a 5b 69   if( strcmp(az[i
67b0: 5d 2c 7a 44 29 3d 3d 30 20 29 20 62 72 65 61 6b  ],zD)==0 ) break
67c0: 3b 0a 20 20 7d 0a 20 20 69 66 28 20 61 7a 5b 69  ;.  }.  if( az[i
67d0: 5d 3d 3d 30 20 29 7b 0a 20 20 20 20 69 66 28 20  ]==0 ){.    if( 
67e0: 7a 44 5b 30 5d 3d 3d 30 20 29 7b 0a 20 20 20 20  zD[0]==0 ){.    
67f0: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a    cgi_printf("%*
6800: 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c  s<option value=\
6810: 22 5c 22 20 73 65 6c 65 63 74 65 64 3e 26 6e 62  "\" selected>&nb
6820: 73 70 3b 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c  sp;</option>\n",
6830: 0a 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22  .       in+2, ""
6840: 29 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20  );.    }else{.  
6850: 20 20 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22      cgi_printf("
6860: 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65  %*s<option value
6870: 3d 5c 22 25 68 5c 22 20 73 65 6c 65 63 74 65 64  =\"%h\" selected
6880: 3e 25 68 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c  >%h</option>\n",
6890: 0a 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22  .       in+2, ""
68a0: 2c 20 7a 44 2c 20 7a 44 29 3b 0a 20 20 20 20 7d  , zD, zD);.    }
68b0: 0a 20 20 7d 0a 20 20 77 68 69 6c 65 28 20 28 7a  .  }.  while( (z
68c0: 56 61 6c 20 3d 20 2a 28 61 7a 2b 2b 29 29 21 3d  Val = *(az++))!=
68d0: 30 20 20 29 7b 0a 20 20 20 20 69 66 28 20 7a 56  0  ){.    if( zV
68e0: 61 6c 5b 30 5d 20 29 7b 0a 20 20 20 20 20 20 63  al[0] ){.      c
68f0: 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f  gi_printf("%*s<o
6900: 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c 22 25 68  ption value=\"%h
6910: 5c 22 25 73 3e 25 68 3c 2f 6f 70 74 69 6f 6e 3e  \"%s>%h</option>
6920: 5c 6e 22 2c 0a 20 20 20 20 20 20 20 20 69 6e 2b  \n",.        in+
6930: 32 2c 20 22 22 2c 0a 20 20 20 20 20 20 20 20 7a  2, "",.        z
6940: 56 61 6c 2c 0a 20 20 20 20 20 20 20 20 73 74 72  Val,.        str
6950: 63 6d 70 28 7a 56 61 6c 2c 20 7a 44 29 20 3f 20  cmp(zVal, zD) ? 
6960: 22 22 20 3a 20 22 20 73 65 6c 65 63 74 65 64 22  "" : " selected"
6970: 2c 0a 20 20 20 20 20 20 20 20 7a 56 61 6c 0a 20  ,.        zVal. 
6980: 20 20 20 20 20 29 3b 0a 20 20 20 20 7d 65 6c 73       );.    }els
6990: 65 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69  e{.      cgi_pri
69a0: 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20  ntf("%*s<option 
69b0: 76 61 6c 75 65 3d 5c 22 5c 22 25 73 3e 26 6e 62  value=\"\"%s>&nb
69c0: 73 70 3b 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c  sp;</option>\n",
69d0: 0a 20 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22  .        in+2, "
69e0: 22 2c 0a 20 20 20 20 20 20 20 20 73 74 72 63 6d  ",.        strcm
69f0: 70 28 7a 56 61 6c 2c 20 7a 44 29 20 3f 20 22 22  p(zVal, zD) ? ""
6a00: 20 3a 20 22 20 73 65 6c 65 63 74 65 64 22 0a 20   : " selected". 
6a10: 20 20 20 20 20 29 3b 0a 20 20 20 20 7d 0a 20 20       );.    }.  
6a20: 7d 0a 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22  }.  cgi_printf("
6a30: 25 2a 73 3c 2f 73 65 6c 65 63 74 3e 5c 6e 22 2c  %*s</select>\n",
6a40: 20 69 6e 2c 20 22 22 29 3b 0a 7d 0a 0a 2f 2a 0a   in, "");.}../*.
6a50: 2a 2a 20 54 68 69 73 20 72 6f 75 74 69 6e 65 20  ** This routine 
6a60: 77 6f 72 6b 73 20 61 20 6c 6f 74 20 6c 69 6b 65  works a lot like
6a70: 20 63 67 69 5f 76 5f 6f 70 74 69 6f 6e 6d 65 6e   cgi_v_optionmen
6a80: 75 28 29 20 65 78 63 65 70 74 20 74 68 61 74 20  u() except that 
6a90: 74 68 65 20 6c 69 73 74 0a 2a 2a 20 69 73 20 61  the list.** is a
6aa0: 20 6c 69 73 74 20 6f 66 20 70 61 69 72 73 2e 20   list of pairs. 
6ab0: 20 54 68 65 20 66 69 72 73 74 20 65 6c 65 6d 65   The first eleme
6ac0: 6e 74 20 6f 66 20 65 61 63 68 20 70 61 69 72 20  nt of each pair 
6ad0: 69 73 20 74 68 65 20 76 61 6c 75 65 20 75 73 65  is the value use
6ae0: 64 0a 2a 2a 20 69 6e 74 65 72 6e 61 6c 6c 79 20  d.** internally 
6af0: 61 6e 64 20 74 68 65 20 73 65 63 6f 6e 64 20 65  and the second e
6b00: 6c 65 6d 65 6e 74 20 69 73 20 74 68 65 20 76 61  lement is the va
6b10: 6c 75 65 20 64 69 73 70 6c 61 79 65 64 20 74 6f  lue displayed to
6b20: 20 74 68 65 20 75 73 65 72 2e 0a 2a 2f 0a 76 6f   the user..*/.vo
6b30: 69 64 20 63 67 69 5f 76 5f 6f 70 74 69 6f 6e 6d  id cgi_v_optionm
6b40: 65 6e 75 32 28 0a 20 20 69 6e 74 20 69 6e 2c 20  enu2(.  int in, 
6b50: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
6b60: 49 6e 64 65 6e 74 20 62 79 20 74 68 69 73 20 61  Indent by this a
6b70: 6d 6f 75 6e 74 20 2a 2f 0a 20 20 63 6f 6e 73 74  mount */.  const
6b80: 20 63 68 61 72 20 2a 7a 50 2c 20 20 20 20 20 20   char *zP,      
6b90: 2f 2a 20 54 68 65 20 71 75 65 72 79 20 70 61 72  /* The query par
6ba0: 61 6d 65 74 65 72 20 6e 61 6d 65 20 2a 2f 0a 20  ameter name */. 
6bb0: 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 44 2c   const char *zD,
6bc0: 20 20 20 20 20 20 2f 2a 20 44 65 66 61 75 6c 74        /* Default
6bd0: 20 76 61 6c 75 65 20 2a 2f 0a 20 20 63 6f 6e 73   value */.  cons
6be0: 74 20 63 68 61 72 20 2a 2a 61 7a 20 20 20 20 20  t char **az     
6bf0: 20 2f 2a 20 4e 55 4c 4c 2d 74 65 72 6d 69 6e 61   /* NULL-termina
6c00: 74 65 64 20 6c 69 73 74 20 6f 66 20 61 6c 6c 6f  ted list of allo
6c10: 77 65 64 20 76 61 6c 75 65 73 20 2a 2f 0a 29 7b  wed values */.){
6c20: 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a  .  const char *z
6c30: 56 61 6c 3b 0a 20 20 69 6e 74 20 69 3b 0a 20 20  Val;.  int i;.  
6c40: 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c  cgi_printf("%*s<
6c50: 73 65 6c 65 63 74 20 73 69 7a 65 3d 31 20 6e 61  select size=1 na
6c60: 6d 65 3d 5c 22 25 73 5c 22 3e 5c 6e 22 2c 20 69  me=\"%s\">\n", i
6c70: 6e 2c 20 22 22 2c 20 7a 50 29 3b 0a 20 20 66 6f  n, "", zP);.  fo
6c80: 72 28 69 3d 30 3b 20 61 7a 5b 69 5d 3b 20 69 2b  r(i=0; az[i]; i+
6c90: 3d 32 29 7b 0a 20 20 20 20 69 66 28 20 73 74 72  =2){.    if( str
6ca0: 63 6d 70 28 61 7a 5b 69 5d 2c 7a 44 29 3d 3d 30  cmp(az[i],zD)==0
6cb0: 20 29 20 62 72 65 61 6b 3b 0a 20 20 7d 0a 20 20   ) break;.  }.  
6cc0: 69 66 28 20 61 7a 5b 69 5d 3d 3d 30 20 29 7b 0a  if( az[i]==0 ){.
6cd0: 20 20 20 20 69 66 28 20 7a 44 5b 30 5d 3d 3d 30      if( zD[0]==0
6ce0: 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72   ){.      cgi_pr
6cf0: 69 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e  intf("%*s<option
6d00: 20 76 61 6c 75 65 3d 5c 22 5c 22 20 73 65 6c 65   value=\"\" sele
6d10: 63 74 65 64 3e 26 6e 62 73 70 3b 3c 2f 6f 70 74  cted>&nbsp;</opt
6d20: 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 20 20 20  ion>\n",.       
6d30: 69 6e 2b 32 2c 20 22 22 29 3b 0a 20 20 20 20 7d  in+2, "");.    }
6d40: 65 6c 73 65 7b 0a 20 20 20 20 20 20 63 67 69 5f  else{.      cgi_
6d50: 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69  printf("%*s<opti
6d60: 6f 6e 20 76 61 6c 75 65 3d 5c 22 25 68 5c 22 20  on value=\"%h\" 
6d70: 73 65 6c 65 63 74 65 64 3e 25 68 3c 2f 6f 70 74  selected>%h</opt
6d80: 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 20 20 20  ion>\n",.       
6d90: 69 6e 2b 32 2c 20 22 22 2c 20 7a 44 2c 20 7a 44  in+2, "", zD, zD
6da0: 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 77  );.    }.  }.  w
6db0: 68 69 6c 65 28 20 28 7a 56 61 6c 20 3d 20 2a 28  hile( (zVal = *(
6dc0: 61 7a 2b 2b 29 29 21 3d 30 20 20 29 7b 0a 20 20  az++))!=0  ){.  
6dd0: 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e    const char *zN
6de0: 61 6d 65 20 3d 20 2a 28 61 7a 2b 2b 29 3b 0a 20  ame = *(az++);. 
6df0: 20 20 20 69 66 28 20 7a 4e 61 6d 65 5b 30 5d 20     if( zName[0] 
6e00: 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69  ){.      cgi_pri
6e10: 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20  ntf("%*s<option 
6e20: 76 61 6c 75 65 3d 5c 22 25 68 5c 22 25 73 3e 25  value=\"%h\"%s>%
6e30: 68 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20  h</option>\n",. 
6e40: 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c         in+2, "",
6e50: 0a 20 20 20 20 20 20 20 20 7a 56 61 6c 2c 0a 20  .        zVal,. 
6e60: 20 20 20 20 20 20 20 73 74 72 63 6d 70 28 7a 56         strcmp(zV
6e70: 61 6c 2c 20 7a 44 29 20 3f 20 22 22 20 3a 20 22  al, zD) ? "" : "
6e80: 20 73 65 6c 65 63 74 65 64 22 2c 0a 20 20 20 20   selected",.    
6e90: 20 20 20 20 7a 4e 61 6d 65 0a 20 20 20 20 20 20      zName.      
6ea0: 29 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20  );.    }else{.  
6eb0: 20 20 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22      cgi_printf("
6ec0: 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65  %*s<option value
6ed0: 3d 5c 22 25 68 5c 22 25 73 3e 26 6e 62 73 70 3b  =\"%h\"%s>&nbsp;
6ee0: 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20  </option>\n",.  
6ef0: 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c 0a        in+2, "",.
6f00: 20 20 20 20 20 20 20 20 7a 56 61 6c 2c 0a 20 20          zVal,.  
6f10: 20 20 20 20 20 20 73 74 72 63 6d 70 28 7a 56 61        strcmp(zVa
6f20: 6c 2c 20 7a 44 29 20 3f 20 22 22 20 3a 20 22 20  l, zD) ? "" : " 
6f30: 73 65 6c 65 63 74 65 64 22 0a 20 20 20 20 20 20  selected".      
6f40: 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 63  );.    }.  }.  c
6f50: 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 2f  gi_printf("%*s</
6f60: 73 65 6c 65 63 74 3e 5c 6e 22 2c 20 69 6e 2c 20  select>\n", in, 
6f70: 22 22 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68  "");.}../*.** Th
6f80: 69 73 20 72 6f 75 74 69 6e 65 20 77 6f 72 6b 73  is routine works
6f90: 20 6c 69 6b 65 20 22 70 72 69 6e 74 66 22 20 65   like "printf" e
6fa0: 78 63 65 70 74 20 74 68 61 74 20 69 74 20 68 61  xcept that it ha
6fb0: 73 20 74 68 65 0a 2a 2a 20 65 78 74 72 61 20 66  s the.** extra f
6fc0: 6f 72 6d 61 74 74 69 6e 67 20 63 61 70 61 62 69  ormatting capabi
6fd0: 6c 69 74 69 65 73 20 73 75 63 68 20 61 73 20 25  lities such as %
6fe0: 68 20 61 6e 64 20 25 74 2e 0a 2a 2f 0a 76 6f 69  h and %t..*/.voi
6ff0: 64 20 63 67 69 5f 70 72 69 6e 74 66 28 63 6f 6e  d cgi_printf(con
7000: 73 74 20 63 68 61 72 20 2a 7a 46 6f 72 6d 61 74  st char *zFormat
7010: 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f 6c 69 73  , ...){.  va_lis
7020: 74 20 61 70 3b 0a 20 20 76 61 5f 73 74 61 72 74  t ap;.  va_start
7030: 28 61 70 2c 7a 46 6f 72 6d 61 74 29 3b 0a 20 20  (ap,zFormat);.  
7040: 76 78 70 72 69 6e 74 66 28 70 43 6f 6e 74 65 6e  vxprintf(pConten
7050: 74 2c 7a 46 6f 72 6d 61 74 2c 61 70 29 3b 0a 20  t,zFormat,ap);. 
7060: 20 76 61 5f 65 6e 64 28 61 70 29 3b 0a 7d 0a 0a   va_end(ap);.}..
7070: 2f 2a 0a 2a 2a 20 54 68 69 73 20 72 6f 75 74 69  /*.** This routi
7080: 6e 65 20 77 6f 72 6b 73 20 6c 69 6b 65 20 22 76  ne works like "v
7090: 70 72 69 6e 74 66 22 20 65 78 63 65 70 74 20 74  printf" except t
70a0: 68 61 74 20 69 74 20 68 61 73 20 74 68 65 0a 2a  hat it has the.*
70b0: 2a 20 65 78 74 72 61 20 66 6f 72 6d 61 74 74 69  * extra formatti
70c0: 6e 67 20 63 61 70 61 62 69 6c 69 74 69 65 73 20  ng capabilities 
70d0: 73 75 63 68 20 61 73 20 25 68 20 61 6e 64 20 25  such as %h and %
70e0: 74 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 76  t..*/.void cgi_v
70f0: 70 72 69 6e 74 66 28 63 6f 6e 73 74 20 63 68 61  printf(const cha
7100: 72 20 2a 7a 46 6f 72 6d 61 74 2c 20 76 61 5f 6c  r *zFormat, va_l
7110: 69 73 74 20 61 70 29 7b 0a 20 20 76 78 70 72 69  ist ap){.  vxpri
7120: 6e 74 66 28 70 43 6f 6e 74 65 6e 74 2c 7a 46 6f  ntf(pContent,zFo
7130: 72 6d 61 74 2c 61 70 29 3b 0a 7d 0a 0a 0a 2f 2a  rmat,ap);.}.../*
7140: 0a 2a 2a 20 53 65 6e 64 20 61 20 72 65 70 6c 79  .** Send a reply
7150: 20 69 6e 64 69 63 61 74 69 6e 67 20 74 68 61 74   indicating that
7160: 20 74 68 65 20 48 54 54 50 20 72 65 71 75 65 73   the HTTP reques
7170: 74 20 77 61 73 20 6d 61 6c 66 6f 72 6d 65 64 0a  t was malformed.
7180: 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20 6d  */.static void m
7190: 61 6c 66 6f 72 6d 65 64 5f 72 65 71 75 65 73 74  alformed_request
71a0: 28 76 6f 69 64 29 7b 0a 20 20 63 67 69 5f 73 65  (void){.  cgi_se
71b0: 74 5f 73 74 61 74 75 73 28 35 30 31 2c 20 22 4e  t_status(501, "N
71c0: 6f 74 20 49 6d 70 6c 65 6d 65 6e 74 65 64 22 29  ot Implemented")
71d0: 3b 0a 20 20 63 67 69 5f 70 72 69 6e 74 66 28 0a  ;.  cgi_printf(.
71e0: 20 20 20 20 22 3c 68 74 6d 6c 3e 3c 62 6f 64 79      "<html><body
71f0: 3e 55 6e 72 65 63 6f 67 6e 69 7a 65 64 20 48 54  >Unrecognized HT
7200: 54 50 20 52 65 71 75 65 73 74 3c 2f 62 6f 64 79  TP Request</body
7210: 3e 3c 2f 68 74 6d 6c 3e 5c 6e 22 0a 20 20 29 3b  ></html>\n".  );
7220: 0a 20 20 63 67 69 5f 72 65 70 6c 79 28 29 3b 0a  .  cgi_reply();.
7230: 20 20 65 78 69 74 28 30 29 3b 0a 7d 0a 0a 2f 2a    exit(0);.}../*
7240: 0a 2a 2a 20 50 61 6e 69 63 20 61 6e 64 20 64 69  .** Panic and di
7250: 65 20 77 68 69 6c 65 20 70 72 6f 63 65 73 73 69  e while processi
7260: 6e 67 20 61 20 77 65 62 70 61 67 65 2e 0a 2a 2f  ng a webpage..*/
7270: 0a 76 6f 69 64 20 63 67 69 5f 70 61 6e 69 63 28  .void cgi_panic(
7280: 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 46 6f 72  const char *zFor
7290: 6d 61 74 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f  mat, ...){.  va_
72a0: 6c 69 73 74 20 61 70 3b 0a 20 20 63 67 69 5f 72  list ap;.  cgi_r
72b0: 65 73 65 74 5f 63 6f 6e 74 65 6e 74 28 29 3b 0a  eset_content();.
72c0: 20 20 63 67 69 5f 73 65 74 5f 73 74 61 74 75 73    cgi_set_status
72d0: 28 35 30 30 2c 20 22 49 6e 74 65 72 6e 61 6c 20  (500, "Internal 
72e0: 53 65 72 76 65 72 20 45 72 72 6f 72 22 29 3b 0a  Server Error");.
72f0: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 0a 20 20    cgi_printf(.  
7300: 20 20 22 3c 68 74 6d 6c 3e 3c 62 6f 64 79 3e 3c    "<html><body><
7310: 68 31 3e 49 6e 74 65 72 6e 61 6c 20 53 65 72 76  h1>Internal Serv
7320: 65 72 20 45 72 72 6f 72 3c 2f 68 31 3e 5c 6e 22  er Error</h1>\n"
7330: 0a 20 20 20 20 22 3c 70 6c 61 69 6e 74 65 78 74  .    "<plaintext
7340: 3e 22 0a 20 20 29 3b 0a 20 20 76 61 5f 73 74 61  >".  );.  va_sta
7350: 72 74 28 61 70 2c 20 7a 46 6f 72 6d 61 74 29 3b  rt(ap, zFormat);
7360: 0a 20 20 76 78 70 72 69 6e 74 66 28 70 43 6f 6e  .  vxprintf(pCon
7370: 74 65 6e 74 2c 7a 46 6f 72 6d 61 74 2c 61 70 29  tent,zFormat,ap)
7380: 3b 0a 20 20 76 61 5f 65 6e 64 28 61 70 29 3b 0a  ;.  va_end(ap);.
7390: 20 20 63 67 69 5f 72 65 70 6c 79 28 29 3b 0a 20    cgi_reply();. 
73a0: 20 65 78 69 74 28 31 29 3b 0a 7d 0a 0a 2f 2a 0a   exit(1);.}../*.
73b0: 2a 2a 20 52 65 6d 6f 76 65 20 74 68 65 20 66 69  ** Remove the fi
73c0: 72 73 74 20 73 70 61 63 65 2d 64 65 6c 69 6d 69  rst space-delimi
73d0: 74 65 64 20 74 6f 6b 65 6e 20 66 72 6f 6d 20 61  ted token from a
73e0: 20 73 74 72 69 6e 67 20 61 6e 64 20 72 65 74 75   string and retu
73f0: 72 6e 0a 2a 2a 20 61 20 70 6f 69 6e 74 65 72 20  rn.** a pointer 
7400: 74 6f 20 69 74 2e 20 20 41 64 64 20 61 20 4e 55  to it.  Add a NU
7410: 4c 4c 20 74 6f 20 74 68 65 20 73 74 72 69 6e 67  LL to the string
7420: 20 74 6f 20 74 65 72 6d 69 6e 61 74 65 20 74 68   to terminate th
7430: 65 20 74 6f 6b 65 6e 2e 0a 2a 2a 20 4d 61 6b 65  e token..** Make
7440: 20 2a 7a 4c 65 66 74 4f 76 65 72 20 70 6f 69 6e   *zLeftOver poin
7450: 74 20 74 6f 20 74 68 65 20 73 74 61 72 74 20 6f  t to the start o
7460: 66 20 74 68 65 20 6e 65 78 74 20 74 6f 6b 65 6e  f the next token
7470: 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 63 68 61 72  ..*/.static char
7480: 20 2a 65 78 74 72 61 63 74 5f 74 6f 6b 65 6e 28   *extract_token(
7490: 63 68 61 72 20 2a 7a 49 6e 70 75 74 2c 20 63 68  char *zInput, ch
74a0: 61 72 20 2a 2a 7a 4c 65 66 74 4f 76 65 72 29 7b  ar **zLeftOver){
74b0: 0a 20 20 63 68 61 72 20 2a 7a 52 65 73 75 6c 74  .  char *zResult
74c0: 20 3d 20 30 3b 0a 20 20 69 66 28 20 7a 49 6e 70   = 0;.  if( zInp
74d0: 75 74 3d 3d 30 20 29 7b 0a 20 20 20 20 69 66 28  ut==0 ){.    if(
74e0: 20 7a 4c 65 66 74 4f 76 65 72 20 29 20 2a 7a 4c   zLeftOver ) *zL
74f0: 65 66 74 4f 76 65 72 20 3d 20 30 3b 0a 20 20 20  eftOver = 0;.   
7500: 20 72 65 74 75 72 6e 20 30 3b 0a 20 20 7d 0a 20   return 0;.  }. 
7510: 20 77 68 69 6c 65 28 20 69 73 73 70 61 63 65 28   while( isspace(
7520: 2a 7a 49 6e 70 75 74 29 20 29 7b 20 7a 49 6e 70  *zInput) ){ zInp
7530: 75 74 2b 2b 3b 20 7d 0a 20 20 7a 52 65 73 75 6c  ut++; }.  zResul
7540: 74 20 3d 20 7a 49 6e 70 75 74 3b 0a 20 20 77 68  t = zInput;.  wh
7550: 69 6c 65 28 20 2a 7a 49 6e 70 75 74 20 26 26 20  ile( *zInput && 
7560: 21 69 73 73 70 61 63 65 28 2a 7a 49 6e 70 75 74  !isspace(*zInput
7570: 29 20 29 7b 20 7a 49 6e 70 75 74 2b 2b 3b 20 7d  ) ){ zInput++; }
7580: 0a 20 20 69 66 28 20 2a 7a 49 6e 70 75 74 20 29  .  if( *zInput )
7590: 7b 0a 20 20 20 20 2a 7a 49 6e 70 75 74 20 3d 20  {.    *zInput = 
75a0: 30 3b 0a 20 20 20 20 7a 49 6e 70 75 74 2b 2b 3b  0;.    zInput++;
75b0: 0a 20 20 20 20 77 68 69 6c 65 28 20 69 73 73 70  .    while( issp
75c0: 61 63 65 28 2a 7a 49 6e 70 75 74 29 20 29 7b 20  ace(*zInput) ){ 
75d0: 7a 49 6e 70 75 74 2b 2b 3b 20 7d 0a 20 20 7d 0a  zInput++; }.  }.
75e0: 20 20 69 66 28 20 7a 4c 65 66 74 4f 76 65 72 20    if( zLeftOver 
75f0: 29 7b 20 2a 7a 4c 65 66 74 4f 76 65 72 20 3d 20  ){ *zLeftOver = 
7600: 7a 49 6e 70 75 74 3b 20 7d 0a 20 20 72 65 74 75  zInput; }.  retu
7610: 72 6e 20 7a 52 65 73 75 6c 74 3b 0a 7d 0a 0a 2f  rn zResult;.}../
7620: 2a 0a 2a 2a 20 54 68 69 73 20 72 6f 75 74 69 6e  *.** This routin
7630: 65 20 68 61 6e 64 6c 65 73 20 61 20 73 69 6e 67  e handles a sing
7640: 6c 65 20 48 54 54 50 20 72 65 71 75 65 73 74 20  le HTTP request 
7650: 77 68 69 63 68 20 69 73 20 63 6f 6d 69 6e 67 20  which is coming 
7660: 69 6e 20 6f 6e 0a 2a 2a 20 73 74 61 6e 64 61 72  in on.** standar
7670: 64 20 69 6e 70 75 74 20 61 6e 64 20 77 68 69 63  d input and whic
7680: 68 20 72 65 70 6c 69 65 73 20 6f 6e 20 73 74 61  h replies on sta
7690: 6e 64 61 72 64 20 6f 75 74 70 75 74 2e 0a 2a 2a  ndard output..**
76a0: 0a 2a 2a 20 54 68 65 20 48 54 54 50 20 72 65 71  .** The HTTP req
76b0: 75 65 73 74 20 69 73 20 72 65 61 64 20 66 72 6f  uest is read fro
76c0: 6d 20 73 74 61 6e 64 61 72 64 20 69 6e 70 75 74  m standard input
76d0: 20 61 6e 64 20 69 73 20 75 73 65 64 20 74 6f 20   and is used to 
76e0: 69 6e 69 74 69 61 6c 69 7a 65 0a 2a 2a 20 65 6e  initialize.** en
76f0: 76 69 72 6f 6e 6d 65 6e 74 20 76 61 72 69 61 62  vironment variab
7700: 6c 65 73 20 61 73 20 70 65 72 20 43 47 49 2e 20  les as per CGI. 
7710: 20 54 68 65 20 63 67 69 5f 69 6e 69 74 28 29 20   The cgi_init() 
7720: 72 6f 75 74 69 6e 65 20 74 6f 20 63 6f 6d 70 6c  routine to compl
7730: 65 74 65 0a 2a 2a 20 74 68 65 20 73 65 74 75 70  ete.** the setup
7740: 2e 20 20 4f 6e 63 65 20 61 6c 6c 20 74 68 65 20  .  Once all the 
7750: 73 65 74 75 70 20 69 73 20 66 69 6e 69 73 68 65  setup is finishe
7760: 64 2c 20 74 68 69 73 20 70 72 6f 63 65 64 75 72  d, this procedur
7770: 65 20 72 65 74 75 72 6e 73 0a 2a 2a 20 61 6e 64  e returns.** and
7780: 20 73 75 62 73 65 71 75 65 6e 74 20 63 6f 64 65   subsequent code
7790: 20 68 61 6e 64 6c 65 73 20 74 68 65 20 61 63 74   handles the act
77a0: 75 61 6c 20 67 65 6e 65 72 61 74 69 6f 6e 20 6f  ual generation o
77b0: 66 20 74 68 65 20 77 65 62 70 61 67 65 2e 0a 2a  f the webpage..*
77c0: 2f 0a 76 6f 69 64 20 63 67 69 5f 68 61 6e 64 6c  /.void cgi_handl
77d0: 65 5f 68 74 74 70 5f 72 65 71 75 65 73 74 28 63  e_http_request(c
77e0: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 49 70 41 64  onst char *zIpAd
77f0: 64 72 29 7b 0a 20 20 63 68 61 72 20 2a 7a 2c 20  dr){.  char *z, 
7800: 2a 7a 54 6f 6b 65 6e 3b 0a 20 20 69 6e 74 20 69  *zToken;.  int i
7810: 3b 0a 20 20 73 74 72 75 63 74 20 73 6f 63 6b 61  ;.  struct socka
7820: 64 64 72 5f 69 6e 20 72 65 6d 6f 74 65 4e 61 6d  ddr_in remoteNam
7830: 65 3b 0a 20 20 73 69 7a 65 5f 74 20 73 69 7a 65  e;.  size_t size
7840: 20 3d 20 73 69 7a 65 6f 66 28 73 74 72 75 63 74   = sizeof(struct
7850: 20 73 6f 63 6b 61 64 64 72 5f 69 6e 29 3b 0a 20   sockaddr_in);. 
7860: 20 63 68 61 72 20 7a 4c 69 6e 65 5b 32 30 30 30   char zLine[2000
7870: 5d 3b 20 20 20 20 20 2f 2a 20 41 20 73 69 6e 67  ];     /* A sing
7880: 6c 65 20 6c 69 6e 65 20 6f 66 20 69 6e 70 75 74  le line of input
7890: 2e 20 2a 2f 0a 0a 20 20 66 75 6c 6c 48 74 74 70  . */..  fullHttp
78a0: 52 65 70 6c 79 20 3d 20 31 3b 0a 20 20 69 66 28  Reply = 1;.  if(
78b0: 20 66 67 65 74 73 28 7a 4c 69 6e 65 2c 20 73 69   fgets(zLine, si
78c0: 7a 65 6f 66 28 7a 4c 69 6e 65 29 2c 67 2e 68 74  zeof(zLine),g.ht
78d0: 74 70 49 6e 29 3d 3d 30 20 29 7b 0a 20 20 20 20  tpIn)==0 ){.    
78e0: 6d 61 6c 66 6f 72 6d 65 64 5f 72 65 71 75 65 73  malformed_reques
78f0: 74 28 29 3b 0a 20 20 7d 0a 20 20 7a 54 6f 6b 65  t();.  }.  zToke
7900: 6e 20 3d 20 65 78 74 72 61 63 74 5f 74 6f 6b 65  n = extract_toke
7910: 6e 28 7a 4c 69 6e 65 2c 20 26 7a 29 3b 0a 20 20  n(zLine, &z);.  
7920: 69 66 28 20 7a 54 6f 6b 65 6e 3d 3d 30 20 29 7b  if( zToken==0 ){
7930: 0a 20 20 20 20 6d 61 6c 66 6f 72 6d 65 64 5f 72  .    malformed_r
7940: 65 71 75 65 73 74 28 29 3b 0a 20 20 7d 0a 20 20  equest();.  }.  
7950: 69 66 28 20 73 74 72 63 6d 70 28 7a 54 6f 6b 65  if( strcmp(zToke
7960: 6e 2c 22 47 45 54 22 29 21 3d 30 20 26 26 20 73  n,"GET")!=0 && s
7970: 74 72 63 6d 70 28 7a 54 6f 6b 65 6e 2c 22 50 4f  trcmp(zToken,"PO
7980: 53 54 22 29 21 3d 30 0a 20 20 20 20 20 20 26 26  ST")!=0.      &&
7990: 20 73 74 72 63 6d 70 28 7a 54 6f 6b 65 6e 2c 22   strcmp(zToken,"
79a0: 48 45 41 44 22 29 21 3d 30 20 29 7b 0a 20 20 20  HEAD")!=0 ){.   
79b0: 20 6d 61 6c 66 6f 72 6d 65 64 5f 72 65 71 75 65   malformed_reque
79c0: 73 74 28 29 3b 0a 20 20 7d 0a 20 20 63 67 69 5f  st();.  }.  cgi_
79d0: 73 65 74 65 6e 76 28 22 47 41 54 45 57 41 59 5f  setenv("GATEWAY_
79e0: 49 4e 54 45 52 46 41 43 45 22 2c 22 43 47 49 2f  INTERFACE","CGI/
79f0: 31 2e 30 22 29 3b 0a 20 20 63 67 69 5f 73 65 74  1.0");.  cgi_set
7a00: 65 6e 76 28 22 52 45 51 55 45 53 54 5f 4d 45 54  env("REQUEST_MET
7a10: 48 4f 44 22 2c 7a 54 6f 6b 65 6e 29 3b 0a 20 20  HOD",zToken);.  
7a20: 7a 54 6f 6b 65 6e 20 3d 20 65 78 74 72 61 63 74  zToken = extract
7a30: 5f 74 6f 6b 65 6e 28 7a 2c 20 26 7a 29 3b 0a 20  _token(z, &z);. 
7a40: 20 69 66 28 20 7a 54 6f 6b 65 6e 3d 3d 30 20 29   if( zToken==0 )
7a50: 7b 0a 20 20 20 20 6d 61 6c 66 6f 72 6d 65 64 5f  {.    malformed_
7a60: 72 65 71 75 65 73 74 28 29 3b 0a 20 20 7d 0a 20  request();.  }. 
7a70: 20 63 67 69 5f 73 65 74 65 6e 76 28 22 52 45 51   cgi_setenv("REQ
7a80: 55 45 53 54 5f 55 52 49 22 2c 20 7a 54 6f 6b 65  UEST_URI", zToke
7a90: 6e 29 3b 0a 20 20 66 6f 72 28 69 3d 30 3b 20 7a  n);.  for(i=0; z
7aa0: 54 6f 6b 65 6e 5b 69 5d 20 26 26 20 7a 54 6f 6b  Token[i] && zTok
7ab0: 65 6e 5b 69 5d 21 3d 27 3f 27 3b 20 69 2b 2b 29  en[i]!='?'; i++)
7ac0: 7b 7d 0a 20 20 69 66 28 20 7a 54 6f 6b 65 6e 5b  {}.  if( zToken[
7ad0: 69 5d 20 29 20 7a 54 6f 6b 65 6e 5b 69 2b 2b 5d  i] ) zToken[i++]
7ae0: 20 3d 20 30 3b 0a 20 20 63 67 69 5f 73 65 74 65   = 0;.  cgi_sete
7af0: 6e 76 28 22 50 41 54 48 5f 49 4e 46 4f 22 2c 20  nv("PATH_INFO", 
7b00: 7a 54 6f 6b 65 6e 29 3b 0a 20 20 63 67 69 5f 73  zToken);.  cgi_s
7b10: 65 74 65 6e 76 28 22 51 55 45 52 59 5f 53 54 52  etenv("QUERY_STR
7b20: 49 4e 47 22 2c 20 26 7a 54 6f 6b 65 6e 5b 69 5d  ING", &zToken[i]
7b30: 29 3b 0a 20 20 69 66 28 20 7a 49 70 41 64 64 72  );.  if( zIpAddr
7b40: 3d 3d 30 20 26 26 0a 20 20 20 20 20 20 20 20 67  ==0 &&.        g
7b50: 65 74 70 65 65 72 6e 61 6d 65 28 66 69 6c 65 6e  etpeername(filen
7b60: 6f 28 67 2e 68 74 74 70 49 6e 29 2c 20 28 73 74  o(g.httpIn), (st
7b70: 72 75 63 74 20 73 6f 63 6b 61 64 64 72 2a 29 26  ruct sockaddr*)&
7b80: 72 65 6d 6f 74 65 4e 61 6d 65 2c 20 0a 20 20 20  remoteName, .   
7b90: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
7ba0: 20 20 20 20 20 20 20 20 20 20 20 20 20 28 73 6f               (so
7bb0: 63 6b 6c 65 6e 5f 74 2a 29 26 73 69 7a 65 29 3e  cklen_t*)&size)>
7bc0: 3d 30 0a 20 20 29 7b 0a 20 20 20 20 7a 49 70 41  =0.  ){.    zIpA
7bd0: 64 64 72 20 3d 20 69 6e 65 74 5f 6e 74 6f 61 28  ddr = inet_ntoa(
7be0: 72 65 6d 6f 74 65 4e 61 6d 65 2e 73 69 6e 5f 61  remoteName.sin_a
7bf0: 64 64 72 29 3b 0a 20 20 7d 0a 20 20 69 66 28 20  ddr);.  }.  if( 
7c00: 7a 49 70 41 64 64 72 20 29 7b 20 20 20 0a 20 20  zIpAddr ){   .  
7c10: 20 20 63 67 69 5f 73 65 74 65 6e 76 28 22 52 45    cgi_setenv("RE
7c20: 4d 4f 54 45 5f 41 44 44 52 22 2c 20 7a 49 70 41  MOTE_ADDR", zIpA
7c30: 64 64 72 29 3b 0a 20 20 20 20 67 2e 7a 49 70 41  ddr);.    g.zIpA
7c40: 64 64 72 20 3d 20 6d 70 72 69 6e 74 66 28 22 25  ddr = mprintf("%
7c50: 73 22 2c 20 7a 49 70 41 64 64 72 29 3b 0a 20 20  s", zIpAddr);.  
7c60: 7d 0a 20 0a 20 20 2f 2a 20 47 65 74 20 61 6c 6c  }. .  /* Get all
7c70: 20 74 68 65 20 6f 70 74 69 6f 6e 61 6c 20 66 69   the optional fi
7c80: 65 6c 64 73 20 74 68 61 74 20 66 6f 6c 6c 6f 77  elds that follow
7c90: 20 74 68 65 20 66 69 72 73 74 20 6c 69 6e 65 2e   the first line.
7ca0: 0a 20 20 2a 2f 0a 20 20 77 68 69 6c 65 28 20 66  .  */.  while( f
7cb0: 67 65 74 73 28 7a 4c 69 6e 65 2c 73 69 7a 65 6f  gets(zLine,sizeo
7cc0: 66 28 7a 4c 69 6e 65 29 2c 67 2e 68 74 74 70 49  f(zLine),g.httpI
7cd0: 6e 29 20 29 7b 0a 20 20 20 20 63 68 61 72 20 2a  n) ){.    char *
7ce0: 7a 46 69 65 6c 64 4e 61 6d 65 3b 0a 20 20 20 20  zFieldName;.    
7cf0: 63 68 61 72 20 2a 7a 56 61 6c 3b 0a 0a 20 20 20  char *zVal;..   
7d00: 20 7a 46 69 65 6c 64 4e 61 6d 65 20 3d 20 65 78   zFieldName = ex
7d10: 74 72 61 63 74 5f 74 6f 6b 65 6e 28 7a 4c 69 6e  tract_token(zLin
7d20: 65 2c 26 7a 56 61 6c 29 3b 0a 20 20 20 20 69 66  e,&zVal);.    if
7d30: 28 20 7a 46 69 65 6c 64 4e 61 6d 65 3d 3d 30 20  ( zFieldName==0 
7d40: 7c 7c 20 2a 7a 46 69 65 6c 64 4e 61 6d 65 3d 3d  || *zFieldName==
7d50: 30 20 29 20 62 72 65 61 6b 3b 0a 20 20 20 20 77  0 ) break;.    w
7d60: 68 69 6c 65 28 20 69 73 73 70 61 63 65 28 2a 7a  hile( isspace(*z
7d70: 56 61 6c 29 20 29 7b 20 7a 56 61 6c 2b 2b 3b 20  Val) ){ zVal++; 
7d80: 7d 0a 20 20 20 20 69 20 3d 20 73 74 72 6c 65 6e  }.    i = strlen
7d90: 28 7a 56 61 6c 29 3b 0a 20 20 20 20 77 68 69 6c  (zVal);.    whil
7da0: 65 28 20 69 3e 30 20 26 26 20 69 73 73 70 61 63  e( i>0 && isspac
7db0: 65 28 7a 56 61 6c 5b 69 2d 31 5d 29 20 29 7b 20  e(zVal[i-1]) ){ 
7dc0: 69 2d 2d 3b 20 7d 0a 20 20 20 20 7a 56 61 6c 5b  i--; }.    zVal[
7dd0: 69 5d 20 3d 20 30 3b 0a 20 20 20 20 66 6f 72 28  i] = 0;.    for(
7de0: 69 3d 30 3b 20 7a 46 69 65 6c 64 4e 61 6d 65 5b  i=0; zFieldName[
7df0: 69 5d 3b 20 69 2b 2b 29 7b 20 7a 46 69 65 6c 64  i]; i++){ zField
7e00: 4e 61 6d 65 5b 69 5d 20 3d 20 74 6f 6c 6f 77 65  Name[i] = tolowe
7e10: 72 28 7a 46 69 65 6c 64 4e 61 6d 65 5b 69 5d 29  r(zFieldName[i])
7e20: 3b 20 7d 0a 20 20 20 20 69 66 28 20 73 74 72 63  ; }.    if( strc
7e30: 6d 70 28 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 75  mp(zFieldName,"u
7e40: 73 65 72 2d 61 67 65 6e 74 3a 22 29 3d 3d 30 20  ser-agent:")==0 
7e50: 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 65 74  ){.      cgi_set
7e60: 65 6e 76 28 22 48 54 54 50 5f 55 53 45 52 5f 41  env("HTTP_USER_A
7e70: 47 45 4e 54 22 2c 20 7a 56 61 6c 29 3b 0a 20 20  GENT", zVal);.  
7e80: 20 20 7d 65 6c 73 65 20 69 66 28 20 73 74 72 63    }else if( strc
7e90: 6d 70 28 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 63  mp(zFieldName,"c
7ea0: 6f 6e 74 65 6e 74 2d 6c 65 6e 67 74 68 3a 22 29  ontent-length:")
7eb0: 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 63 67 69  ==0 ){.      cgi
7ec0: 5f 73 65 74 65 6e 76 28 22 43 4f 4e 54 45 4e 54  _setenv("CONTENT
7ed0: 5f 4c 45 4e 47 54 48 22 2c 20 7a 56 61 6c 29 3b  _LENGTH", zVal);
7ee0: 0a 20 20 20 20 7d 65 6c 73 65 20 69 66 28 20 73  .    }else if( s
7ef0: 74 72 63 6d 70 28 7a 46 69 65 6c 64 4e 61 6d 65  trcmp(zFieldName
7f00: 2c 22 72 65 66 65 72 65 72 3a 22 29 3d 3d 30 20  ,"referer:")==0 
7f10: 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 65 74  ){.      cgi_set
7f20: 65 6e 76 28 22 48 54 54 50 5f 52 45 46 45 52 45  env("HTTP_REFERE
7f30: 52 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d  R", zVal);.    }
7f40: 65 6c 73 65 20 69 66 28 20 73 74 72 63 6d 70 28  else if( strcmp(
7f50: 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 68 6f 73 74  zFieldName,"host
7f60: 3a 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20  :")==0 ){.      
7f70: 63 67 69 5f 73 65 74 65 6e 76 28 22 48 54 54 50  cgi_setenv("HTTP
7f80: 5f 48 4f 53 54 22 2c 20 7a 56 61 6c 29 3b 0a 20  _HOST", zVal);. 
7f90: 20 20 20 7d 65 6c 73 65 20 69 66 28 20 73 74 72     }else if( str
7fa0: 63 6d 70 28 7a 46 69 65 6c 64 4e 61 6d 65 2c 22  cmp(zFieldName,"
7fb0: 63 6f 6e 74 65 6e 74 2d 74 79 70 65 3a 22 29 3d  content-type:")=
7fc0: 3d 30 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f  =0 ){.      cgi_
7fd0: 73 65 74 65 6e 76 28 22 43 4f 4e 54 45 4e 54 5f  setenv("CONTENT_
7fe0: 54 59 50 45 22 2c 20 7a 56 61 6c 29 3b 0a 20 20  TYPE", zVal);.  
7ff0: 20 20 7d 65 6c 73 65 20 69 66 28 20 73 74 72 63    }else if( strc
8000: 6d 70 28 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 63  mp(zFieldName,"c
8010: 6f 6f 6b 69 65 3a 22 29 3d 3d 30 20 29 7b 0a 20  ookie:")==0 ){. 
8020: 20 20 20 20 20 63 67 69 5f 73 65 74 65 6e 76 28       cgi_setenv(
8030: 22 48 54 54 50 5f 43 4f 4f 4b 49 45 22 2c 20 7a  "HTTP_COOKIE", z
8040: 56 61 6c 29 3b 0a 20 20 20 20 7d 65 6c 73 65 20  Val);.    }else 
8050: 69 66 28 20 73 74 72 63 6d 70 28 7a 46 69 65 6c  if( strcmp(zFiel
8060: 64 4e 61 6d 65 2c 22 69 66 2d 6e 6f 6e 65 2d 6d  dName,"if-none-m
8070: 61 74 63 68 3a 22 29 3d 3d 30 20 29 7b 0a 20 20  atch:")==0 ){.  
8080: 20 20 20 20 63 67 69 5f 73 65 74 65 6e 76 28 22      cgi_setenv("
8090: 48 54 54 50 5f 49 46 5f 4e 4f 4e 45 5f 4d 41 54  HTTP_IF_NONE_MAT
80a0: 43 48 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20  CH", zVal);.    
80b0: 7d 65 6c 73 65 20 69 66 28 20 73 74 72 63 6d 70  }else if( strcmp
80c0: 28 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 69 66 2d  (zFieldName,"if-
80d0: 6d 6f 64 69 66 69 65 64 2d 73 69 6e 63 65 3a 22  modified-since:"
80e0: 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 63 67  )==0 ){.      cg
80f0: 69 5f 73 65 74 65 6e 76 28 22 48 54 54 50 5f 49  i_setenv("HTTP_I
8100: 46 5f 4d 4f 44 49 46 49 45 44 5f 53 49 4e 43 45  F_MODIFIED_SINCE
8110: 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d 0a  ", zVal);.    }.
8120: 20 20 7d 0a 0a 20 20 63 67 69 5f 69 6e 69 74 28    }..  cgi_init(
8130: 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 4d 61 78 69  );.}../*.** Maxi
8140: 6d 75 6d 20 6e 75 6d 62 65 72 20 6f 66 20 63 68  mum number of ch
8150: 69 6c 64 20 70 72 6f 63 65 73 73 65 73 20 74 68  ild processes th
8160: 61 74 20 77 65 20 63 61 6e 20 68 61 76 65 20 72  at we can have r
8170: 75 6e 6e 69 6e 67 0a 2a 2a 20 61 74 20 6f 6e 65  unning.** at one
8180: 20 74 69 6d 65 20 62 65 66 6f 72 65 20 77 65 20   time before we 
8190: 73 74 61 72 74 20 73 6c 6f 77 69 6e 67 20 74 68  start slowing th
81a0: 69 6e 67 73 20 64 6f 77 6e 2e 0a 2a 2f 0a 23 64  ings down..*/.#d
81b0: 65 66 69 6e 65 20 4d 41 58 5f 50 41 52 41 4c 4c  efine MAX_PARALL
81c0: 45 4c 20 32 0a 0a 2f 2a 0a 2a 2a 20 49 6d 70 6c  EL 2../*.** Impl
81d0: 65 6d 65 6e 74 20 61 6e 20 48 54 54 50 20 73 65  ement an HTTP se
81e0: 72 76 65 72 20 64 61 65 6d 6f 6e 20 6c 69 73 74  rver daemon list
81f0: 65 6e 69 6e 67 20 6f 6e 20 70 6f 72 74 20 69 50  ening on port iP
8200: 6f 72 74 2e 0a 2a 2a 0a 2a 2a 20 41 73 20 6e 65  ort..**.** As ne
8210: 77 20 63 6f 6e 6e 65 63 74 69 6f 6e 73 20 61 72  w connections ar
8220: 72 69 76 65 2c 20 66 6f 72 6b 20 61 20 63 68 69  rive, fork a chi
8230: 6c 64 20 61 6e 64 20 6c 65 74 20 63 68 69 6c 64  ld and let child
8240: 20 72 65 74 75 72 6e 0a 2a 2a 20 6f 75 74 20 6f   return.** out o
8250: 66 20 74 68 69 73 20 70 72 6f 63 65 64 75 72 65  f this procedure
8260: 20 63 61 6c 6c 2e 20 20 54 68 65 20 63 68 69 6c   call.  The chil
8270: 64 20 77 69 6c 6c 20 68 61 6e 64 6c 65 20 74 68  d will handle th
8280: 65 20 72 65 71 75 65 73 74 2e 0a 2a 2a 20 54 68  e request..** Th
8290: 65 20 70 61 72 65 6e 74 20 6e 65 76 65 72 20 72  e parent never r
82a0: 65 74 75 72 6e 73 20 66 72 6f 6d 20 74 68 69 73  eturns from this
82b0: 20 70 72 6f 63 65 64 75 72 65 2e 0a 2a 2a 0a 2a   procedure..**.*
82c0: 2a 20 52 65 74 75 72 6e 20 30 20 74 6f 20 65 61  * Return 0 to ea
82d0: 63 68 20 63 68 69 6c 64 20 61 73 20 69 74 20 72  ch child as it r
82e0: 75 6e 73 2e 20 20 49 66 20 75 6e 61 62 6c 65 20  uns.  If unable 
82f0: 74 6f 20 65 73 74 61 62 6c 69 73 68 20 61 0a 2a  to establish a.*
8300: 2a 20 6c 69 73 74 65 6e 69 6e 67 20 73 6f 63 6b  * listening sock
8310: 65 74 2c 20 72 65 74 75 72 6e 20 6e 6f 6e 2d 7a  et, return non-z
8320: 65 72 6f 2e 0a 2a 2f 0a 69 6e 74 20 63 67 69 5f  ero..*/.int cgi_
8330: 68 74 74 70 5f 73 65 72 76 65 72 28 69 6e 74 20  http_server(int 
8340: 69 50 6f 72 74 29 7b 0a 23 69 66 64 65 66 20 5f  iPort){.#ifdef _
8350: 5f 4d 49 4e 47 57 33 32 5f 5f 0a 20 20 66 70 72  _MINGW32__.  fpr
8360: 69 6e 74 66 28 73 74 64 65 72 72 2c 22 73 65 72  intf(stderr,"ser
8370: 76 65 72 20 6e 6f 74 20 79 65 74 20 61 76 61 69  ver not yet avai
8380: 6c 61 62 6c 65 20 69 6e 20 77 69 6e 64 6f 77 73  lable in windows
8390: 20 76 65 72 73 69 6f 6e 20 6f 66 20 66 6f 73 73   version of foss
83a0: 69 6c 5c 6e 22 29 3b 0a 20 20 65 78 69 74 28 31  il\n");.  exit(1
83b0: 29 3b 0a 23 65 6c 73 65 0a 20 20 69 6e 74 20 6c  );.#else.  int l
83c0: 69 73 74 65 6e 65 72 3b 20 20 20 20 20 20 20 20  istener;        
83d0: 20 20 20 20 20 20 20 20 2f 2a 20 54 68 65 20 73          /* The s
83e0: 65 72 76 65 72 20 73 6f 63 6b 65 74 20 2a 2f 0a  erver socket */.
83f0: 20 20 69 6e 74 20 63 6f 6e 6e 65 63 74 69 6f 6e    int connection
8400: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f  ;              /
8410: 2a 20 41 20 73 6f 63 6b 65 74 20 66 6f 72 20 65  * A socket for e
8420: 61 63 68 20 69 6e 64 69 76 69 64 75 61 6c 20 63  ach individual c
8430: 6f 6e 6e 65 63 74 69 6f 6e 20 2a 2f 0a 20 20 66  onnection */.  f
8440: 64 5f 73 65 74 20 72 65 61 64 66 64 73 3b 20 20  d_set readfds;  
8450: 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 53              /* S
8460: 65 74 20 6f 66 20 66 69 6c 65 20 64 65 73 63 72  et of file descr
8470: 69 70 74 6f 72 73 20 66 6f 72 20 73 65 6c 65 63  iptors for selec
8480: 74 28 29 20 2a 2f 0a 20 20 73 69 7a 65 5f 74 20  t() */.  size_t 
8490: 6c 65 6e 61 64 64 72 3b 20 20 20 20 20 20 20 20  lenaddr;        
84a0: 20 20 20 20 20 20 2f 2a 20 4c 65 6e 67 74 68 20        /* Length 
84b0: 6f 66 20 74 68 65 20 69 6e 61 64 64 72 20 73 74  of the inaddr st
84c0: 72 75 63 74 75 72 65 20 2a 2f 0a 20 20 69 6e 74  ructure */.  int
84d0: 20 63 68 69 6c 64 3b 20 20 20 20 20 20 20 20 20   child;         
84e0: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 50 49 44            /* PID
84f0: 20 6f 66 20 74 68 65 20 63 68 69 6c 64 20 70 72   of the child pr
8500: 6f 63 65 73 73 20 2a 2f 0a 20 20 69 6e 74 20 6e  ocess */.  int n
8510: 63 68 69 6c 64 72 65 6e 20 3d 20 30 3b 20 20 20  children = 0;   
8520: 20 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62 65          /* Numbe
8530: 72 20 6f 66 20 63 68 69 6c 64 20 70 72 6f 63 65  r of child proce
8540: 73 73 65 73 20 2a 2f 0a 20 20 73 74 72 75 63 74  sses */.  struct
8550: 20 74 69 6d 65 76 61 6c 20 64 65 6c 61 79 3b 20   timeval delay; 
8560: 20 20 20 20 20 20 20 2f 2a 20 48 6f 77 20 6c 6f         /* How lo
8570: 6e 67 20 74 6f 20 77 61 69 74 20 69 6e 73 69 64  ng to wait insid
8580: 65 20 73 65 6c 65 63 74 28 29 20 2a 2f 0a 20 20  e select() */.  
8590: 73 74 72 75 63 74 20 73 6f 63 6b 61 64 64 72 5f  struct sockaddr_
85a0: 69 6e 20 69 6e 61 64 64 72 3b 20 20 20 2f 2a 20  in inaddr;   /* 
85b0: 54 68 65 20 73 6f 63 6b 65 74 20 61 64 64 72 65  The socket addre
85c0: 73 73 20 2a 2f 0a 20 20 69 6e 74 20 6f 70 74 20  ss */.  int opt 
85d0: 3d 20 31 3b 20 20 20 20 20 20 20 20 20 20 20 20  = 1;            
85e0: 20 20 20 20 20 2f 2a 20 73 65 74 73 6f 63 6b 6f       /* setsocko
85f0: 70 74 20 66 6c 61 67 20 2a 2f 0a 0a 20 20 6d 65  pt flag */..  me
8600: 6d 73 65 74 28 26 69 6e 61 64 64 72 2c 20 30 2c  mset(&inaddr, 0,
8610: 20 73 69 7a 65 6f 66 28 69 6e 61 64 64 72 29 29   sizeof(inaddr))
8620: 3b 0a 20 20 69 6e 61 64 64 72 2e 73 69 6e 5f 66  ;.  inaddr.sin_f
8630: 61 6d 69 6c 79 20 3d 20 41 46 5f 49 4e 45 54 3b  amily = AF_INET;
8640: 0a 20 20 69 6e 61 64 64 72 2e 73 69 6e 5f 61 64  .  inaddr.sin_ad
8650: 64 72 2e 73 5f 61 64 64 72 20 3d 20 49 4e 41 44  dr.s_addr = INAD
8660: 44 52 5f 41 4e 59 3b 0a 20 20 69 6e 61 64 64 72  DR_ANY;.  inaddr
8670: 2e 73 69 6e 5f 70 6f 72 74 20 3d 20 68 74 6f 6e  .sin_port = hton
8680: 73 28 69 50 6f 72 74 29 3b 0a 20 20 6c 69 73 74  s(iPort);.  list
8690: 65 6e 65 72 20 3d 20 73 6f 63 6b 65 74 28 41 46  ener = socket(AF
86a0: 5f 49 4e 45 54 2c 20 53 4f 43 4b 5f 53 54 52 45  _INET, SOCK_STRE
86b0: 41 4d 2c 20 30 29 3b 0a 20 20 69 66 28 20 6c 69  AM, 0);.  if( li
86c0: 73 74 65 6e 65 72 3c 30 20 29 7b 0a 20 20 20 20  stener<0 ){.    
86d0: 72 65 74 75 72 6e 20 31 3b 0a 20 20 7d 0a 0a 20  return 1;.  }.. 
86e0: 20 2f 2a 20 69 66 20 77 65 20 63 61 6e 27 74 20   /* if we can't 
86f0: 74 65 72 6d 69 6e 61 74 65 20 6e 69 63 65 6c 79  terminate nicely
8700: 2c 20 61 74 20 6c 65 61 73 74 20 61 6c 6c 6f 77  , at least allow
8710: 20 74 68 65 20 73 6f 63 6b 65 74 20 74 6f 20 62   the socket to b
8720: 65 20 72 65 75 73 65 64 20 2a 2f 0a 20 20 73 65  e reused */.  se
8730: 74 73 6f 63 6b 6f 70 74 28 6c 69 73 74 65 6e 65  tsockopt(listene
8740: 72 2c 53 4f 4c 5f 53 4f 43 4b 45 54 2c 53 4f 5f  r,SOL_SOCKET,SO_
8750: 52 45 55 53 45 41 44 44 52 2c 26 6f 70 74 2c 73  REUSEADDR,&opt,s
8760: 69 7a 65 6f 66 28 6f 70 74 29 29 3b 0a 0a 20 20  izeof(opt));..  
8770: 69 66 28 20 62 69 6e 64 28 6c 69 73 74 65 6e 65  if( bind(listene
8780: 72 2c 20 28 73 74 72 75 63 74 20 73 6f 63 6b 61  r, (struct socka
8790: 64 64 72 2a 29 26 69 6e 61 64 64 72 2c 20 73 69  ddr*)&inaddr, si
87a0: 7a 65 6f 66 28 69 6e 61 64 64 72 29 29 3c 30 20  zeof(inaddr))<0 
87b0: 29 7b 0a 20 20 20 20 63 6c 6f 73 65 28 6c 69 73  ){.    close(lis
87c0: 74 65 6e 65 72 29 3b 0a 20 20 20 20 72 65 74 75  tener);.    retu
87d0: 72 6e 20 31 3b 0a 20 20 7d 0a 20 20 6c 69 73 74  rn 1;.  }.  list
87e0: 65 6e 28 6c 69 73 74 65 6e 65 72 2c 31 30 29 3b  en(listener,10);
87f0: 0a 20 20 77 68 69 6c 65 28 20 31 20 29 7b 0a 20  .  while( 1 ){. 
8800: 20 20 20 69 66 28 20 6e 63 68 69 6c 64 72 65 6e     if( nchildren
8810: 3e 4d 41 58 5f 50 41 52 41 4c 4c 45 4c 20 29 7b  >MAX_PARALLEL ){
8820: 0a 20 20 20 20 20 20 2f 2a 20 53 6c 6f 77 20 64  .      /* Slow d
8830: 6f 77 6e 20 69 66 20 63 6f 6e 6e 65 63 74 69 6f  own if connectio
8840: 6e 73 20 61 72 65 20 61 72 72 69 76 69 6e 67 20  ns are arriving 
8850: 74 6f 6f 20 66 61 73 74 20 2a 2f 0a 20 20 20 20  too fast */.    
8860: 20 20 73 6c 65 65 70 28 20 6e 63 68 69 6c 64 72    sleep( nchildr
8870: 65 6e 2d 4d 41 58 5f 50 41 52 41 4c 4c 45 4c 20  en-MAX_PARALLEL 
8880: 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20 64 65 6c  );.    }.    del
8890: 61 79 2e 74 76 5f 73 65 63 20 3d 20 36 30 3b 0a  ay.tv_sec = 60;.
88a0: 20 20 20 20 64 65 6c 61 79 2e 74 76 5f 75 73 65      delay.tv_use
88b0: 63 20 3d 20 30 3b 0a 20 20 20 20 46 44 5f 5a 45  c = 0;.    FD_ZE
88c0: 52 4f 28 26 72 65 61 64 66 64 73 29 3b 0a 20 20  RO(&readfds);.  
88d0: 20 20 46 44 5f 53 45 54 28 20 6c 69 73 74 65 6e    FD_SET( listen
88e0: 65 72 2c 20 26 72 65 61 64 66 64 73 29 3b 0a 20  er, &readfds);. 
88f0: 20 20 20 69 66 28 20 73 65 6c 65 63 74 28 20 6c     if( select( l
8900: 69 73 74 65 6e 65 72 2b 31 2c 20 26 72 65 61 64  istener+1, &read
8910: 66 64 73 2c 20 30 2c 20 30 2c 20 26 64 65 6c 61  fds, 0, 0, &dela
8920: 79 29 20 29 7b 0a 20 20 20 20 20 20 6c 65 6e 61  y) ){.      lena
8930: 64 64 72 20 3d 20 73 69 7a 65 6f 66 28 69 6e 61  ddr = sizeof(ina
8940: 64 64 72 29 3b 0a 20 20 20 20 20 20 63 6f 6e 6e  ddr);.      conn
8950: 65 63 74 69 6f 6e 20 3d 20 61 63 63 65 70 74 28  ection = accept(
8960: 6c 69 73 74 65 6e 65 72 2c 20 28 73 74 72 75 63  listener, (struc
8970: 74 20 73 6f 63 6b 61 64 64 72 2a 29 26 69 6e 61  t sockaddr*)&ina
8980: 64 64 72 2c 0a 20 20 20 20 20 20 20 20 20 20 20  ddr,.           
8990: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
89a0: 20 20 20 20 20 20 20 20 20 28 73 6f 63 6b 6c 65           (sockle
89b0: 6e 5f 74 2a 29 20 26 6c 65 6e 61 64 64 72 29 3b  n_t*) &lenaddr);
89c0: 0a 20 20 20 20 20 20 69 66 28 20 63 6f 6e 6e 65  .      if( conne
89d0: 63 74 69 6f 6e 3e 3d 30 20 29 7b 0a 20 20 20 20  ction>=0 ){.    
89e0: 20 20 20 20 63 68 69 6c 64 20 3d 20 66 6f 72 6b      child = fork
89f0: 28 29 3b 0a 20 20 20 20 20 20 20 20 69 66 28 20  ();.        if( 
8a00: 63 68 69 6c 64 21 3d 30 20 29 7b 0a 20 20 20 20  child!=0 ){.    
8a10: 20 20 20 20 20 20 69 66 28 20 63 68 69 6c 64 3e        if( child>
8a20: 30 20 29 20 6e 63 68 69 6c 64 72 65 6e 2b 2b 3b  0 ) nchildren++;
8a30: 0a 20 20 20 20 20 20 20 20 20 20 63 6c 6f 73 65  .          close
8a40: 28 63 6f 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 20  (connection);.  
8a50: 20 20 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20        }else{.   
8a60: 20 20 20 20 20 20 20 63 6c 6f 73 65 28 30 29 3b         close(0);
8a70: 0a 20 20 20 20 20 20 20 20 20 20 64 75 70 28 63  .          dup(c
8a80: 6f 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 20 20 20  onnection);.    
8a90: 20 20 20 20 20 20 63 6c 6f 73 65 28 31 29 3b 0a        close(1);.
8aa0: 20 20 20 20 20 20 20 20 20 20 64 75 70 28 63 6f            dup(co
8ab0: 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 20 20 20 20  nnection);.     
8ac0: 20 20 20 20 20 69 66 28 20 21 67 2e 66 48 74 74       if( !g.fHtt
8ad0: 70 54 72 61 63 65 20 29 7b 0a 20 20 20 20 20 20  pTrace ){.      
8ae0: 20 20 20 20 20 20 63 6c 6f 73 65 28 32 29 3b 0a        close(2);.
8af0: 20 20 20 20 20 20 20 20 20 20 20 20 64 75 70 28              dup(
8b00: 63 6f 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 20 20  connection);.   
8b10: 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 20         }.       
8b20: 20 20 20 63 6c 6f 73 65 28 63 6f 6e 6e 65 63 74     close(connect
8b30: 69 6f 6e 29 3b 0a 20 20 20 20 20 20 20 20 20 20  ion);.          
8b40: 72 65 74 75 72 6e 20 30 3b 0a 20 20 20 20 20 20  return 0;.      
8b50: 20 20 7d 0a 20 20 20 20 20 20 7d 0a 20 20 20 20    }.      }.    
8b60: 7d 0a 20 20 20 20 2f 2a 20 42 75 72 79 20 64 65  }.    /* Bury de
8b70: 61 64 20 63 68 69 6c 64 72 65 6e 20 2a 2f 0a 20  ad children */. 
8b80: 20 20 20 77 68 69 6c 65 28 20 77 61 69 74 70 69     while( waitpi
8b90: 64 28 30 2c 20 30 2c 20 57 4e 4f 48 41 4e 47 29  d(0, 0, WNOHANG)
8ba0: 3e 30 20 29 7b 0a 20 20 20 20 20 20 6e 63 68 69  >0 ){.      nchi
8bb0: 6c 64 72 65 6e 2d 2d 3b 0a 20 20 20 20 7d 0a 20  ldren--;.    }. 
8bc0: 20 7d 0a 20 20 2f 2a 20 4e 4f 54 20 52 45 41 43   }.  /* NOT REAC
8bd0: 48 45 44 20 2a 2f 20 20 0a 20 20 65 78 69 74 28  HED */  .  exit(
8be0: 31 29 3b 0a 23 65 6e 64 69 66 0a 7d 0a 0a 0a 2f  1);.#endif.}.../
8bf0: 2a 0a 2a 2a 20 4e 61 6d 65 20 6f 66 20 64 61 79  *.** Name of day
8c00: 73 20 61 6e 64 20 6d 6f 6e 74 68 73 2e 0a 2a 2f  s and months..*/
8c10: 0a 73 74 61 74 69 63 20 63 6f 6e 73 74 20 63 68  .static const ch
8c20: 61 72 20 2a 61 7a 44 61 79 73 5b 5d 20 3d 0a 20  ar *azDays[] =. 
8c30: 20 20 20 7b 22 53 75 6e 22 2c 20 22 4d 6f 6e 22     {"Sun", "Mon"
8c40: 2c 20 22 54 75 65 22 2c 20 22 57 65 64 22 2c 20  , "Tue", "Wed", 
8c50: 22 54 68 75 22 2c 20 22 46 72 69 22 2c 20 22 53  "Thu", "Fri", "S
8c60: 61 74 22 2c 20 30 7d 3b 0a 73 74 61 74 69 63 20  at", 0};.static 
8c70: 63 6f 6e 73 74 20 63 68 61 72 20 2a 61 7a 4d 6f  const char *azMo
8c80: 6e 74 68 73 5b 5d 20 3d 0a 20 20 20 20 7b 22 4a  nths[] =.    {"J
8c90: 61 6e 22 2c 20 22 46 65 62 22 2c 20 22 4d 61 72  an", "Feb", "Mar
8ca0: 22 2c 20 22 41 70 72 22 2c 20 22 4d 61 79 22 2c  ", "Apr", "May",
8cb0: 20 22 4a 75 6e 22 2c 0a 20 20 20 20 20 22 4a 75   "Jun",.     "Ju
8cc0: 6c 22 2c 20 22 41 75 67 22 2c 20 22 53 65 70 22  l", "Aug", "Sep"
8cd0: 2c 20 22 4f 63 74 22 2c 20 22 4e 6f 76 22 2c 20  , "Oct", "Nov", 
8ce0: 22 44 65 63 22 2c 20 30 7d 3b 0a 0a 0a 2f 2a 0a  "Dec", 0};.../*.
8cf0: 2a 2a 20 52 65 74 75 72 6e 73 20 61 6e 20 52 46  ** Returns an RF
8d00: 43 38 32 32 2d 66 6f 72 6d 61 74 74 65 64 20 74  C822-formatted t
8d10: 69 6d 65 20 73 74 72 69 6e 67 20 73 75 69 74 61  ime string suita
8d20: 62 6c 65 20 66 6f 72 20 48 54 54 50 20 68 65 61  ble for HTTP hea
8d30: 64 65 72 73 2c 20 61 6d 6f 6e 67 0a 2a 2a 20 6f  ders, among.** o
8d40: 74 68 65 72 20 74 68 69 6e 67 73 2e 0a 2a 2a 20  ther things..** 
8d50: 52 65 74 75 72 6e 65 64 20 74 69 6d 65 7a 6f 6e  Returned timezon
8d60: 65 20 69 73 20 61 6c 77 61 79 73 20 47 4d 54 20  e is always GMT 
8d70: 61 73 20 72 65 71 75 69 72 65 64 20 62 79 20 48  as required by H
8d80: 54 54 50 2f 31 2e 31 20 73 70 65 63 69 66 69 63  TTP/1.1 specific
8d90: 61 74 69 6f 6e 2e 0a 2a 2a 0a 2a 2a 20 53 65 65  ation..**.** See
8da0: 20 68 74 74 70 3a 2f 2f 77 77 77 2e 66 61 71 73   http://www.faqs
8db0: 2e 6f 72 67 2f 72 66 63 73 2f 72 66 63 38 32 32  .org/rfcs/rfc822
8dc0: 2e 68 74 6d 6c 2c 20 73 65 63 74 69 6f 6e 20 35  .html, section 5
8dd0: 0a 2a 2a 20 61 6e 64 20 68 74 74 70 3a 2f 2f 77  .** and http://w
8de0: 77 77 2e 66 61 71 73 2e 6f 72 67 2f 72 66 63 73  ww.faqs.org/rfcs
8df0: 2f 72 66 63 32 36 31 36 2e 68 74 6d 6c 2c 20 73  /rfc2616.html, s
8e00: 65 63 74 69 6f 6e 20 33 2e 33 2e 0a 2a 2f 0a 63  ection 3.3..*/.c
8e10: 68 61 72 20 2a 63 67 69 5f 72 66 63 38 32 32 5f  har *cgi_rfc822_
8e20: 64 61 74 65 73 74 61 6d 70 28 74 69 6d 65 5f 74  datestamp(time_t
8e30: 20 6e 6f 77 29 7b 0a 20 20 73 74 72 75 63 74 20   now){.  struct 
8e40: 74 6d 20 2a 70 54 6d 3b 0a 20 20 70 54 6d 20 3d  tm *pTm;.  pTm =
8e50: 20 67 6d 74 69 6d 65 28 26 6e 6f 77 29 3b 0a 20   gmtime(&now);. 
8e60: 20 69 66 28 20 70 54 6d 3d 3d 30 20 29 20 72 65   if( pTm==0 ) re
8e70: 74 75 72 6e 20 22 22 3b 0a 20 20 72 65 74 75 72  turn "";.  retur
8e80: 6e 20 6d 70 72 69 6e 74 66 28 22 25 73 2c 20 25  n mprintf("%s, %
8e90: 64 20 25 73 20 25 30 32 64 20 25 30 32 64 3a 25  d %s %02d %02d:%
8ea0: 30 32 64 3a 25 30 32 64 20 47 4d 54 22 2c 0a 20  02d:%02d GMT",. 
8eb0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
8ec0: 61 7a 44 61 79 73 5b 70 54 6d 2d 3e 74 6d 5f 77  azDays[pTm->tm_w
8ed0: 64 61 79 5d 2c 20 70 54 6d 2d 3e 74 6d 5f 6d 64  day], pTm->tm_md
8ee0: 61 79 2c 20 61 7a 4d 6f 6e 74 68 73 5b 70 54 6d  ay, azMonths[pTm
8ef0: 2d 3e 74 6d 5f 6d 6f 6e 5d 2c 0a 20 20 20 20 20  ->tm_mon],.     
8f00: 20 20 20 20 20 20 20 20 20 20 20 20 70 54 6d 2d              pTm-
8f10: 3e 74 6d 5f 79 65 61 72 2b 31 39 30 30 2c 20 70  >tm_year+1900, p
8f20: 54 6d 2d 3e 74 6d 5f 68 6f 75 72 2c 20 70 54 6d  Tm->tm_hour, pTm
8f30: 2d 3e 74 6d 5f 6d 69 6e 2c 20 70 54 6d 2d 3e 74  ->tm_min, pTm->t
8f40: 6d 5f 73 65 63 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a  m_sec);.}../*.**
8f50: 20 50 61 72 73 65 20 61 6e 20 52 46 43 38 32 32   Parse an RFC822
8f60: 2d 66 6f 72 6d 61 74 74 65 64 20 74 69 6d 65 73  -formatted times
8f70: 74 61 6d 70 20 61 73 20 77 65 27 64 20 65 78 70  tamp as we'd exp
8f80: 65 63 74 20 66 72 6f 6d 20 48 54 54 50 20 61 6e  ect from HTTP an
8f90: 64 20 72 65 74 75 72 6e 0a 2a 2a 20 61 20 55 6e  d return.** a Un
8fa0: 69 78 20 65 70 6f 63 68 20 74 69 6d 65 2e 20 3c  ix epoch time. <
8fb0: 3d 20 7a 65 72 6f 20 69 73 20 72 65 74 75 72 6e  = zero is return
8fc0: 65 64 20 6f 6e 20 66 61 69 6c 75 72 65 2e 0a 2a  ed on failure..*
8fd0: 2a 0a 2a 2a 20 4e 6f 74 65 20 74 68 61 74 20 74  *.** Note that t
8fe0: 68 69 73 20 77 6f 6e 27 74 20 68 61 6e 64 6c 65  his won't handle
8ff0: 20 61 6c 6c 20 74 68 65 20 5f 61 6c 6c 6f 77 65   all the _allowe
9000: 64 5f 20 48 54 54 50 20 66 6f 72 6d 61 74 73 2c  d_ HTTP formats,
9010: 20 6a 75 73 74 20 74 68 65 0a 2a 2a 20 6d 6f 73   just the.** mos
9020: 74 20 70 6f 70 75 6c 61 72 20 6f 6e 65 20 28 74  t popular one (t
9030: 68 65 20 6f 6e 65 20 67 65 6e 65 72 61 74 65 64  he one generated
9040: 20 62 79 20 63 67 69 5f 72 66 63 38 32 32 5f 64   by cgi_rfc822_d
9050: 61 74 65 73 74 61 6d 70 28 29 2c 20 61 63 74 75  atestamp(), actu
9060: 61 6c 6c 79 29 2e 0a 2a 2f 0a 74 69 6d 65 5f 74  ally)..*/.time_t
9070: 20 63 67 69 5f 72 66 63 38 32 32 5f 70 61 72 73   cgi_rfc822_pars
9080: 65 64 61 74 65 28 63 6f 6e 73 74 20 63 68 61 72  edate(const char
9090: 20 2a 7a 44 61 74 65 29 7b 0a 20 20 73 74 72 75   *zDate){.  stru
90a0: 63 74 20 74 6d 20 74 3b 0a 20 20 63 68 61 72 20  ct tm t;.  char 
90b0: 7a 49 67 6e 6f 72 65 5b 31 36 5d 3b 0a 20 20 63  zIgnore[16];.  c
90c0: 68 61 72 20 7a 4d 6f 6e 74 68 5b 31 36 5d 3b 0a  har zMonth[16];.
90d0: 0a 20 20 6d 65 6d 73 65 74 28 26 74 2c 20 30 2c  .  memset(&t, 0,
90e0: 20 73 69 7a 65 6f 66 28 74 29 29 3b 0a 20 20 69   sizeof(t));.  i
90f0: 66 28 20 37 3d 3d 73 73 63 61 6e 66 28 7a 44 61  f( 7==sscanf(zDa
9100: 74 65 2c 20 22 25 31 32 5b 41 2d 5a 61 2d 7a 2c  te, "%12[A-Za-z,
9110: 5d 20 25 64 20 25 31 32 5b 41 2d 5a 61 2d 7a 5d  ] %d %12[A-Za-z]
9120: 20 25 64 20 25 64 3a 25 64 3a 25 64 22 2c 20 7a   %d %d:%d:%d", z
9130: 49 67 6e 6f 72 65 2c 0a 20 20 20 20 20 20 20 20  Ignore,.        
9140: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 26                 &
9150: 74 2e 74 6d 5f 6d 64 61 79 2c 20 7a 4d 6f 6e 74  t.tm_mday, zMont
9160: 68 2c 20 26 74 2e 74 6d 5f 79 65 61 72 2c 20 26  h, &t.tm_year, &
9170: 74 2e 74 6d 5f 68 6f 75 72 2c 20 26 74 2e 74 6d  t.tm_hour, &t.tm
9180: 5f 6d 69 6e 2c 0a 20 20 20 20 20 20 20 20 20 20  _min,.          
9190: 20 20 20 20 20 20 20 20 20 20 20 20 20 26 74 2e               &t.
91a0: 74 6d 5f 73 65 63 29 29 7b 0a 0a 20 20 20 20 69  tm_sec)){..    i
91b0: 66 28 20 74 2e 74 6d 5f 79 65 61 72 20 3e 20 31  f( t.tm_year > 1
91c0: 39 30 30 20 29 20 74 2e 74 6d 5f 79 65 61 72 20  900 ) t.tm_year 
91d0: 2d 3d 20 31 39 30 30 3b 0a 20 20 20 20 66 6f 72  -= 1900;.    for
91e0: 28 74 2e 74 6d 5f 6d 6f 6e 3d 30 3b 20 61 7a 4d  (t.tm_mon=0; azM
91f0: 6f 6e 74 68 73 5b 74 2e 74 6d 5f 6d 6f 6e 5d 3b  onths[t.tm_mon];
9200: 20 74 2e 74 6d 5f 6d 6f 6e 2b 2b 29 7b 0a 20 20   t.tm_mon++){.  
9210: 20 20 20 20 69 66 28 20 21 73 74 72 6e 63 61 73      if( !strncas
9220: 65 63 6d 70 28 20 61 7a 4d 6f 6e 74 68 73 5b 74  ecmp( azMonths[t
9230: 2e 74 6d 5f 6d 6f 6e 5d 2c 20 7a 4d 6f 6e 74 68  .tm_mon], zMonth
9240: 2c 20 33 20 29 29 7b 0a 20 20 20 20 20 20 20 20  , 3 )){.        
9250: 72 65 74 75 72 6e 20 6d 6b 67 6d 74 69 6d 65 28  return mkgmtime(
9260: 26 74 29 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20  &t);.      }.   
9270: 20 7d 0a 20 20 7d 0a 0a 20 20 72 65 74 75 72 6e   }.  }..  return
9280: 20 30 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 43 6f 6e   0;.}../*.** Con
9290: 76 65 72 74 20 61 20 73 74 72 75 63 74 20 74 6d  vert a struct tm
92a0: 2a 20 74 68 61 74 20 72 65 70 72 65 73 65 6e 74  * that represent
92b0: 73 20 61 20 6d 6f 6d 65 6e 74 20 69 6e 20 55 54  s a moment in UT
92c0: 43 20 69 6e 74 6f 20 74 68 65 20 6e 75 6d 62 65  C into the numbe
92d0: 72 0a 2a 2a 20 6f 66 20 73 65 63 6f 6e 64 73 20  r.** of seconds 
92e0: 69 6e 20 31 39 37 30 2c 20 55 54 43 2e 0a 2a 2f  in 1970, UTC..*/
92f0: 0a 74 69 6d 65 5f 74 20 6d 6b 67 6d 74 69 6d 65  .time_t mkgmtime
9300: 28 73 74 72 75 63 74 20 74 6d 20 2a 70 29 7b 0a  (struct tm *p){.
9310: 20 20 74 69 6d 65 5f 74 20 74 3b 0a 20 20 69 6e    time_t t;.  in
9320: 74 20 6e 44 61 79 3b 0a 20 20 69 6e 74 20 69 73  t nDay;.  int is
9330: 4c 65 61 70 59 72 3b 0a 20 20 2f 2a 20 44 61 79  LeapYr;.  /* Day
9340: 73 20 69 6e 20 65 61 63 68 20 6d 6f 6e 74 68 3a  s in each month:
9350: 20 20 20 20 20 20 20 33 31 2c 20 32 38 2c 20 33         31, 28, 3
9360: 31 2c 20 33 30 2c 20 33 31 2c 20 33 30 2c 20 33  1, 30, 31, 30, 3
9370: 31 2c 20 33 31 2c 20 33 30 2c 20 33 31 2c 20 33  1, 31, 30, 31, 3
9380: 30 2c 20 33 31 20 2a 2f 0a 20 20 73 74 61 74 69  0, 31 */.  stati
9390: 63 20 69 6e 74 20 70 72 69 6f 72 44 61 79 73 5b  c int priorDays[
93a0: 5d 20 20 20 3d 20 7b 20 20 30 2c 20 33 31 2c 20  ]   = {  0, 31, 
93b0: 35 39 2c 20 39 30 2c 31 32 30 2c 31 35 31 2c 31  59, 90,120,151,1
93c0: 38 31 2c 32 31 32 2c 32 34 33 2c 32 37 33 2c 33  81,212,243,273,3
93d0: 30 34 2c 33 33 34 20 7d 3b 0a 20 20 69 66 28 20  04,334 };.  if( 
93e0: 70 2d 3e 74 6d 5f 6d 6f 6e 3c 30 20 29 7b 0a 20  p->tm_mon<0 ){. 
93f0: 20 20 20 69 6e 74 20 6e 59 65 61 72 20 3d 20 28     int nYear = (
9400: 31 31 20 2d 20 70 2d 3e 74 6d 5f 6d 6f 6e 29 2f  11 - p->tm_mon)/
9410: 31 32 3b 0a 20 20 20 20 70 2d 3e 74 6d 5f 79 65  12;.    p->tm_ye
9420: 61 72 20 2d 3d 20 6e 59 65 61 72 3b 0a 20 20 20  ar -= nYear;.   
9430: 20 70 2d 3e 74 6d 5f 6d 6f 6e 20 2b 3d 20 6e 59   p->tm_mon += nY
9440: 65 61 72 2a 31 32 3b 0a 20 20 7d 65 6c 73 65 20  ear*12;.  }else 
9450: 69 66 28 20 70 2d 3e 74 6d 5f 6d 6f 6e 3e 31 31  if( p->tm_mon>11
9460: 20 29 7b 0a 20 20 20 20 70 2d 3e 74 6d 5f 79 65   ){.    p->tm_ye
9470: 61 72 20 2b 3d 20 70 2d 3e 74 6d 5f 6d 6f 6e 2f  ar += p->tm_mon/
9480: 31 32 3b 0a 20 20 20 20 70 2d 3e 74 6d 5f 6d 6f  12;.    p->tm_mo
9490: 6e 20 25 3d 20 31 32 3b 0a 20 20 7d 0a 20 20 69  n %= 12;.  }.  i
94a0: 73 4c 65 61 70 59 72 20 3d 20 70 2d 3e 74 6d 5f  sLeapYr = p->tm_
94b0: 79 65 61 72 25 34 3d 3d 30 20 26 26 20 28 70 2d  year%4==0 && (p-
94c0: 3e 74 6d 5f 79 65 61 72 25 31 30 30 21 3d 30 20  >tm_year%100!=0 
94d0: 7c 7c 20 28 70 2d 3e 74 6d 5f 79 65 61 72 2b 33  || (p->tm_year+3
94e0: 30 30 29 25 34 30 30 3d 3d 30 29 3b 0a 20 20 70  00)%400==0);.  p
94f0: 2d 3e 74 6d 5f 79 64 61 79 20 3d 20 70 72 69 6f  ->tm_yday = prio
9500: 72 44 61 79 73 5b 70 2d 3e 74 6d 5f 6d 6f 6e 5d  rDays[p->tm_mon]
9510: 20 2b 20 70 2d 3e 74 6d 5f 6d 64 61 79 20 2d 20   + p->tm_mday - 
9520: 31 3b 0a 20 20 69 66 28 20 69 73 4c 65 61 70 59  1;.  if( isLeapY
9530: 72 20 26 26 20 70 2d 3e 74 6d 5f 6d 6f 6e 3e 31  r && p->tm_mon>1
9540: 20 29 20 70 2d 3e 74 6d 5f 79 64 61 79 2b 2b 3b   ) p->tm_yday++;
9550: 0a 20 20 6e 44 61 79 20 3d 20 28 70 2d 3e 74 6d  .  nDay = (p->tm
9560: 5f 79 65 61 72 2d 37 30 29 2a 33 36 35 20 2b 20  _year-70)*365 + 
9570: 28 70 2d 3e 74 6d 5f 79 65 61 72 2d 36 39 29 2f  (p->tm_year-69)/
9580: 34 20 2d 70 2d 3e 74 6d 5f 79 65 61 72 2f 31 30  4 -p->tm_year/10
9590: 30 20 2b 20 0a 20 20 20 20 20 20 20 20 20 28 70  0 + .         (p
95a0: 2d 3e 74 6d 5f 79 65 61 72 2b 33 30 30 29 2f 34  ->tm_year+300)/4
95b0: 30 30 20 2b 20 70 2d 3e 74 6d 5f 79 64 61 79 3b  00 + p->tm_yday;
95c0: 0a 20 20 74 20 3d 20 28 28 6e 44 61 79 2a 32 34  .  t = ((nDay*24
95d0: 20 2b 20 70 2d 3e 74 6d 5f 68 6f 75 72 29 2a 36   + p->tm_hour)*6
95e0: 30 20 2b 20 70 2d 3e 74 6d 5f 6d 69 6e 29 2a 36  0 + p->tm_min)*6
95f0: 30 20 2b 20 70 2d 3e 74 6d 5f 73 65 63 3b 0a 20  0 + p->tm_sec;. 
9600: 20 72 65 74 75 72 6e 20 74 3b 0a 7d 0a 0a 2f 2a   return t;.}../*
9610: 0a 2a 2a 20 43 68 65 63 6b 20 74 68 65 20 6f 62  .** Check the ob
9620: 6a 65 63 74 54 69 6d 65 20 61 67 61 69 6e 73 74  jectTime against
9630: 20 74 68 65 20 49 66 2d 4d 6f 64 69 66 69 65 64   the If-Modified
9640: 2d 53 69 6e 63 65 20 72 65 71 75 65 73 74 20 68  -Since request h
9650: 65 61 64 65 72 2e 20 49 66 20 74 68 65 0a 2a 2a  eader. If the.**
9660: 20 6f 62 6a 65 63 74 20 74 69 6d 65 20 69 73 6e   object time isn
9670: 27 74 20 61 6e 79 20 6e 65 77 65 72 20 74 68 61  't any newer tha
9680: 6e 20 74 68 65 20 68 65 61 64 65 72 2c 20 77 65  n the header, we
9690: 20 69 6d 6d 65 64 69 61 74 65 6c 79 20 73 65 6e   immediately sen
96a0: 64 20 62 61 63 6b 0a 2a 2a 20 61 20 33 30 34 20  d back.** a 304 
96b0: 72 65 70 6c 79 20 61 6e 64 20 65 78 69 74 2e 0a  reply and exit..
96c0: 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 6d 6f 64 69  */.void cgi_modi
96d0: 66 69 65 64 5f 73 69 6e 63 65 28 74 69 6d 65 5f  fied_since(time_
96e0: 74 20 6f 62 6a 65 63 74 54 69 6d 65 29 7b 0a 20  t objectTime){. 
96f0: 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 49 66   const char *zIf
9700: 20 3d 20 50 28 22 48 54 54 50 5f 49 46 5f 4d 4f   = P("HTTP_IF_MO
9710: 44 49 46 49 45 44 5f 53 49 4e 43 45 22 29 3b 0a  DIFIED_SINCE");.
9720: 20 20 69 66 28 20 7a 49 66 3d 3d 30 20 29 20 72    if( zIf==0 ) r
9730: 65 74 75 72 6e 3b 0a 20 20 69 66 28 20 6f 62 6a  eturn;.  if( obj
9740: 65 63 74 54 69 6d 65 20 3e 20 63 67 69 5f 72 66  ectTime > cgi_rf
9750: 63 38 32 32 5f 70 61 72 73 65 64 61 74 65 28 7a  c822_parsedate(z
9760: 49 66 29 20 29 20 72 65 74 75 72 6e 3b 0a 20 20  If) ) return;.  
9770: 63 67 69 5f 73 65 74 5f 73 74 61 74 75 73 28 33  cgi_set_status(3
9780: 30 34 2c 22 4e 6f 74 20 4d 6f 64 69 66 69 65 64  04,"Not Modified
9790: 22 29 3b 0a 20 20 63 67 69 5f 72 65 73 65 74 5f  ");.  cgi_reset_
97a0: 63 6f 6e 74 65 6e 74 28 29 3b 0a 20 20 63 67 69  content();.  cgi
97b0: 5f 72 65 70 6c 79 28 29 3b 0a 20 20 65 78 69 74  _reply();.  exit
97c0: 28 30 29 3b 0a 7d 0a                             (0);.}.