Hex Artifact Content
Not logged in

Artifact 1c055038717de0273b07651aef227c7341d76a5f:

File src/cgi.c part of check-in [fb358ca492] - Progress toward getting ticketing working. We can enter a new ticket and display it. Cannot yet edit a ticket. by drh on 2007-11-24 19:33:46. Also file src/cgi.c part of check-in [d0305b305a] - Merged mainline into my branch to get the newest application. by aku on 2007-12-05 08:07:46.

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 23 65 6e 64 69 66 20  ),(y)))..#endif 
08a0: 2f 2a 20 49 4e 54 45 52 46 41 43 45 20 2a 2f 0a  /* INTERFACE */.
08b0: 0a 2f 2a 0a 2a 2a 20 50 72 6f 76 69 64 65 20 61  ./*.** Provide a
08c0: 20 72 65 6c 69 61 62 6c 65 20 69 6d 70 6c 65 6d   reliable implem
08d0: 65 6e 74 61 74 69 6f 6e 20 6f 66 20 61 20 63 61  entation of a ca
08e0: 73 65 6c 65 73 73 20 73 74 72 69 6e 67 20 63 6f  seless string co
08f0: 6d 70 61 72 69 73 6f 6e 0a 2a 2a 20 66 75 6e 63  mparison.** func
0900: 74 69 6f 6e 2e 0a 2a 2f 0a 23 64 65 66 69 6e 65  tion..*/.#define
0910: 20 73 74 72 69 63 6d 70 20 73 71 6c 69 74 65 33   stricmp sqlite3
0920: 53 74 72 49 43 6d 70 0a 65 78 74 65 72 6e 20 69  StrICmp.extern i
0930: 6e 74 20 73 71 6c 69 74 65 33 53 74 72 49 43 6d  nt sqlite3StrICm
0940: 70 28 63 6f 6e 73 74 20 63 68 61 72 2a 2c 20 63  p(const char*, c
0950: 6f 6e 73 74 20 63 68 61 72 2a 29 3b 0a 0a 2f 2a  onst char*);../*
0960: 0a 2a 2a 20 54 68 65 20 62 6f 64 79 20 6f 66 20  .** The body of 
0970: 74 68 65 20 48 54 54 50 20 72 65 70 6c 79 20 74  the HTTP reply t
0980: 65 78 74 20 69 73 20 73 74 6f 72 65 64 20 68 65  ext is stored he
0990: 72 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 42 6c  re..*/.static Bl
09a0: 6f 62 20 63 67 69 43 6f 6e 74 65 6e 74 20 3d 20  ob cgiContent = 
09b0: 42 4c 4f 42 5f 49 4e 49 54 49 41 4c 49 5a 45 52  BLOB_INITIALIZER
09c0: 3b 0a 0a 2f 2a 0a 2a 2a 20 41 70 70 65 6e 64 20  ;../*.** Append 
09d0: 72 65 70 6c 79 20 63 6f 6e 74 65 6e 74 20 74 6f  reply content to
09e0: 20 77 68 61 74 20 61 6c 72 65 61 64 79 20 65 78   what already ex
09f0: 69 73 74 73 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67  ists..*/.void cg
0a00: 69 5f 61 70 70 65 6e 64 5f 63 6f 6e 74 65 6e 74  i_append_content
0a10: 28 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 44 61  (const char *zDa
0a20: 74 61 2c 20 69 6e 74 20 6e 41 6d 74 29 7b 0a 20  ta, int nAmt){. 
0a30: 20 62 6c 6f 62 5f 61 70 70 65 6e 64 28 26 63 67   blob_append(&cg
0a40: 69 43 6f 6e 74 65 6e 74 2c 20 7a 44 61 74 61 2c  iContent, zData,
0a50: 20 6e 41 6d 74 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a   nAmt);.}../*.**
0a60: 20 52 65 73 65 74 20 74 68 65 20 48 54 54 50 20   Reset the HTTP 
0a70: 72 65 70 6c 79 20 74 65 78 74 20 74 6f 20 62 65  reply text to be
0a80: 20 61 6e 20 65 6d 70 74 79 20 73 74 72 69 6e 67   an empty string
0a90: 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 72 65  ..*/.void cgi_re
0aa0: 73 65 74 5f 63 6f 6e 74 65 6e 74 28 76 6f 69 64  set_content(void
0ab0: 29 7b 0a 20 20 62 6c 6f 62 5f 72 65 73 65 74 28  ){.  blob_reset(
0ac0: 26 63 67 69 43 6f 6e 74 65 6e 74 29 3b 0a 7d 0a  &cgiContent);.}.
0ad0: 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 61 20  ./*.** Return a 
0ae0: 70 6f 69 6e 74 65 72 20 74 6f 20 74 68 65 20 43  pointer to the C
0af0: 47 49 20 6f 75 74 70 75 74 20 62 6c 6f 62 2e 0a  GI output blob..
0b00: 2a 2f 0a 42 6c 6f 62 20 2a 63 67 69 5f 6f 75 74  */.Blob *cgi_out
0b10: 70 75 74 5f 62 6c 6f 62 28 76 6f 69 64 29 7b 0a  put_blob(void){.
0b20: 20 20 72 65 74 75 72 6e 20 26 63 67 69 43 6f 6e    return &cgiCon
0b30: 74 65 6e 74 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 52  tent;.}../*.** R
0b40: 65 74 75 72 6e 20 61 20 70 6f 69 6e 74 65 72 20  eturn a pointer 
0b50: 74 6f 20 74 68 65 20 48 54 54 50 20 72 65 70 6c  to the HTTP repl
0b60: 79 20 74 65 78 74 2e 0a 2a 2f 0a 63 68 61 72 20  y text..*/.char 
0b70: 2a 63 67 69 5f 65 78 74 72 61 63 74 5f 63 6f 6e  *cgi_extract_con
0b80: 74 65 6e 74 28 69 6e 74 20 2a 70 6e 41 6d 74 29  tent(int *pnAmt)
0b90: 7b 0a 20 20 72 65 74 75 72 6e 20 62 6c 6f 62 5f  {.  return blob_
0ba0: 62 75 66 66 65 72 28 26 63 67 69 43 6f 6e 74 65  buffer(&cgiConte
0bb0: 6e 74 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 41 64  nt);.}../*.** Ad
0bc0: 64 69 74 69 6f 6e 61 6c 20 69 6e 66 6f 72 6d 61  ditional informa
0bd0: 74 69 6f 6e 20 75 73 65 64 20 74 6f 20 66 6f 72  tion used to for
0be0: 6d 20 74 68 65 20 48 54 54 50 20 72 65 70 6c 79  m the HTTP reply
0bf0: 0a 2a 2f 0a 73 74 61 74 69 63 20 63 68 61 72 20  .*/.static char 
0c00: 2a 7a 43 6f 6e 74 65 6e 74 54 79 70 65 20 3d 20  *zContentType = 
0c10: 22 74 65 78 74 2f 68 74 6d 6c 22 3b 20 20 20 20  "text/html";    
0c20: 20 2f 2a 20 43 6f 6e 74 65 6e 74 20 74 79 70 65   /* Content type
0c30: 20 6f 66 20 74 68 65 20 72 65 70 6c 79 20 2a 2f   of the reply */
0c40: 0a 73 74 61 74 69 63 20 63 68 61 72 20 2a 7a 52  .static char *zR
0c50: 65 70 6c 79 53 74 61 74 75 73 20 3d 20 22 4f 4b  eplyStatus = "OK
0c60: 22 3b 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a  ";            /*
0c70: 20 52 65 70 6c 79 20 73 74 61 74 75 73 20 64 65   Reply status de
0c80: 73 63 72 69 70 74 69 6f 6e 20 2a 2f 0a 73 74 61  scription */.sta
0c90: 74 69 63 20 69 6e 74 20 69 52 65 70 6c 79 53 74  tic int iReplySt
0ca0: 61 74 75 73 20 3d 20 32 30 30 3b 20 20 20 20 20  atus = 200;     
0cb0: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 52 65 70            /* Rep
0cc0: 6c 79 20 73 74 61 74 75 73 20 63 6f 64 65 20 2a  ly status code *
0cd0: 2f 0a 73 74 61 74 69 63 20 42 6c 6f 62 20 65 78  /.static Blob ex
0ce0: 74 72 61 48 65 61 64 65 72 20 3d 20 42 4c 4f 42  traHeader = BLOB
0cf0: 5f 49 4e 49 54 49 41 4c 49 5a 45 52 3b 20 20 2f  _INITIALIZER;  /
0d00: 2a 20 45 78 74 72 61 20 68 65 61 64 65 72 20 74  * Extra header t
0d10: 65 78 74 20 2a 2f 0a 73 74 61 74 69 63 20 69 6e  ext */.static in
0d20: 74 20 66 75 6c 6c 48 74 74 70 52 65 70 6c 79 20  t fullHttpReply 
0d30: 3d 20 30 3b 20 20 20 20 20 20 2f 2a 20 54 72 75  = 0;      /* Tru
0d40: 65 20 66 6f 72 20 61 20 66 75 6c 6c 2d 62 6c 6f  e for a full-blo
0d50: 77 6e 20 48 54 54 50 20 68 65 61 64 65 72 20 2a  wn HTTP header *
0d60: 2f 0a 0a 2f 2a 0a 2a 2a 20 53 65 74 20 74 68 65  /../*.** Set the
0d70: 20 72 65 70 6c 79 20 63 6f 6e 74 65 6e 74 20 74   reply content t
0d80: 79 70 65 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f  ype.*/.void cgi_
0d90: 73 65 74 5f 63 6f 6e 74 65 6e 74 5f 74 79 70 65  set_content_type
0da0: 28 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 54 79  (const char *zTy
0db0: 70 65 29 7b 0a 20 20 7a 43 6f 6e 74 65 6e 74 54  pe){.  zContentT
0dc0: 79 70 65 20 3d 20 6d 70 72 69 6e 74 66 28 22 25  ype = mprintf("%
0dd0: 73 22 2c 20 7a 54 79 70 65 29 3b 0a 7d 0a 0a 2f  s", zType);.}../
0de0: 2a 0a 2a 2a 20 53 65 74 20 74 68 65 20 72 65 70  *.** Set the rep
0df0: 6c 79 20 63 6f 6e 74 65 6e 74 20 74 6f 20 74 68  ly content to th
0e00: 65 20 73 70 65 63 69 66 69 65 64 20 42 4c 4f 42  e specified BLOB
0e10: 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 73 65  ..*/.void cgi_se
0e20: 74 5f 63 6f 6e 74 65 6e 74 28 42 6c 6f 62 20 2a  t_content(Blob *
0e30: 70 4e 65 77 43 6f 6e 74 65 6e 74 29 7b 0a 20 20  pNewContent){.  
0e40: 62 6c 6f 62 5f 72 65 73 65 74 28 26 63 67 69 43  blob_reset(&cgiC
0e50: 6f 6e 74 65 6e 74 29 3b 0a 20 20 63 67 69 43 6f  ontent);.  cgiCo
0e60: 6e 74 65 6e 74 20 3d 20 2a 70 4e 65 77 43 6f 6e  ntent = *pNewCon
0e70: 74 65 6e 74 3b 0a 20 20 62 6c 6f 62 5f 7a 65 72  tent;.  blob_zer
0e80: 6f 28 70 4e 65 77 43 6f 6e 74 65 6e 74 29 3b 0a  o(pNewContent);.
0e90: 7d 0a 0a 2f 2a 0a 2a 2a 20 53 65 74 20 74 68 65  }../*.** Set the
0ea0: 20 72 65 70 6c 79 20 73 74 61 74 75 73 20 63 6f   reply status co
0eb0: 64 65 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 73  de.*/.void cgi_s
0ec0: 65 74 5f 73 74 61 74 75 73 28 69 6e 74 20 69 53  et_status(int iS
0ed0: 74 61 74 2c 20 63 6f 6e 73 74 20 63 68 61 72 20  tat, const char 
0ee0: 2a 7a 53 74 61 74 29 7b 0a 20 20 7a 52 65 70 6c  *zStat){.  zRepl
0ef0: 79 53 74 61 74 75 73 20 3d 20 6d 70 72 69 6e 74  yStatus = mprint
0f00: 66 28 22 25 73 22 2c 20 7a 53 74 61 74 29 3b 0a  f("%s", zStat);.
0f10: 20 20 69 52 65 70 6c 79 53 74 61 74 75 73 20 3d    iReplyStatus =
0f20: 20 69 53 74 61 74 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a   iStat;.}../*.**
0f30: 20 41 70 70 65 6e 64 20 74 65 78 74 20 74 6f 20   Append text to 
0f40: 74 68 65 20 68 65 61 64 65 72 20 6f 66 20 61 6e  the header of an
0f50: 20 48 54 54 50 20 72 65 70 6c 79 0a 2a 2f 0a 76   HTTP reply.*/.v
0f60: 6f 69 64 20 63 67 69 5f 61 70 70 65 6e 64 5f 68  oid cgi_append_h
0f70: 65 61 64 65 72 28 63 6f 6e 73 74 20 63 68 61 72  eader(const char
0f80: 20 2a 7a 4c 69 6e 65 29 7b 0a 20 20 62 6c 6f 62   *zLine){.  blob
0f90: 5f 61 70 70 65 6e 64 28 26 65 78 74 72 61 48 65  _append(&extraHe
0fa0: 61 64 65 72 2c 20 7a 4c 69 6e 65 2c 20 2d 31 29  ader, zLine, -1)
0fb0: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 65 74 20 61  ;.}../*.** Set a
0fc0: 20 63 6f 6f 6b 69 65 2e 0a 2a 2a 0a 2a 2a 20 5a   cookie..**.** Z
0fd0: 65 72 6f 20 6c 69 66 65 74 69 6d 65 20 69 6d 70  ero lifetime imp
0fe0: 6c 69 65 73 20 61 20 73 65 73 73 69 6f 6e 20 63  lies a session c
0ff0: 6f 6f 6b 69 65 2e 0a 2a 2f 0a 76 6f 69 64 20 63  ookie..*/.void c
1000: 67 69 5f 73 65 74 5f 63 6f 6f 6b 69 65 28 0a 20  gi_set_cookie(. 
1010: 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e 61   const char *zNa
1020: 6d 65 2c 20 20 20 20 2f 2a 20 4e 61 6d 65 20 6f  me,    /* Name o
1030: 66 20 74 68 65 20 63 6f 6f 6b 69 65 20 2a 2f 0a  f the cookie */.
1040: 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 56    const char *zV
1050: 61 6c 75 65 2c 20 20 20 2f 2a 20 56 61 6c 75 65  alue,   /* Value
1060: 20 6f 66 20 74 68 65 20 63 6f 6f 6b 69 65 2e 20   of the cookie. 
1070: 20 41 75 74 6f 6d 61 74 69 63 61 6c 6c 79 20 65   Automatically e
1080: 73 63 61 70 65 64 20 2a 2f 0a 20 20 63 6f 6e 73  scaped */.  cons
1090: 74 20 63 68 61 72 20 2a 7a 50 61 74 68 2c 20 20  t char *zPath,  
10a0: 20 20 2f 2a 20 50 61 74 68 20 63 6f 6f 6b 69 65    /* Path cookie
10b0: 20 61 70 70 6c 69 65 73 20 74 6f 2e 20 20 4e 55   applies to.  NU
10c0: 4c 4c 20 6d 65 61 6e 73 20 22 2f 22 20 2a 2f 0a  LL means "/" */.
10d0: 20 20 69 6e 74 20 6c 69 66 65 74 69 6d 65 20 20    int lifetime  
10e0: 20 20 20 20 20 20 20 20 2f 2a 20 45 78 70 69 72          /* Expir
10f0: 61 74 69 6f 6e 20 6f 66 20 74 68 65 20 63 6f 6f  ation of the coo
1100: 6b 69 65 20 69 6e 20 73 65 63 6f 6e 64 73 20 66  kie in seconds f
1110: 72 6f 6d 20 6e 6f 77 20 2a 2f 0a 29 7b 0a 20 20  rom now */.){.  
1120: 69 66 28 20 7a 50 61 74 68 3d 3d 30 20 29 20 7a  if( zPath==0 ) z
1130: 50 61 74 68 20 3d 20 67 2e 7a 54 6f 70 3b 0a 20  Path = g.zTop;. 
1140: 20 69 66 28 20 6c 69 66 65 74 69 6d 65 3e 30 20   if( lifetime>0 
1150: 29 7b 0a 20 20 20 20 6c 69 66 65 74 69 6d 65 20  ){.    lifetime 
1160: 2b 3d 20 28 69 6e 74 29 74 69 6d 65 28 30 29 3b  += (int)time(0);
1170: 0a 20 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64  .    blob_append
1180: 66 28 26 65 78 74 72 61 48 65 61 64 65 72 2c 0a  f(&extraHeader,.
1190: 20 20 20 20 20 20 20 22 53 65 74 2d 43 6f 6f 6b         "Set-Cook
11a0: 69 65 3a 20 25 73 3d 25 74 3b 20 50 61 74 68 3d  ie: %s=%t; Path=
11b0: 25 73 3b 20 65 78 70 69 72 65 73 3d 25 73 3b 20  %s; expires=%s; 
11c0: 56 65 72 73 69 6f 6e 3d 31 5c 72 5c 6e 22 2c 0a  Version=1\r\n",.
11d0: 20 20 20 20 20 20 20 20 7a 4e 61 6d 65 2c 20 7a          zName, z
11e0: 56 61 6c 75 65 2c 20 7a 50 61 74 68 2c 20 63 67  Value, zPath, cg
11f0: 69 5f 72 66 63 38 32 32 5f 64 61 74 65 73 74 61  i_rfc822_datesta
1200: 6d 70 28 6c 69 66 65 74 69 6d 65 29 29 3b 0a 20  mp(lifetime));. 
1210: 20 7d 65 6c 73 65 7b 0a 20 20 20 20 62 6c 6f 62   }else{.    blob
1220: 5f 61 70 70 65 6e 64 66 28 26 65 78 74 72 61 48  _appendf(&extraH
1230: 65 61 64 65 72 2c 0a 20 20 20 20 20 20 20 22 53  eader,.       "S
1240: 65 74 2d 43 6f 6f 6b 69 65 3a 20 25 73 3d 25 74  et-Cookie: %s=%t
1250: 3b 20 50 61 74 68 3d 25 73 3b 20 56 65 72 73 69  ; Path=%s; Versi
1260: 6f 6e 3d 31 5c 72 5c 6e 22 2c 0a 20 20 20 20 20  on=1\r\n",.     
1270: 20 20 7a 4e 61 6d 65 2c 20 7a 56 61 6c 75 65 2c    zName, zValue,
1280: 20 7a 50 61 74 68 29 3b 0a 20 20 7d 0a 7d 0a 0a   zPath);.  }.}..
1290: 23 69 66 20 30 0a 2f 2a 0a 2a 2a 20 41 64 64 20  #if 0./*.** Add 
12a0: 61 6e 20 45 54 61 67 20 68 65 61 64 65 72 20 6c  an ETag header l
12b0: 69 6e 65 0a 2a 2f 0a 73 74 61 74 69 63 20 63 68  ine.*/.static ch
12c0: 61 72 20 2a 63 67 69 5f 61 64 64 5f 65 74 61 67  ar *cgi_add_etag
12d0: 28 63 68 61 72 20 2a 7a 54 78 74 2c 20 69 6e 74  (char *zTxt, int
12e0: 20 6e 4c 65 6e 29 7b 0a 20 20 4d 44 35 43 6f 6e   nLen){.  MD5Con
12f0: 74 65 78 74 20 63 74 78 3b 0a 20 20 75 6e 73 69  text ctx;.  unsi
1300: 67 6e 65 64 20 63 68 61 72 20 64 69 67 65 73 74  gned char digest
1310: 5b 31 36 5d 3b 0a 20 20 69 6e 74 20 69 2c 20 6a  [16];.  int i, j
1320: 3b 0a 20 20 63 68 61 72 20 7a 45 54 61 67 5b 36  ;.  char zETag[6
1330: 34 5d 3b 0a 0a 20 20 4d 44 35 49 6e 69 74 28 26  4];..  MD5Init(&
1340: 63 74 78 29 3b 0a 20 20 4d 44 35 55 70 64 61 74  ctx);.  MD5Updat
1350: 65 28 26 63 74 78 2c 7a 54 78 74 2c 6e 4c 65 6e  e(&ctx,zTxt,nLen
1360: 29 3b 0a 20 20 4d 44 35 46 69 6e 61 6c 28 64 69  );.  MD5Final(di
1370: 67 65 73 74 2c 26 63 74 78 29 3b 0a 20 20 66 6f  gest,&ctx);.  fo
1380: 72 28 6a 3d 69 3d 30 3b 20 69 3c 31 36 3b 20 69  r(j=i=0; i<16; i
1390: 2b 2b 2c 6a 2b 3d 32 29 7b 0a 20 20 20 20 62 70  ++,j+=2){.    bp
13a0: 72 69 6e 74 66 28 26 7a 45 54 61 67 5b 6a 5d 2c  rintf(&zETag[j],
13b0: 73 69 7a 65 6f 66 28 7a 45 54 61 67 29 2d 6a 2c  sizeof(zETag)-j,
13c0: 22 25 30 32 78 22 2c 28 69 6e 74 29 64 69 67 65  "%02x",(int)dige
13d0: 73 74 5b 69 5d 29 3b 0a 20 20 7d 0a 20 20 62 6c  st[i]);.  }.  bl
13e0: 6f 62 5f 61 70 70 65 6e 64 66 28 26 65 78 74 72  ob_appendf(&extr
13f0: 61 48 65 61 64 65 72 2c 20 22 45 54 61 67 3a 20  aHeader, "ETag: 
1400: 25 73 5c 72 5c 6e 22 2c 20 7a 45 54 61 67 29 3b  %s\r\n", zETag);
1410: 0a 20 20 72 65 74 75 72 6e 20 73 74 72 64 75 70  .  return strdup
1420: 28 7a 45 54 61 67 29 3b 0a 7d 0a 0a 2f 2a 0a 2a  (zETag);.}../*.*
1430: 2a 20 44 6f 20 73 6f 6d 65 20 63 61 63 68 65 20  * Do some cache 
1440: 63 6f 6e 74 72 6f 6c 20 73 74 75 66 66 2e 20 46  control stuff. F
1450: 69 72 73 74 2c 20 77 65 20 67 65 6e 65 72 61 74  irst, we generat
1460: 65 20 61 6e 20 45 54 61 67 20 61 6e 64 20 69 6e  e an ETag and in
1470: 63 6c 75 64 65 20 69 74 20 69 6e 0a 2a 2a 20 74  clude it in.** t
1480: 68 65 20 72 65 73 70 6f 6e 73 65 20 68 65 61 64  he response head
1490: 65 72 73 2e 20 53 65 63 6f 6e 64 2c 20 77 65 20  ers. Second, we 
14a0: 64 6f 20 77 68 61 74 65 76 65 72 20 69 73 20 6e  do whatever is n
14b0: 65 63 65 73 73 61 72 79 20 74 6f 20 64 65 74 65  ecessary to dete
14c0: 72 6d 69 6e 65 20 69 66 0a 2a 2a 20 74 68 65 20  rmine if.** the 
14d0: 72 65 71 75 65 73 74 20 77 61 73 20 61 73 6b 69  request was aski
14e0: 6e 67 20 61 62 6f 75 74 20 63 61 63 68 69 6e 67  ng about caching
14f0: 20 61 6e 64 20 77 68 65 74 68 65 72 20 77 65 20   and whether we 
1500: 6e 65 65 64 20 74 6f 20 73 65 6e 64 20 62 61 63  need to send bac
1510: 6b 20 74 68 65 0a 2a 2a 20 72 65 73 70 6f 6e 73  k the.** respons
1520: 65 20 62 6f 64 79 2e 20 49 66 20 77 65 20 73 68  e body. If we sh
1530: 6f 75 6c 64 6e 27 74 20 73 65 6e 64 20 61 20 62  ouldn't send a b
1540: 6f 64 79 2c 20 72 65 74 75 72 6e 20 6e 6f 6e 2d  ody, return non-
1550: 7a 65 72 6f 2e 0a 2a 2a 0a 2a 2a 20 43 75 72 72  zero..**.** Curr
1560: 65 6e 74 6c 79 2c 20 77 65 20 6a 75 73 74 20 63  ently, we just c
1570: 68 65 63 6b 20 74 68 65 20 45 54 61 67 20 61 67  heck the ETag ag
1580: 61 69 6e 73 74 20 61 6e 79 20 49 66 2d 4e 6f 6e  ainst any If-Non
1590: 65 2d 4d 61 74 63 68 20 68 65 61 64 65 72 2e 0a  e-Match header..
15a0: 2a 2a 0a 2a 2a 20 46 49 58 4d 45 3a 20 49 6e 20  **.** FIXME: In 
15b0: 73 6f 6d 65 20 63 61 73 65 73 20 28 61 74 74 61  some cases (atta
15c0: 63 68 6d 65 6e 74 73 2c 20 66 69 6c 65 20 63 6f  chments, file co
15d0: 6e 74 65 6e 74 73 29 20 77 65 20 63 6f 75 6c 64  ntents) we could
15e0: 20 63 68 65 63 6b 0a 2a 2a 20 49 66 2d 4d 6f 64   check.** If-Mod
15f0: 69 66 69 65 64 2d 53 69 6e 63 65 20 68 65 61 64  ified-Since head
1600: 65 72 73 20 61 6e 64 20 61 6c 77 61 79 73 20 69  ers and always i
1610: 6e 63 6c 75 64 65 20 4c 61 73 74 2d 4d 6f 64 69  nclude Last-Modi
1620: 66 69 65 64 20 69 6e 20 72 65 73 70 6f 6e 73 65  fied in response
1630: 73 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74  s..*/.static int
1640: 20 63 68 65 63 6b 5f 63 61 63 68 65 5f 63 6f 6e   check_cache_con
1650: 74 72 6f 6c 28 76 6f 69 64 29 7b 0a 20 20 2f 2a  trol(void){.  /*
1660: 20 46 49 58 4d 45 3a 20 74 68 65 72 65 27 73 20   FIXME: there's 
1670: 73 6f 6d 65 20 67 6f 74 63 68 61 73 20 77 74 68  some gotchas wth
1680: 20 63 6f 6f 6b 69 65 73 20 61 6e 64 20 73 6f 6d   cookies and som
1690: 65 20 68 65 61 64 65 72 73 2e 20 2a 2f 0a 20 20  e headers. */.  
16a0: 63 68 61 72 20 2a 7a 45 54 61 67 20 3d 20 63 67  char *zETag = cg
16b0: 69 5f 61 64 64 5f 65 74 61 67 28 62 6c 6f 62 5f  i_add_etag(blob_
16c0: 62 75 66 66 65 72 28 26 63 67 69 43 6f 6e 74 65  buffer(&cgiConte
16d0: 6e 74 29 2c 62 6c 6f 62 5f 73 69 7a 65 28 26 63  nt),blob_size(&c
16e0: 67 69 43 6f 6e 74 65 6e 74 29 29 3b 0a 20 20 63  giContent));.  c
16f0: 68 61 72 20 2a 7a 4d 61 74 63 68 20 3d 20 50 28  har *zMatch = P(
1700: 22 48 54 54 50 5f 49 46 5f 4e 4f 4e 45 5f 4d 41  "HTTP_IF_NONE_MA
1710: 54 43 48 22 29 3b 0a 0a 20 20 69 66 28 20 7a 45  TCH");..  if( zE
1720: 54 61 67 21 3d 30 20 26 26 20 7a 4d 61 74 63 68  Tag!=0 && zMatch
1730: 21 3d 30 20 29 20 7b 0a 20 20 20 20 63 68 61 72  !=0 ) {.    char
1740: 20 2a 7a 42 75 66 20 3d 20 73 74 72 64 75 70 28   *zBuf = strdup(
1750: 7a 4d 61 74 63 68 29 3b 0a 20 20 20 20 69 66 28  zMatch);.    if(
1760: 20 7a 42 75 66 21 3d 30 20 29 7b 0a 20 20 20 20   zBuf!=0 ){.    
1770: 20 20 63 68 61 72 20 2a 7a 54 6f 6b 20 3d 20 30    char *zTok = 0
1780: 3b 0a 20 20 20 20 20 20 63 68 61 72 20 2a 7a 50  ;.      char *zP
1790: 6f 73 3b 0a 20 20 20 20 20 20 66 6f 72 28 20 7a  os;.      for( z
17a0: 54 6f 6b 20 3d 20 73 74 72 74 6f 6b 5f 72 28 7a  Tok = strtok_r(z
17b0: 42 75 66 2c 20 22 2c 5c 22 22 2c 26 7a 50 6f 73  Buf, ",\"",&zPos
17c0: 29 3b 0a 20 20 20 20 20 20 20 20 20 20 20 7a 54  );.           zT
17d0: 6f 6b 20 26 26 20 73 74 72 63 61 73 65 63 6d 70  ok && strcasecmp
17e0: 28 7a 54 6f 6b 2c 7a 45 54 61 67 29 3b 0a 20 20  (zTok,zETag);.  
17f0: 20 20 20 20 20 20 20 20 20 7a 54 6f 6b 20 3d 20           zTok = 
1800: 20 73 74 72 74 6f 6b 5f 72 28 30 2c 20 22 2c 5c   strtok_r(0, ",\
1810: 22 22 2c 26 7a 50 6f 73 29 29 7b 7d 0a 20 20 20  "",&zPos)){}.   
1820: 20 20 20 66 72 65 65 28 7a 42 75 66 29 3b 0a 20     free(zBuf);. 
1830: 20 20 20 20 20 69 66 28 7a 54 6f 6b 29 20 72 65       if(zTok) re
1840: 74 75 72 6e 20 31 3b 0a 20 20 20 20 7d 0a 20 20  turn 1;.    }.  
1850: 7d 0a 20 20 0a 20 20 72 65 74 75 72 6e 20 30 3b  }.  .  return 0;
1860: 0a 7d 0a 23 65 6e 64 69 66 0a 0a 2f 2a 0a 2a 2a  .}.#endif../*.**
1870: 20 44 6f 20 61 20 6e 6f 72 6d 61 6c 20 48 54 54   Do a normal HTT
1880: 50 20 72 65 70 6c 79 0a 2a 2f 0a 76 6f 69 64 20  P reply.*/.void 
1890: 63 67 69 5f 72 65 70 6c 79 28 76 6f 69 64 29 7b  cgi_reply(void){
18a0: 0a 20 20 69 66 28 20 69 52 65 70 6c 79 53 74 61  .  if( iReplySta
18b0: 74 75 73 3c 3d 30 20 29 7b 0a 20 20 20 20 69 52  tus<=0 ){.    iR
18c0: 65 70 6c 79 53 74 61 74 75 73 20 3d 20 32 30 30  eplyStatus = 200
18d0: 3b 0a 20 20 20 20 7a 52 65 70 6c 79 53 74 61 74  ;.    zReplyStat
18e0: 75 73 20 3d 20 22 4f 4b 22 3b 0a 20 20 7d 0a 0a  us = "OK";.  }..
18f0: 23 69 66 20 30 0a 20 20 69 66 28 20 69 52 65 70  #if 0.  if( iRep
1900: 6c 79 53 74 61 74 75 73 3d 3d 32 30 30 20 26 26  lyStatus==200 &&
1910: 20 63 68 65 63 6b 5f 63 61 63 68 65 5f 63 6f 6e   check_cache_con
1920: 74 72 6f 6c 28 29 20 29 20 7b 0a 20 20 20 20 2f  trol() ) {.    /
1930: 2a 20 63 68 61 6e 67 65 20 74 68 65 20 73 74 61  * change the sta
1940: 74 75 73 20 74 6f 20 22 75 6e 63 68 61 6e 67 65  tus to "unchange
1950: 64 22 20 61 6e 64 20 77 65 20 63 61 6e 20 73 6b  d" and we can sk
1960: 69 70 20 73 65 6e 64 69 6e 67 20 74 68 65 0a 20  ip sending the. 
1970: 20 20 20 2a 2a 20 61 63 74 75 61 6c 20 72 65 73     ** actual res
1980: 70 6f 6e 73 65 20 62 6f 64 79 2e 20 4f 62 76 69  ponse body. Obvi
1990: 6f 75 73 6c 79 20 77 65 20 6f 6e 6c 79 20 64 6f  ously we only do
19a0: 20 74 68 69 73 20 77 68 65 6e 20 77 65 20 5f 68   this when we _h
19b0: 61 76 65 5f 20 61 0a 20 20 20 20 2a 2a 20 62 6f  ave_ a.    ** bo
19c0: 64 79 20 28 63 6f 64 65 20 32 30 30 29 2e 0a 20  dy (code 200).. 
19d0: 20 20 20 2a 2f 0a 20 20 20 20 69 52 65 70 6c 79     */.    iReply
19e0: 53 74 61 74 75 73 20 3d 20 33 30 34 3b 0a 20 20  Status = 304;.  
19f0: 20 20 7a 52 65 70 6c 79 53 74 61 74 75 73 20 3d    zReplyStatus =
1a00: 20 22 4e 6f 74 20 4d 6f 64 69 66 69 65 64 22 3b   "Not Modified";
1a10: 0a 20 20 7d 0a 23 65 6e 64 69 66 0a 0a 20 20 69  .  }.#endif..  i
1a20: 66 28 20 66 75 6c 6c 48 74 74 70 52 65 70 6c 79  f( fullHttpReply
1a30: 20 29 7b 0a 20 20 20 20 70 72 69 6e 74 66 28 22   ){.    printf("
1a40: 48 54 54 50 2f 31 2e 30 20 25 64 20 25 73 5c 72  HTTP/1.0 %d %s\r
1a50: 5c 6e 22 2c 20 69 52 65 70 6c 79 53 74 61 74 75  \n", iReplyStatu
1a60: 73 2c 20 7a 52 65 70 6c 79 53 74 61 74 75 73 29  s, zReplyStatus)
1a70: 3b 0a 20 20 20 20 70 72 69 6e 74 66 28 22 44 61  ;.    printf("Da
1a80: 74 65 3a 20 25 73 5c 72 5c 6e 22 2c 20 63 67 69  te: %s\r\n", cgi
1a90: 5f 72 66 63 38 32 32 5f 64 61 74 65 73 74 61 6d  _rfc822_datestam
1aa0: 70 28 74 69 6d 65 28 30 29 29 29 3b 0a 20 20 20  p(time(0)));.   
1ab0: 20 70 72 69 6e 74 66 28 22 43 6f 6e 6e 65 63 74   printf("Connect
1ac0: 69 6f 6e 3a 20 63 6c 6f 73 65 5c 72 5c 6e 22 29  ion: close\r\n")
1ad0: 3b 0a 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 70  ;.  }else{.    p
1ae0: 72 69 6e 74 66 28 22 53 74 61 74 75 73 3a 20 25  rintf("Status: %
1af0: 64 20 25 73 5c 72 5c 6e 22 2c 20 69 52 65 70 6c  d %s\r\n", iRepl
1b00: 79 53 74 61 74 75 73 2c 20 7a 52 65 70 6c 79 53  yStatus, zReplyS
1b10: 74 61 74 75 73 29 3b 0a 20 20 7d 0a 0a 20 20 69  tatus);.  }..  i
1b20: 66 28 20 62 6c 6f 62 5f 73 69 7a 65 28 26 65 78  f( blob_size(&ex
1b30: 74 72 61 48 65 61 64 65 72 29 3e 30 20 29 7b 0a  traHeader)>0 ){.
1b40: 20 20 20 20 70 72 69 6e 74 66 28 22 25 73 22 2c      printf("%s",
1b50: 20 62 6c 6f 62 5f 62 75 66 66 65 72 28 26 65 78   blob_buffer(&ex
1b60: 74 72 61 48 65 61 64 65 72 29 29 3b 0a 20 20 7d  traHeader));.  }
1b70: 0a 0a 20 20 69 66 28 20 67 2e 69 73 43 6f 6e 73  ..  if( g.isCons
1b80: 74 20 29 7b 0a 20 20 20 20 2f 2a 20 63 6f 6e 73  t ){.    /* cons
1b90: 74 61 6e 74 20 6d 65 61 6e 73 20 74 68 61 74 20  tant means that 
1ba0: 74 68 65 20 69 6e 70 75 74 20 55 52 4c 20 77 69  the input URL wi
1bb0: 6c 6c 20 5f 6e 65 76 65 72 5f 20 67 65 6e 65 72  ll _never_ gener
1bc0: 61 74 65 20 61 6e 79 74 68 69 6e 67 0a 20 20 20  ate anything.   
1bd0: 20 2a 2a 20 65 6c 73 65 2e 20 49 6e 20 74 68 65   ** else. In the
1be0: 20 63 61 73 65 20 6f 66 20 61 74 74 61 63 68 6d   case of attachm
1bf0: 65 6e 74 73 2c 20 74 68 65 20 63 6f 6e 74 65 6e  ents, the conten
1c00: 74 73 20 77 6f 6e 27 74 20 63 68 61 6e 67 65 20  ts won't change 
1c10: 62 65 63 61 75 73 65 0a 20 20 20 20 2a 2a 20 61  because.    ** a
1c20: 6e 20 61 74 74 65 6d 70 74 20 74 6f 20 63 68 61  n attempt to cha
1c30: 6e 67 65 20 74 68 65 6d 20 67 65 6e 65 72 61 74  nge them generat
1c40: 65 73 20 61 20 6e 65 77 20 61 74 74 61 63 68 6d  es a new attachm
1c50: 65 6e 74 20 6e 75 6d 62 65 72 2e 20 49 6e 20 74  ent number. In t
1c60: 68 65 0a 20 20 20 20 2a 2a 20 63 61 73 65 20 6f  he.    ** case o
1c70: 66 20 6d 6f 73 74 20 2f 67 65 74 66 69 6c 65 20  f most /getfile 
1c80: 63 61 6c 6c 73 20 66 6f 72 20 73 70 65 63 69 66  calls for specif
1c90: 69 63 20 76 65 72 73 69 6f 6e 73 2c 20 74 68 65  ic versions, the
1ca0: 20 6f 6e 6c 79 20 77 61 79 20 74 68 65 0a 20 20   only way the.  
1cb0: 20 20 2a 2a 20 63 6f 6e 74 65 6e 74 20 63 68 61    ** content cha
1cc0: 6e 67 65 73 20 69 73 20 69 66 20 73 6f 6d 65 6f  nges is if someo
1cd0: 6e 65 20 62 72 65 61 6b 73 20 74 68 65 20 53 43  ne breaks the SC
1ce0: 4d 2e 20 41 6e 64 20 69 66 20 74 68 61 74 20 68  M. And if that h
1cf0: 61 70 70 65 6e 73 2c 20 61 0a 20 20 20 20 2a 2a  appens, a.    **
1d00: 20 73 74 61 6c 65 20 63 61 63 68 65 20 69 73 20   stale cache is 
1d10: 74 68 65 20 6c 65 61 73 74 20 6f 66 20 74 68 65  the least of the
1d20: 20 70 72 6f 62 6c 65 6d 2e 20 53 6f 20 77 65 20   problem. So we 
1d30: 70 72 6f 76 69 64 65 20 61 6e 20 45 78 70 69 72  provide an Expir
1d40: 65 73 0a 20 20 20 20 2a 2a 20 68 65 61 64 65 72  es.    ** header
1d50: 20 73 65 74 20 74 6f 20 61 20 72 65 61 73 6f 6e   set to a reason
1d60: 61 62 6c 65 20 70 65 72 69 6f 64 20 28 64 65 66  able period (def
1d70: 61 75 6c 74 3a 20 6f 6e 65 20 77 65 65 6b 29 2e  ault: one week).
1d80: 0a 20 20 20 20 2a 2f 0a 20 20 20 20 2f 2a 74 69  .    */.    /*ti
1d90: 6d 65 5f 74 20 65 78 70 69 72 65 73 20 3d 20 74  me_t expires = t
1da0: 69 6d 65 28 30 29 20 2b 20 61 74 6f 69 28 64 62  ime(0) + atoi(db
1db0: 5f 63 6f 6e 66 69 67 28 22 63 6f 6e 73 74 61 6e  _config("constan
1dc0: 74 5f 65 78 70 69 72 65 73 22 2c 22 36 30 34 38  t_expires","6048
1dd0: 30 30 22 29 29 3b 2a 2f 0a 20 20 20 20 74 69 6d  00"));*/.    tim
1de0: 65 5f 74 20 65 78 70 69 72 65 73 20 3d 20 74 69  e_t expires = ti
1df0: 6d 65 28 30 29 20 2b 20 36 30 34 38 30 30 3b 0a  me(0) + 604800;.
1e00: 20 20 20 20 70 72 69 6e 74 66 28 20 22 45 78 70      printf( "Exp
1e10: 69 72 65 73 3a 20 25 73 5c 72 5c 6e 22 2c 20 63  ires: %s\r\n", c
1e20: 67 69 5f 72 66 63 38 32 32 5f 64 61 74 65 73 74  gi_rfc822_datest
1e30: 61 6d 70 28 65 78 70 69 72 65 73 29 29 3b 0a 20  amp(expires));. 
1e40: 20 7d 0a 0a 20 20 2f 2a 20 43 6f 6e 74 65 6e 74   }..  /* Content
1e50: 20 69 6e 74 65 6e 64 65 64 20 66 6f 72 20 6c 6f   intended for lo
1e60: 67 67 65 64 20 69 6e 20 75 73 65 72 73 20 73 68  gged in users sh
1e70: 6f 75 6c 64 20 6f 6e 6c 79 20 62 65 20 63 61 63  ould only be cac
1e80: 68 65 64 20 69 6e 0a 20 20 2a 2a 20 74 68 65 20  hed in.  ** the 
1e90: 62 72 6f 77 73 65 72 2c 20 6e 6f 74 20 73 6f 6d  browser, not som
1ea0: 65 20 73 68 61 72 65 64 20 6c 6f 63 61 74 69 6f  e shared locatio
1eb0: 6e 2e 0a 20 20 2a 2f 0a 20 20 70 72 69 6e 74 66  n..  */.  printf
1ec0: 28 22 43 61 63 68 65 2d 63 6f 6e 74 72 6f 6c 3a  ("Cache-control:
1ed0: 20 70 72 69 76 61 74 65 5c 72 5c 6e 22 29 3b 0a   private\r\n");.
1ee0: 0a 23 69 66 20 46 4f 53 53 49 4c 5f 49 31 38 4e  .#if FOSSIL_I18N
1ef0: 0a 20 20 70 72 69 6e 74 66 28 20 22 43 6f 6e 74  .  printf( "Cont
1f00: 65 6e 74 2d 54 79 70 65 3a 20 25 73 3b 20 63 68  ent-Type: %s; ch
1f10: 61 72 73 65 74 3d 25 73 5c 72 5c 6e 22 2c 20 7a  arset=%s\r\n", z
1f20: 43 6f 6e 74 65 6e 74 54 79 70 65 2c 20 6e 6c 5f  ContentType, nl_
1f30: 6c 61 6e 67 69 6e 66 6f 28 43 4f 44 45 53 45 54  langinfo(CODESET
1f40: 29 29 3b 0a 23 65 6c 73 65 0a 20 20 70 72 69 6e  ));.#else.  prin
1f50: 74 66 28 20 22 43 6f 6e 74 65 6e 74 2d 54 79 70  tf( "Content-Typ
1f60: 65 3a 20 25 73 3b 20 63 68 61 72 73 65 74 3d 49  e: %s; charset=I
1f70: 53 4f 2d 38 38 35 39 2d 31 5c 72 5c 6e 22 2c 20  SO-8859-1\r\n", 
1f80: 7a 43 6f 6e 74 65 6e 74 54 79 70 65 29 3b 0a 23  zContentType);.#
1f90: 65 6e 64 69 66 0a 20 20 69 66 28 20 73 74 72 63  endif.  if( strc
1fa0: 6d 70 28 7a 43 6f 6e 74 65 6e 74 54 79 70 65 2c  mp(zContentType,
1fb0: 22 61 70 70 6c 69 63 61 74 69 6f 6e 2f 78 2d 66  "application/x-f
1fc0: 6f 73 73 69 6c 22 29 3d 3d 30 20 29 7b 0a 20 20  ossil")==0 ){.  
1fd0: 20 20 62 6c 6f 62 5f 63 6f 6d 70 72 65 73 73 28    blob_compress(
1fe0: 26 63 67 69 43 6f 6e 74 65 6e 74 2c 20 26 63 67  &cgiContent, &cg
1ff0: 69 43 6f 6e 74 65 6e 74 29 3b 0a 20 20 7d 0a 0a  iContent);.  }..
2000: 20 20 69 66 28 20 69 52 65 70 6c 79 53 74 61 74    if( iReplyStat
2010: 75 73 20 21 3d 20 33 30 34 20 29 20 7b 0a 20 20  us != 304 ) {.  
2020: 20 20 70 72 69 6e 74 66 28 20 22 43 6f 6e 74 65    printf( "Conte
2030: 6e 74 2d 4c 65 6e 67 74 68 3a 20 25 64 5c 72 5c  nt-Length: %d\r\
2040: 6e 22 2c 20 62 6c 6f 62 5f 73 69 7a 65 28 26 63  n", blob_size(&c
2050: 67 69 43 6f 6e 74 65 6e 74 29 20 29 3b 0a 20 20  giContent) );.  
2060: 7d 0a 20 20 70 72 69 6e 74 66 28 22 5c 72 5c 6e  }.  printf("\r\n
2070: 22 29 3b 0a 20 20 69 66 28 20 62 6c 6f 62 5f 73  ");.  if( blob_s
2080: 69 7a 65 28 26 63 67 69 43 6f 6e 74 65 6e 74 29  ize(&cgiContent)
2090: 3e 30 20 26 26 20 69 52 65 70 6c 79 53 74 61 74  >0 && iReplyStat
20a0: 75 73 20 21 3d 20 33 30 34 20 29 7b 0a 20 20 20  us != 304 ){.   
20b0: 20 66 77 72 69 74 65 28 62 6c 6f 62 5f 62 75 66   fwrite(blob_buf
20c0: 66 65 72 28 26 63 67 69 43 6f 6e 74 65 6e 74 29  fer(&cgiContent)
20d0: 2c 20 31 2c 20 62 6c 6f 62 5f 73 69 7a 65 28 26  , 1, blob_size(&
20e0: 63 67 69 43 6f 6e 74 65 6e 74 29 2c 20 73 74 64  cgiContent), std
20f0: 6f 75 74 29 3b 0a 20 20 7d 0a 20 20 43 47 49 44  out);.  }.  CGID
2100: 45 42 55 47 28 28 22 44 4f 4e 45 5c 6e 22 29 29  EBUG(("DONE\n"))
2110: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 44 6f 20 61 20  ;.}../*.** Do a 
2120: 72 65 64 69 72 65 63 74 20 72 65 71 75 65 73 74  redirect request
2130: 20 74 6f 20 74 68 65 20 55 52 4c 20 67 69 76 65   to the URL give
2140: 6e 20 69 6e 20 74 68 65 20 61 72 67 75 6d 65 6e  n in the argumen
2150: 74 2e 0a 2a 2a 0a 2a 2a 20 54 68 65 20 55 52 4c  t..**.** The URL
2160: 20 6d 75 73 74 20 62 65 20 72 65 6c 61 74 69 76   must be relativ
2170: 65 20 74 6f 20 74 68 65 20 62 61 73 65 20 6f 66  e to the base of
2180: 20 74 68 65 20 66 6f 73 73 69 6c 20 73 65 72 76   the fossil serv
2190: 65 72 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f  er..*/.void cgi_
21a0: 72 65 64 69 72 65 63 74 28 63 6f 6e 73 74 20 63  redirect(const c
21b0: 68 61 72 20 2a 7a 55 52 4c 29 7b 0a 20 20 63 68  har *zURL){.  ch
21c0: 61 72 20 2a 7a 4c 6f 63 61 74 69 6f 6e 3b 0a 20  ar *zLocation;. 
21d0: 20 43 47 49 44 45 42 55 47 28 28 22 72 65 64 69   CGIDEBUG(("redi
21e0: 72 65 63 74 20 74 6f 20 25 73 5c 6e 22 2c 20 7a  rect to %s\n", z
21f0: 55 52 4c 29 29 3b 0a 20 20 69 66 28 20 73 74 72  URL));.  if( str
2200: 6e 63 6d 70 28 7a 55 52 4c 2c 22 68 74 74 70 3a  ncmp(zURL,"http:
2210: 22 2c 35 29 3d 3d 30 20 7c 7c 20 73 74 72 6e 63  ",5)==0 || strnc
2220: 6d 70 28 7a 55 52 4c 2c 22 68 74 74 70 73 3a 22  mp(zURL,"https:"
2230: 2c 36 29 3d 3d 30 20 7c 7c 20 2a 7a 55 52 4c 3d  ,6)==0 || *zURL=
2240: 3d 27 2f 27 20 29 7b 0a 20 20 20 20 7a 4c 6f 63  ='/' ){.    zLoc
2250: 61 74 69 6f 6e 20 3d 20 6d 70 72 69 6e 74 66 28  ation = mprintf(
2260: 22 4c 6f 63 61 74 69 6f 6e 3a 20 25 73 5c 72 5c  "Location: %s\r\
2270: 6e 22 2c 20 7a 55 52 4c 29 3b 0a 20 20 7d 65 6c  n", zURL);.  }el
2280: 73 65 7b 0a 20 20 20 20 7a 4c 6f 63 61 74 69 6f  se{.    zLocatio
2290: 6e 20 3d 20 6d 70 72 69 6e 74 66 28 22 4c 6f 63  n = mprintf("Loc
22a0: 61 74 69 6f 6e 3a 20 25 73 2f 25 73 5c 72 5c 6e  ation: %s/%s\r\n
22b0: 22 2c 20 67 2e 7a 42 61 73 65 55 52 4c 2c 20 7a  ", g.zBaseURL, z
22c0: 55 52 4c 29 3b 0a 20 20 7d 0a 20 20 63 67 69 5f  URL);.  }.  cgi_
22d0: 61 70 70 65 6e 64 5f 68 65 61 64 65 72 28 7a 4c  append_header(zL
22e0: 6f 63 61 74 69 6f 6e 29 3b 0a 20 20 63 67 69 5f  ocation);.  cgi_
22f0: 72 65 73 65 74 5f 63 6f 6e 74 65 6e 74 28 29 3b  reset_content();
2300: 0a 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 3c  .  cgi_printf("<
2310: 68 74 6d 6c 3e 5c 6e 3c 70 3e 52 65 64 69 72 65  html>\n<p>Redire
2320: 63 74 20 74 6f 20 25 68 3c 2f 70 3e 5c 6e 3c 2f  ct to %h</p>\n</
2330: 68 74 6d 6c 3e 5c 6e 22 2c 20 7a 55 52 4c 29 3b  html>\n", zURL);
2340: 0a 20 20 63 67 69 5f 73 65 74 5f 73 74 61 74 75  .  cgi_set_statu
2350: 73 28 33 30 32 2c 20 22 4d 6f 76 65 64 20 54 65  s(302, "Moved Te
2360: 6d 70 6f 72 61 72 69 6c 79 22 29 3b 0a 20 20 66  mporarily");.  f
2370: 72 65 65 28 7a 4c 6f 63 61 74 69 6f 6e 29 3b 0a  ree(zLocation);.
2380: 20 20 63 67 69 5f 72 65 70 6c 79 28 29 3b 0a 20    cgi_reply();. 
2390: 20 65 78 69 74 28 30 29 3b 0a 7d 0a 76 6f 69 64   exit(0);.}.void
23a0: 20 63 67 69 5f 72 65 64 69 72 65 63 74 66 28 63   cgi_redirectf(c
23b0: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 46 6f 72 6d  onst char *zForm
23c0: 61 74 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f 6c  at, ...){.  va_l
23d0: 69 73 74 20 61 70 3b 0a 20 20 76 61 5f 73 74 61  ist ap;.  va_sta
23e0: 72 74 28 61 70 2c 20 7a 46 6f 72 6d 61 74 29 3b  rt(ap, zFormat);
23f0: 0a 20 20 63 67 69 5f 72 65 64 69 72 65 63 74 28  .  cgi_redirect(
2400: 76 6d 70 72 69 6e 74 66 28 7a 46 6f 72 6d 61 74  vmprintf(zFormat
2410: 2c 20 61 70 29 29 3b 0a 20 20 76 61 5f 65 6e 64  , ap));.  va_end
2420: 28 61 70 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 49  (ap);.}../*.** I
2430: 6e 66 6f 72 6d 61 74 69 6f 6e 20 61 62 6f 75 74  nformation about
2440: 20 61 6c 6c 20 71 75 65 72 79 20 70 61 72 61 6d   all query param
2450: 65 74 65 72 73 20 61 6e 64 20 63 6f 6f 6b 69 65  eters and cookie
2460: 73 20 61 72 65 20 73 74 6f 72 65 64 0a 2a 2a 20  s are stored.** 
2470: 69 6e 20 74 68 65 73 65 20 76 61 72 69 61 62 6c  in these variabl
2480: 65 73 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e  es..*/.static in
2490: 74 20 6e 41 6c 6c 6f 63 51 50 20 3d 20 30 3b 20  t nAllocQP = 0; 
24a0: 2f 2a 20 53 70 61 63 65 20 61 6c 6c 6f 63 61 74  /* Space allocat
24b0: 65 64 20 66 6f 72 20 61 50 61 72 61 6d 51 50 5b  ed for aParamQP[
24c0: 5d 20 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20  ] */.static int 
24d0: 6e 55 73 65 64 51 50 20 3d 20 30 3b 20 20 2f 2a  nUsedQP = 0;  /*
24e0: 20 53 70 61 63 65 20 61 63 74 75 61 6c 6c 79 20   Space actually 
24f0: 75 73 65 64 20 69 6e 20 61 50 61 72 61 6d 51 50  used in aParamQP
2500: 5b 5d 20 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74  [] */.static int
2510: 20 73 6f 72 74 51 50 20 3d 20 30 3b 20 20 20 2f   sortQP = 0;   /
2520: 2a 20 54 72 75 65 20 69 66 20 61 50 61 72 61 6d  * True if aParam
2530: 51 50 5b 5d 20 6e 65 65 64 73 20 73 6f 72 74 69  QP[] needs sorti
2540: 6e 67 20 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74  ng */.static int
2550: 20 73 65 71 51 50 20 3d 20 30 3b 20 20 20 20 2f   seqQP = 0;    /
2560: 2a 20 53 65 71 75 65 6e 63 65 20 6e 75 6d 62 65  * Sequence numbe
2570: 72 73 20 2a 2f 0a 73 74 61 74 69 63 20 73 74 72  rs */.static str
2580: 75 63 74 20 51 50 61 72 61 6d 20 7b 20 20 20 2f  uct QParam {   /
2590: 2a 20 4f 6e 65 20 65 6e 74 72 79 20 66 6f 72 20  * One entry for 
25a0: 65 61 63 68 20 71 75 65 72 79 20 70 61 72 61 6d  each query param
25b0: 65 74 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 2a  eter or cookie *
25c0: 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a  /.  const char *
25d0: 7a 4e 61 6d 65 3b 20 20 20 20 20 20 20 20 2f 2a  zName;        /*
25e0: 20 50 61 72 61 6d 65 74 65 72 20 6f 72 20 63 6f   Parameter or co
25f0: 6f 6b 69 65 20 6e 61 6d 65 20 2a 2f 0a 20 20 63  okie name */.  c
2600: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 56 61 6c 75  onst char *zValu
2610: 65 3b 20 20 20 20 20 20 20 2f 2a 20 56 61 6c 75  e;       /* Valu
2620: 65 20 6f 66 20 74 68 65 20 71 75 65 72 79 20 70  e of the query p
2630: 61 72 61 6d 65 74 65 72 20 6f 72 20 63 6f 6f 6b  arameter or cook
2640: 69 65 20 2a 2f 0a 20 20 69 6e 74 20 73 65 71 3b  ie */.  int seq;
2650: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
2660: 20 20 2f 2a 20 4f 72 64 65 72 20 6f 66 20 69 6e    /* Order of in
2670: 73 65 72 74 69 6f 6e 20 2a 2f 0a 7d 20 2a 61 50  sertion */.} *aP
2680: 61 72 61 6d 51 50 3b 20 20 20 20 20 20 20 20 20  aramQP;         
2690: 20 20 20 20 2f 2a 20 41 6e 20 61 72 72 61 79 20      /* An array 
26a0: 6f 66 20 61 6c 6c 20 70 61 72 61 6d 65 74 65 72  of all parameter
26b0: 73 20 61 6e 64 20 63 6f 6f 6b 69 65 73 20 2a 2f  s and cookies */
26c0: 0a 0a 2f 2a 0a 2a 2a 20 41 64 64 20 61 6e 6f 74  ../*.** Add anot
26d0: 68 65 72 20 71 75 65 72 79 20 70 61 72 61 6d 65  her query parame
26e0: 74 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 74 6f  ter or cookie to
26f0: 20 74 68 65 20 70 61 72 61 6d 65 74 65 72 20 73   the parameter s
2700: 65 74 2e 0a 2a 2a 20 7a 4e 61 6d 65 20 69 73 20  et..** zName is 
2710: 74 68 65 20 6e 61 6d 65 20 6f 66 20 74 68 65 20  the name of the 
2720: 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 20  query parameter 
2730: 6f 72 20 63 6f 6f 6b 69 65 20 61 6e 64 20 7a 56  or cookie and zV
2740: 61 6c 75 65 0a 2a 2a 20 69 73 20 69 74 73 20 66  alue.** is its f
2750: 75 6c 6c 79 20 64 65 63 6f 64 65 64 20 76 61 6c  ully decoded val
2760: 75 65 2e 0a 2a 2a 0a 2a 2a 20 7a 4e 61 6d 65 20  ue..**.** zName 
2770: 61 6e 64 20 7a 56 61 6c 75 65 20 61 72 65 20 6e  and zValue are n
2780: 6f 74 20 63 6f 70 69 65 64 20 61 6e 64 20 6d 75  ot copied and mu
2790: 73 74 20 6e 6f 74 20 63 68 61 6e 67 65 20 6f 72  st not change or
27a0: 20 62 65 0a 2a 2a 20 64 65 61 6c 6c 6f 63 61 74   be.** deallocat
27b0: 65 64 20 61 66 74 65 72 20 74 68 69 73 20 72 6f  ed after this ro
27c0: 75 74 69 6e 65 20 72 65 74 75 72 6e 73 2e 0a 2a  utine returns..*
27d0: 2f 0a 76 6f 69 64 20 63 67 69 5f 73 65 74 5f 70  /.void cgi_set_p
27e0: 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79 28  arameter_nocopy(
27f0: 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d  const char *zNam
2800: 65 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a  e, const char *z
2810: 56 61 6c 75 65 29 7b 0a 20 20 69 66 28 20 6e 41  Value){.  if( nA
2820: 6c 6c 6f 63 51 50 3c 3d 6e 55 73 65 64 51 50 20  llocQP<=nUsedQP 
2830: 29 7b 0a 20 20 20 20 6e 41 6c 6c 6f 63 51 50 20  ){.    nAllocQP 
2840: 3d 20 6e 41 6c 6c 6f 63 51 50 2a 32 20 2b 20 31  = nAllocQP*2 + 1
2850: 30 3b 0a 20 20 20 20 61 50 61 72 61 6d 51 50 20  0;.    aParamQP 
2860: 3d 20 72 65 61 6c 6c 6f 63 28 20 61 50 61 72 61  = realloc( aPara
2870: 6d 51 50 2c 20 6e 41 6c 6c 6f 63 51 50 2a 73 69  mQP, nAllocQP*si
2880: 7a 65 6f 66 28 61 50 61 72 61 6d 51 50 5b 30 5d  zeof(aParamQP[0]
2890: 29 20 29 3b 0a 20 20 20 20 69 66 28 20 61 50 61  ) );.    if( aPa
28a0: 72 61 6d 51 50 3d 3d 30 20 29 20 65 78 69 74 28  ramQP==0 ) exit(
28b0: 31 29 3b 0a 20 20 7d 0a 20 20 61 50 61 72 61 6d  1);.  }.  aParam
28c0: 51 50 5b 6e 55 73 65 64 51 50 5d 2e 7a 4e 61 6d  QP[nUsedQP].zNam
28d0: 65 20 3d 20 7a 4e 61 6d 65 3b 0a 20 20 61 50 61  e = zName;.  aPa
28e0: 72 61 6d 51 50 5b 6e 55 73 65 64 51 50 5d 2e 7a  ramQP[nUsedQP].z
28f0: 56 61 6c 75 65 20 3d 20 7a 56 61 6c 75 65 3b 0a  Value = zValue;.
2900: 20 20 61 50 61 72 61 6d 51 50 5b 6e 55 73 65 64    aParamQP[nUsed
2910: 51 50 5d 2e 73 65 71 20 3d 20 73 65 71 51 50 2b  QP].seq = seqQP+
2920: 2b 3b 0a 20 20 6e 55 73 65 64 51 50 2b 2b 3b 0a  +;.  nUsedQP++;.
2930: 20 20 73 6f 72 74 51 50 20 3d 20 31 3b 0a 7d 0a    sortQP = 1;.}.
2940: 0a 2f 2a 0a 2a 2a 20 41 64 64 20 61 6e 6f 74 68  ./*.** Add anoth
2950: 65 72 20 71 75 65 72 79 20 70 61 72 61 6d 65 74  er query paramet
2960: 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 74 6f 20  er or cookie to 
2970: 74 68 65 20 70 61 72 61 6d 65 74 65 72 20 73 65  the parameter se
2980: 74 2e 0a 2a 2a 20 7a 4e 61 6d 65 20 69 73 20 74  t..** zName is t
2990: 68 65 20 6e 61 6d 65 20 6f 66 20 74 68 65 20 71  he name of the q
29a0: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 20 6f  uery parameter o
29b0: 72 20 63 6f 6f 6b 69 65 20 61 6e 64 20 7a 56 61  r cookie and zVa
29c0: 6c 75 65 0a 2a 2a 20 69 73 20 69 74 73 20 66 75  lue.** is its fu
29d0: 6c 6c 79 20 64 65 63 6f 64 65 64 20 76 61 6c 75  lly decoded valu
29e0: 65 2e 0a 2a 2a 0a 2a 2a 20 43 6f 70 69 65 73 20  e..**.** Copies 
29f0: 61 72 65 20 6d 61 64 65 20 6f 66 20 62 6f 74 68  are made of both
2a00: 20 74 68 65 20 7a 4e 61 6d 65 20 61 6e 64 20 7a   the zName and z
2a10: 56 61 6c 75 65 20 70 61 72 61 6d 65 74 65 72 73  Value parameters
2a20: 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 73 65  ..*/.void cgi_se
2a30: 74 5f 70 61 72 61 6d 65 74 65 72 28 63 6f 6e 73  t_parameter(cons
2a40: 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65 2c 20 63  t char *zName, c
2a50: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 56 61 6c 75  onst char *zValu
2a60: 65 29 7b 0a 20 20 63 67 69 5f 73 65 74 5f 70 61  e){.  cgi_set_pa
2a70: 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79 28 6d  rameter_nocopy(m
2a80: 70 72 69 6e 74 66 28 22 25 73 22 2c 7a 4e 61 6d  printf("%s",zNam
2a90: 65 29 2c 20 6d 70 72 69 6e 74 66 28 22 25 73 22  e), mprintf("%s"
2aa0: 2c 7a 56 61 6c 75 65 29 29 3b 0a 7d 0a 0a 2f 2a  ,zValue));.}../*
2ab0: 0a 2a 2a 20 52 65 70 6c 61 63 65 20 61 20 70 61  .** Replace a pa
2ac0: 72 61 6d 65 74 65 72 20 77 69 74 68 20 61 20 6e  rameter with a n
2ad0: 65 77 20 76 61 6c 75 65 2e 0a 2a 2f 0a 76 6f 69  ew value..*/.voi
2ae0: 64 20 63 67 69 5f 72 65 70 6c 61 63 65 5f 70 61  d cgi_replace_pa
2af0: 72 61 6d 65 74 65 72 28 63 6f 6e 73 74 20 63 68  rameter(const ch
2b00: 61 72 20 2a 7a 4e 61 6d 65 2c 20 63 6f 6e 73 74  ar *zName, const
2b10: 20 63 68 61 72 20 2a 7a 56 61 6c 75 65 29 7b 0a   char *zValue){.
2b20: 20 20 69 6e 74 20 69 3b 0a 20 20 66 6f 72 28 69    int i;.  for(i
2b30: 3d 30 3b 20 69 3c 6e 55 73 65 64 51 50 3b 20 69  =0; i<nUsedQP; i
2b40: 2b 2b 29 7b 0a 20 20 20 20 69 66 28 20 73 74 72  ++){.    if( str
2b50: 63 6d 70 28 61 50 61 72 61 6d 51 50 5b 69 5d 2e  cmp(aParamQP[i].
2b60: 7a 4e 61 6d 65 2c 7a 4e 61 6d 65 29 3d 3d 30 20  zName,zName)==0 
2b70: 29 7b 0a 20 20 20 20 20 20 61 50 61 72 61 6d 51  ){.      aParamQ
2b80: 50 5b 69 5d 2e 7a 56 61 6c 75 65 20 3d 20 7a 56  P[i].zValue = zV
2b90: 61 6c 75 65 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a  alue;.    }.  }.
2ba0: 7d 0a 0a 2f 2a 0a 2a 2a 20 41 64 64 20 61 20 71  }../*.** Add a q
2bb0: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 2e 20  uery parameter. 
2bc0: 20 54 68 65 20 7a 4e 61 6d 65 20 70 6f 72 74 69   The zName porti
2bd0: 6f 6e 20 69 73 20 66 69 78 65 64 20 62 75 74 20  on is fixed but 
2be0: 61 20 63 6f 70 79 0a 2a 2a 20 6d 75 73 74 20 62  a copy.** must b
2bf0: 65 20 6d 61 64 65 20 6f 66 20 7a 56 61 6c 75 65  e made of zValue
2c00: 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 73 65  ..*/.void cgi_se
2c10: 74 65 6e 76 28 63 6f 6e 73 74 20 63 68 61 72 20  tenv(const char 
2c20: 2a 7a 4e 61 6d 65 2c 20 63 6f 6e 73 74 20 63 68  *zName, const ch
2c30: 61 72 20 2a 7a 56 61 6c 75 65 29 7b 0a 20 20 63  ar *zValue){.  c
2c40: 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 74 65 72  gi_set_parameter
2c50: 5f 6e 6f 63 6f 70 79 28 7a 4e 61 6d 65 2c 20 6d  _nocopy(zName, m
2c60: 70 72 69 6e 74 66 28 22 25 73 22 2c 7a 56 61 6c  printf("%s",zVal
2c70: 75 65 29 29 3b 0a 7d 0a 20 0a 0a 2f 2a 0a 2a 2a  ue));.}. ../*.**
2c80: 20 41 64 64 20 61 20 6c 69 73 74 20 6f 66 20 71   Add a list of q
2c90: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 73 20  uery parameters 
2ca0: 6f 72 20 63 6f 6f 6b 69 65 73 20 74 6f 20 74 68  or cookies to th
2cb0: 65 20 70 61 72 61 6d 65 74 65 72 20 73 65 74 2e  e parameter set.
2cc0: 0a 2a 2a 0a 2a 2a 20 45 61 63 68 20 70 61 72 61  .**.** Each para
2cd0: 6d 65 74 65 72 20 69 73 20 6f 66 20 74 68 65 20  meter is of the 
2ce0: 66 6f 72 6d 20 4e 41 4d 45 3d 56 41 4c 55 45 2e  form NAME=VALUE.
2cf0: 20 20 42 6f 74 68 20 74 68 65 20 4e 41 4d 45 20    Both the NAME 
2d00: 61 6e 64 20 74 68 65 0a 2a 2a 20 56 41 4c 55 45  and the.** VALUE
2d10: 20 6d 61 79 20 62 65 20 75 72 6c 2d 65 6e 63 6f   may be url-enco
2d20: 64 65 64 20 28 22 2b 22 20 66 6f 72 20 73 70 61  ded ("+" for spa
2d30: 63 65 2c 20 22 25 48 48 22 20 66 6f 72 20 6f 74  ce, "%HH" for ot
2d40: 68 65 72 20 73 70 65 63 69 61 6c 0a 2a 2a 20 63  her special.** c
2d50: 68 61 72 61 63 74 65 72 73 29 2e 20 20 42 75 74  haracters).  But
2d60: 20 74 68 69 73 20 72 6f 75 74 69 6e 65 20 61 73   this routine as
2d70: 73 75 6d 65 73 20 74 68 61 74 20 4e 41 4d 45 20  sumes that NAME 
2d80: 63 6f 6e 74 61 69 6e 73 20 6e 6f 0a 2a 2a 20 73  contains no.** s
2d90: 70 65 63 69 61 6c 20 63 68 61 72 61 63 74 65 72  pecial character
2da0: 20 61 6e 64 20 74 68 65 72 65 66 6f 72 65 20 64   and therefore d
2db0: 6f 65 73 20 6e 6f 74 20 64 65 63 6f 64 65 20 69  oes not decode i
2dc0: 74 2e 0a 2a 2a 0a 2a 2a 20 49 66 20 4e 41 4d 45  t..**.** If NAME
2dd0: 20 62 65 67 69 6e 73 20 77 69 74 68 20 61 6e 6f   begins with ano
2de0: 74 68 65 72 20 6f 74 68 65 72 20 74 68 61 6e 20  ther other than 
2df0: 61 20 6c 6f 77 65 72 2d 63 61 73 65 20 6c 65 74  a lower-case let
2e00: 74 65 72 20 74 68 65 6e 0a 2a 2a 20 74 68 65 20  ter then.** the 
2e10: 65 6e 74 69 72 65 20 4e 41 4d 45 3d 56 41 4c 55  entire NAME=VALU
2e20: 45 20 74 65 72 6d 20 69 73 20 69 67 6e 6f 72 65  E term is ignore
2e30: 64 2e 20 20 48 65 6e 63 65 3a 0a 2a 2a 0a 2a 2a  d.  Hence:.**.**
2e40: 20 20 20 20 20 20 2a 20 20 63 6f 6f 6b 69 65 73        *  cookies
2e50: 20 61 6e 64 20 71 75 65 72 79 20 70 61 72 61 6d   and query param
2e60: 65 74 65 72 73 20 74 68 61 74 20 68 61 76 65 20  eters that have 
2e70: 75 70 70 65 72 63 61 73 65 20 6e 61 6d 65 73 0a  uppercase names.
2e80: 2a 2a 20 20 20 20 20 20 20 20 20 61 72 65 20 69  **         are i
2e90: 67 6e 6f 72 65 64 2e 0a 2a 2a 0a 2a 2a 20 20 20  gnored..**.**   
2ea0: 20 20 20 2a 20 20 69 74 20 69 73 20 69 6d 70 6f     *  it is impo
2eb0: 73 73 69 62 6c 65 20 66 6f 72 20 61 20 63 6f 6f  ssible for a coo
2ec0: 6b 69 65 20 6f 72 20 71 75 65 72 79 20 70 61 72  kie or query par
2ed0: 61 6d 65 74 65 72 20 74 6f 0a 2a 2a 20 20 20 20  ameter to.**    
2ee0: 20 20 20 20 20 6f 76 65 72 72 69 64 65 20 74 68       override th
2ef0: 65 20 76 61 6c 75 65 20 6f 66 20 61 6e 20 65 6e  e value of an en
2f00: 76 69 72 6f 6e 6d 65 6e 74 20 76 61 72 69 61 62  vironment variab
2f10: 6c 65 20 73 69 6e 63 65 0a 2a 2a 20 20 20 20 20  le since.**     
2f20: 20 20 20 20 65 6e 76 69 72 6f 6e 6d 65 6e 74 20      environment 
2f30: 76 61 72 69 61 62 6c 65 73 20 61 6c 77 61 79 73  variables always
2f40: 20 68 61 76 65 20 75 70 70 65 72 63 61 73 65 20   have uppercase 
2f50: 6e 61 6d 65 73 2e 0a 2a 2a 0a 2a 2a 20 50 61 72  names..**.** Par
2f60: 61 6d 65 74 65 72 73 20 61 72 65 20 73 65 70 61  ameters are sepa
2f70: 72 61 74 65 64 20 62 79 20 74 68 65 20 22 74 65  rated by the "te
2f80: 72 6d 69 6e 61 74 6f 72 22 20 63 68 61 72 61 63  rminator" charac
2f90: 74 65 72 2e 20 20 57 68 69 74 65 73 70 61 63 65  ter.  Whitespace
2fa0: 0a 2a 2a 20 62 65 66 6f 72 65 20 74 68 65 20 4e  .** before the N
2fb0: 41 4d 45 20 69 73 20 69 67 6e 6f 72 65 64 2e 0a  AME is ignored..
2fc0: 2a 2a 0a 2a 2a 20 54 68 65 20 69 6e 70 75 74 20  **.** The input 
2fd0: 73 74 72 69 6e 67 20 22 7a 22 20 69 73 20 6d 6f  string "z" is mo
2fe0: 64 69 66 69 65 64 20 62 75 74 20 6e 6f 20 63 6f  dified but no co
2ff0: 70 69 65 73 20 69 73 20 6d 61 64 65 2e 20 20 22  pies is made.  "
3000: 7a 22 0a 2a 2a 20 73 68 6f 75 6c 64 20 6e 6f 74  z".** should not
3010: 20 62 65 20 64 65 61 6c 6c 6f 63 61 74 65 64 20   be deallocated 
3020: 6f 72 20 63 68 61 6e 67 65 64 20 61 67 61 69 6e  or changed again
3030: 20 61 66 74 65 72 20 74 68 69 73 20 72 6f 75 74   after this rout
3040: 69 6e 65 0a 2a 2a 20 72 65 74 75 72 6e 73 20 6f  ine.** returns o
3050: 72 20 69 74 20 77 69 6c 6c 20 63 6f 72 72 75 70  r it will corrup
3060: 74 20 74 68 65 20 70 61 72 61 6d 65 74 65 72 20  t the parameter 
3070: 74 61 62 6c 65 2e 0a 2a 2f 0a 73 74 61 74 69 63  table..*/.static
3080: 20 76 6f 69 64 20 61 64 64 5f 70 61 72 61 6d 5f   void add_param_
3090: 6c 69 73 74 28 63 68 61 72 20 2a 7a 2c 20 69 6e  list(char *z, in
30a0: 74 20 74 65 72 6d 69 6e 61 74 6f 72 29 7b 0a 20  t terminator){. 
30b0: 20 77 68 69 6c 65 28 20 2a 7a 20 29 7b 0a 20 20   while( *z ){.  
30c0: 20 20 63 68 61 72 20 2a 7a 4e 61 6d 65 3b 0a 20    char *zName;. 
30d0: 20 20 20 63 68 61 72 20 2a 7a 56 61 6c 75 65 3b     char *zValue;
30e0: 0a 20 20 20 20 77 68 69 6c 65 28 20 69 73 73 70  .    while( issp
30f0: 61 63 65 28 2a 7a 29 20 29 7b 20 7a 2b 2b 3b 20  ace(*z) ){ z++; 
3100: 7d 0a 20 20 20 20 7a 4e 61 6d 65 20 3d 20 7a 3b  }.    zName = z;
3110: 0a 20 20 20 20 77 68 69 6c 65 28 20 2a 7a 20 26  .    while( *z &
3120: 26 20 2a 7a 21 3d 27 3d 27 20 26 26 20 2a 7a 21  & *z!='=' && *z!
3130: 3d 74 65 72 6d 69 6e 61 74 6f 72 20 29 7b 20 7a  =terminator ){ z
3140: 2b 2b 3b 20 7d 0a 20 20 20 20 69 66 28 20 2a 7a  ++; }.    if( *z
3150: 3d 3d 27 3d 27 20 29 7b 0a 20 20 20 20 20 20 2a  =='=' ){.      *
3160: 7a 20 3d 20 30 3b 0a 20 20 20 20 20 20 7a 2b 2b  z = 0;.      z++
3170: 3b 0a 20 20 20 20 20 20 7a 56 61 6c 75 65 20 3d  ;.      zValue =
3180: 20 7a 3b 0a 20 20 20 20 20 20 77 68 69 6c 65 28   z;.      while(
3190: 20 2a 7a 20 26 26 20 2a 7a 21 3d 74 65 72 6d 69   *z && *z!=termi
31a0: 6e 61 74 6f 72 20 29 7b 20 7a 2b 2b 3b 20 7d 0a  nator ){ z++; }.
31b0: 20 20 20 20 20 20 69 66 28 20 2a 7a 20 29 7b 0a        if( *z ){.
31c0: 20 20 20 20 20 20 20 20 2a 7a 20 3d 20 30 3b 0a          *z = 0;.
31d0: 20 20 20 20 20 20 20 20 7a 2b 2b 3b 0a 20 20 20          z++;.   
31e0: 20 20 20 7d 0a 20 20 20 20 20 20 64 65 68 74 74     }.      dehtt
31f0: 70 69 7a 65 28 7a 56 61 6c 75 65 29 3b 0a 20 20  pize(zValue);.  
3200: 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20 20 69    }else{.      i
3210: 66 28 20 2a 7a 20 29 7b 20 2a 7a 2b 2b 20 3d 20  f( *z ){ *z++ = 
3220: 30 3b 20 7d 0a 20 20 20 20 20 20 7a 56 61 6c 75  0; }.      zValu
3230: 65 20 3d 20 22 22 3b 0a 20 20 20 20 7d 0a 20 20  e = "";.    }.  
3240: 20 20 69 66 28 20 69 73 6c 6f 77 65 72 28 7a 4e    if( islower(zN
3250: 61 6d 65 5b 30 5d 29 20 29 7b 0a 20 20 20 20 20  ame[0]) ){.     
3260: 20 63 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 74   cgi_set_paramet
3270: 65 72 5f 6e 6f 63 6f 70 79 28 7a 4e 61 6d 65 2c  er_nocopy(zName,
3280: 20 7a 56 61 6c 75 65 29 3b 0a 20 20 20 20 7d 0a   zValue);.    }.
3290: 20 20 7d 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 2a 70 7a    }.}../*.** *pz
32a0: 20 69 73 20 61 20 73 74 72 69 6e 67 20 74 68 61   is a string tha
32b0: 74 20 63 6f 6e 73 69 73 74 73 20 6f 66 20 6d 75  t consists of mu
32c0: 6c 74 69 70 6c 65 20 6c 69 6e 65 73 20 6f 66 20  ltiple lines of 
32d0: 74 65 78 74 2e 20 20 54 68 69 73 0a 2a 2a 20 72  text.  This.** r
32e0: 6f 75 74 69 6e 65 20 66 69 6e 64 73 20 74 68 65  outine finds the
32f0: 20 65 6e 64 20 6f 66 20 74 68 65 20 63 75 72 72   end of the curr
3300: 65 6e 74 20 6c 69 6e 65 20 6f 66 20 74 65 78 74  ent line of text
3310: 20 61 6e 64 20 63 6f 6e 76 65 72 74 73 0a 2a 2a   and converts.**
3320: 20 74 68 65 20 22 5c 6e 22 20 6f 72 20 22 5c 72   the "\n" or "\r
3330: 5c 6e 22 20 74 68 61 74 20 65 6e 64 73 20 74 68  \n" that ends th
3340: 61 74 20 6c 69 6e 65 20 69 6e 74 6f 20 61 20 22  at line into a "
3350: 5c 30 30 30 22 2e 20 20 49 74 20 74 68 65 6e 0a  \000".  It then.
3360: 2a 2a 20 61 64 76 61 6e 63 65 73 20 2a 70 7a 20  ** advances *pz 
3370: 74 6f 20 74 68 65 20 62 65 67 69 6e 6e 69 6e 67  to the beginning
3380: 20 6f 66 20 74 68 65 20 6e 65 78 74 20 6c 69 6e   of the next lin
3390: 65 20 61 6e 64 20 72 65 74 75 72 6e 73 20 74 68  e and returns th
33a0: 65 0a 2a 2a 20 70 72 65 76 69 6f 75 73 20 76 61  e.** previous va
33b0: 6c 75 65 20 6f 66 20 2a 70 7a 20 28 77 68 69 63  lue of *pz (whic
33c0: 68 20 69 73 20 74 68 65 20 73 74 61 72 74 20 6f  h is the start o
33d0: 66 20 74 68 65 20 63 75 72 72 65 6e 74 20 6c 69  f the current li
33e0: 6e 65 2e 29 0a 2a 2f 0a 73 74 61 74 69 63 20 63  ne.).*/.static c
33f0: 68 61 72 20 2a 67 65 74 5f 6c 69 6e 65 5f 66 72  har *get_line_fr
3400: 6f 6d 5f 73 74 72 69 6e 67 28 63 68 61 72 20 2a  om_string(char *
3410: 2a 70 7a 2c 20 69 6e 74 20 2a 70 4c 65 6e 29 7b  *pz, int *pLen){
3420: 0a 20 20 63 68 61 72 20 2a 7a 20 3d 20 2a 70 7a  .  char *z = *pz
3430: 3b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 69 66 28  ;.  int i;.  if(
3440: 20 7a 5b 30 5d 3d 3d 30 20 29 20 72 65 74 75 72   z[0]==0 ) retur
3450: 6e 20 30 3b 0a 20 20 66 6f 72 28 69 3d 30 3b 20  n 0;.  for(i=0; 
3460: 7a 5b 69 5d 3b 20 69 2b 2b 29 7b 0a 20 20 20 20  z[i]; i++){.    
3470: 69 66 28 20 7a 5b 69 5d 3d 3d 27 5c 6e 27 20 29  if( z[i]=='\n' )
3480: 7b 0a 20 20 20 20 20 20 69 66 28 20 69 3e 30 20  {.      if( i>0 
3490: 26 26 20 7a 5b 69 2d 31 5d 3d 3d 27 5c 72 27 20  && z[i-1]=='\r' 
34a0: 29 7b 0a 20 20 20 20 20 20 20 20 7a 5b 69 2d 31  ){.        z[i-1
34b0: 5d 20 3d 20 30 3b 0a 20 20 20 20 20 20 7d 65 6c  ] = 0;.      }el
34c0: 73 65 7b 0a 20 20 20 20 20 20 20 20 7a 5b 69 5d  se{.        z[i]
34d0: 20 3d 20 30 3b 0a 20 20 20 20 20 20 7d 0a 20 20   = 0;.      }.  
34e0: 20 20 20 20 69 2b 2b 3b 0a 20 20 20 20 20 20 62      i++;.      b
34f0: 72 65 61 6b 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a  reak;.    }.  }.
3500: 20 20 2a 70 7a 20 3d 20 26 7a 5b 69 5d 3b 0a 20    *pz = &z[i];. 
3510: 20 2a 70 4c 65 6e 20 2d 3d 20 69 3b 0a 20 20 72   *pLen -= i;.  r
3520: 65 74 75 72 6e 20 7a 3b 0a 7d 0a 0a 2f 2a 0a 2a  eturn z;.}../*.*
3530: 2a 20 54 68 65 20 69 6e 70 75 74 20 2a 70 7a 20  * The input *pz 
3540: 70 6f 69 6e 74 73 20 74 6f 20 63 6f 6e 74 65 6e  points to conten
3550: 74 20 74 68 61 74 20 69 73 20 74 65 72 6d 69 6e  t that is termin
3560: 61 74 65 64 20 62 79 20 61 20 22 5c 72 5c 6e 22  ated by a "\r\n"
3570: 0a 2a 2a 20 66 6f 6c 6c 6f 77 65 64 20 62 79 20  .** followed by 
3580: 74 68 65 20 62 6f 75 6e 64 72 79 20 6d 61 72 6b  the boundry mark
3590: 65 72 20 7a 42 6f 75 6e 64 72 79 2e 20 20 41 6e  er zBoundry.  An
35a0: 20 65 78 74 72 61 20 22 2d 2d 22 20 6d 61 79 20   extra "--" may 
35b0: 6f 72 0a 2a 2a 20 6d 61 79 20 6e 6f 74 20 62 65  or.** may not be
35c0: 20 61 70 70 65 6e 64 65 64 20 74 6f 20 74 68 65   appended to the
35d0: 20 62 6f 75 6e 64 72 79 20 6d 61 72 6b 65 72 2e   boundry marker.
35e0: 20 20 54 68 65 72 65 20 61 72 65 20 2a 70 4c 65    There are *pLe
35f0: 6e 20 63 68 61 72 61 63 74 65 72 73 0a 2a 2a 20  n characters.** 
3600: 69 6e 20 2a 70 7a 2e 0a 2a 2a 0a 2a 2a 20 54 68  in *pz..**.** Th
3610: 69 73 20 72 6f 75 74 69 6e 65 20 61 64 64 73 20  is routine adds 
3620: 61 20 22 5c 30 30 30 22 20 74 6f 20 74 68 65 20  a "\000" to the 
3630: 65 6e 64 20 6f 66 20 74 68 65 20 63 6f 6e 74 65  end of the conte
3640: 6e 74 20 28 6f 76 65 72 77 72 69 74 69 6e 67 0a  nt (overwriting.
3650: 2a 2a 20 74 68 65 20 22 5c 72 5c 6e 22 29 20 61  ** the "\r\n") a
3660: 6e 64 20 72 65 74 75 72 6e 73 20 61 20 70 6f 69  nd returns a poi
3670: 6e 74 65 72 20 74 6f 20 74 68 65 20 63 6f 6e 74  nter to the cont
3680: 65 6e 74 2e 20 20 54 68 65 20 2a 70 7a 20 69 6e  ent.  The *pz in
3690: 70 75 74 0a 2a 2a 20 69 73 20 61 64 6a 75 73 74  put.** is adjust
36a0: 65 64 20 74 6f 20 70 6f 69 6e 74 20 74 6f 20 74  ed to point to t
36b0: 68 65 20 66 69 72 73 74 20 6c 69 6e 65 20 66 6f  he first line fo
36c0: 6c 6c 6f 77 69 6e 67 20 74 68 65 20 62 6f 75 6e  llowing the boun
36d0: 64 72 79 2e 0a 2a 2a 20 54 68 65 20 6c 65 6e 67  dry..** The leng
36e0: 74 68 20 6f 66 20 74 68 65 20 63 6f 6e 74 65 6e  th of the conten
36f0: 74 20 69 73 20 73 74 6f 72 65 64 20 69 6e 20 2a  t is stored in *
3700: 70 6e 43 6f 6e 74 65 6e 74 2e 0a 2a 2f 0a 73 74  pnContent..*/.st
3710: 61 74 69 63 20 63 68 61 72 20 2a 67 65 74 5f 62  atic char *get_b
3720: 6f 75 6e 64 65 64 5f 63 6f 6e 74 65 6e 74 28 0a  ounded_content(.
3730: 20 20 63 68 61 72 20 2a 2a 70 7a 2c 20 20 20 20    char **pz,    
3740: 20 20 20 20 20 2f 2a 20 43 6f 6e 74 65 6e 74 20       /* Content 
3750: 74 61 6b 65 6e 20 66 72 6f 6d 20 68 65 72 65 20  taken from here 
3760: 2a 2f 0a 20 20 69 6e 74 20 2a 70 4c 65 6e 2c 20  */.  int *pLen, 
3770: 20 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d 62 65          /* Numbe
3780: 72 20 6f 66 20 62 79 74 65 73 20 6f 66 20 64 61  r of bytes of da
3790: 74 61 20 69 6e 20 28 2a 70 7a 29 5b 5d 20 2a 2f  ta in (*pz)[] */
37a0: 0a 20 20 63 68 61 72 20 2a 7a 42 6f 75 6e 64 72  .  char *zBoundr
37b0: 79 2c 20 20 20 20 2f 2a 20 42 6f 75 6e 64 72 79  y,    /* Boundry
37c0: 20 74 65 78 74 20 6d 61 72 6b 69 6e 67 20 74 68   text marking th
37d0: 65 20 65 6e 64 20 6f 66 20 63 6f 6e 74 65 6e 74  e end of content
37e0: 20 2a 2f 0a 20 20 69 6e 74 20 2a 70 6e 43 6f 6e   */.  int *pnCon
37f0: 74 65 6e 74 20 20 20 20 20 2f 2a 20 57 72 69 74  tent     /* Writ
3800: 65 20 74 68 65 20 73 69 7a 65 20 6f 66 20 74 68  e the size of th
3810: 65 20 63 6f 6e 74 65 6e 74 20 68 65 72 65 20 2a  e content here *
3820: 2f 0a 29 7b 0a 20 20 63 68 61 72 20 2a 7a 20 3d  /.){.  char *z =
3830: 20 2a 70 7a 3b 0a 20 20 69 6e 74 20 6c 65 6e 20   *pz;.  int len 
3840: 3d 20 2a 70 4c 65 6e 3b 0a 20 20 69 6e 74 20 69  = *pLen;.  int i
3850: 3b 0a 20 20 69 6e 74 20 6e 42 6f 75 6e 64 72 79  ;.  int nBoundry
3860: 20 3d 20 73 74 72 6c 65 6e 28 7a 42 6f 75 6e 64   = strlen(zBound
3870: 72 79 29 3b 0a 20 20 2a 70 6e 43 6f 6e 74 65 6e  ry);.  *pnConten
3880: 74 20 3d 20 6c 65 6e 3b 0a 20 20 66 6f 72 28 69  t = len;.  for(i
3890: 3d 30 3b 20 69 3c 6c 65 6e 3b 20 69 2b 2b 29 7b  =0; i<len; i++){
38a0: 0a 20 20 20 20 69 66 28 20 7a 5b 69 5d 3d 3d 27  .    if( z[i]=='
38b0: 5c 6e 27 20 26 26 20 73 74 72 6e 63 6d 70 28 7a  \n' && strncmp(z
38c0: 42 6f 75 6e 64 72 79 2c 20 26 7a 5b 69 2b 31 5d  Boundry, &z[i+1]
38d0: 2c 20 6e 42 6f 75 6e 64 72 79 29 3d 3d 30 20 29  , nBoundry)==0 )
38e0: 7b 0a 20 20 20 20 20 20 69 66 28 20 69 3e 30 20  {.      if( i>0 
38f0: 26 26 20 7a 5b 69 2d 31 5d 3d 3d 27 5c 72 27 20  && z[i-1]=='\r' 
3900: 29 20 69 2d 2d 3b 0a 20 20 20 20 20 20 7a 5b 69  ) i--;.      z[i
3910: 5d 20 3d 20 30 3b 0a 20 20 20 20 20 20 2a 70 6e  ] = 0;.      *pn
3920: 43 6f 6e 74 65 6e 74 20 3d 20 69 3b 0a 20 20 20  Content = i;.   
3930: 20 20 20 69 20 2b 3d 20 6e 42 6f 75 6e 64 72 79     i += nBoundry
3940: 3b 0a 20 20 20 20 20 20 62 72 65 61 6b 3b 0a 20  ;.      break;. 
3950: 20 20 20 7d 0a 20 20 7d 0a 20 20 2a 70 7a 20 3d     }.  }.  *pz =
3960: 20 26 7a 5b 69 5d 3b 0a 20 20 67 65 74 5f 6c 69   &z[i];.  get_li
3970: 6e 65 5f 66 72 6f 6d 5f 73 74 72 69 6e 67 28 70  ne_from_string(p
3980: 7a 2c 20 70 4c 65 6e 29 3b 0a 20 20 72 65 74 75  z, pLen);.  retu
3990: 72 6e 20 7a 3b 20 20 20 20 20 20 0a 7d 0a 0a 2f  rn z;      .}../
39a0: 2a 0a 2a 2a 20 54 6f 6b 65 6e 69 7a 65 20 61 20  *.** Tokenize a 
39b0: 6c 69 6e 65 20 6f 66 20 74 65 78 74 20 69 6e 74  line of text int
39c0: 6f 20 61 73 20 6d 61 6e 79 20 61 73 20 6e 41 72  o as many as nAr
39d0: 67 20 74 6f 6b 65 6e 73 2e 20 20 4d 61 6b 65 0a  g tokens.  Make.
39e0: 2a 2a 20 61 7a 41 72 67 5b 5d 20 70 6f 69 6e 74  ** azArg[] point
39f0: 20 74 6f 20 74 68 65 20 73 74 61 72 74 20 6f 66   to the start of
3a00: 20 65 61 63 68 20 74 6f 6b 65 6e 2e 0a 2a 2a 0a   each token..**.
3a10: 2a 2a 20 54 6f 6b 65 6e 73 20 63 6f 6e 73 69 73  ** Tokens consis
3a20: 74 20 6f 66 20 73 70 61 63 65 20 6f 72 20 73 65  t of space or se
3a30: 6d 69 2d 63 6f 6c 6f 6e 20 64 65 6c 69 6d 69 74  mi-colon delimit
3a40: 65 64 20 77 6f 72 64 73 20 6f 72 0a 2a 2a 20 73  ed words or.** s
3a50: 74 72 69 6e 67 73 20 69 6e 73 69 64 65 20 64 6f  trings inside do
3a60: 75 62 6c 65 2d 71 75 6f 74 65 73 2e 20 20 45 78  uble-quotes.  Ex
3a70: 61 6d 70 6c 65 3a 0a 2a 2a 0a 2a 2a 20 20 20 20  ample:.**.**    
3a80: 63 6f 6e 74 65 6e 74 2d 64 69 73 70 6f 73 69 74  content-disposit
3a90: 69 6f 6e 3a 20 66 6f 72 6d 2d 64 61 74 61 3b 20  ion: form-data; 
3aa0: 6e 61 6d 65 3d 22 66 6e 22 3b 20 66 69 6c 65 6e  name="fn"; filen
3ab0: 61 6d 65 3d 22 69 6e 64 65 78 2e 68 74 6d 6c 22  ame="index.html"
3ac0: 0a 2a 2a 0a 2a 2a 20 54 68 65 20 6c 69 6e 65 20  .**.** The line 
3ad0: 61 62 6f 76 65 20 69 73 20 74 6f 6b 65 6e 69 7a  above is tokeniz
3ae0: 65 64 20 61 73 20 66 6f 6c 6c 6f 77 73 3a 0a 2a  ed as follows:.*
3af0: 2a 0a 2a 2a 20 20 20 20 61 7a 41 72 67 5b 30 5d  *.**    azArg[0]
3b00: 20 3d 20 22 63 6f 6e 74 65 6e 74 2d 64 69 73 70   = "content-disp
3b10: 6f 73 69 74 69 6f 6e 3a 22 0a 2a 2a 20 20 20 20  osition:".**    
3b20: 61 7a 41 72 67 5b 31 5d 20 3d 20 22 66 6f 72 6d  azArg[1] = "form
3b30: 2d 64 61 74 61 22 0a 2a 2a 20 20 20 20 61 7a 41  -data".**    azA
3b40: 72 67 5b 32 5d 20 3d 20 22 6e 61 6d 65 3d 22 0a  rg[2] = "name=".
3b50: 2a 2a 20 20 20 20 61 7a 41 72 67 5b 33 5d 20 3d  **    azArg[3] =
3b60: 20 22 66 6e 22 0a 2a 2a 20 20 20 20 61 7a 41 72   "fn".**    azAr
3b70: 67 5b 34 5d 20 3d 20 22 66 69 6c 65 6e 61 6d 65  g[4] = "filename
3b80: 3d 22 0a 2a 2a 20 20 20 20 61 7a 41 72 67 5b 35  =".**    azArg[5
3b90: 5d 20 3d 20 22 69 6e 64 65 78 2e 68 74 6d 6c 22  ] = "index.html"
3ba0: 0a 2a 2a 20 20 20 20 61 7a 41 72 67 5b 36 5d 20  .**    azArg[6] 
3bb0: 3d 20 30 3b 0a 2a 2a 0a 2a 2a 20 27 5c 30 30 30  = 0;.**.** '\000
3bc0: 27 20 63 68 61 72 61 63 74 65 72 73 20 61 72 65  ' characters are
3bd0: 20 69 6e 73 65 72 74 65 64 20 69 6e 20 7a 5b 5d   inserted in z[]
3be0: 20 61 74 20 74 68 65 20 65 6e 64 20 6f 66 20 65   at the end of e
3bf0: 61 63 68 20 74 6f 6b 65 6e 2e 0a 2a 2a 20 54 68  ach token..** Th
3c00: 69 73 20 72 6f 75 74 69 6e 65 20 72 65 74 75 72  is routine retur
3c10: 6e 73 20 74 68 65 20 74 6f 74 61 6c 20 6e 75 6d  ns the total num
3c20: 62 65 72 20 6f 66 20 74 6f 6b 65 6e 73 20 6f 6e  ber of tokens on
3c30: 20 74 68 65 20 6c 69 6e 65 2c 20 36 0a 2a 2a 20   the line, 6.** 
3c40: 69 6e 20 74 68 65 20 65 78 61 6d 70 6c 65 20 61  in the example a
3c50: 62 6f 76 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20  bove..*/.static 
3c60: 69 6e 74 20 74 6f 6b 65 6e 69 7a 65 5f 6c 69 6e  int tokenize_lin
3c70: 65 28 63 68 61 72 20 2a 7a 2c 20 69 6e 74 20 6d  e(char *z, int m
3c80: 78 41 72 67 2c 20 63 68 61 72 20 2a 2a 61 7a 41  xArg, char **azA
3c90: 72 67 29 7b 0a 20 20 69 6e 74 20 69 20 3d 20 30  rg){.  int i = 0
3ca0: 3b 0a 20 20 77 68 69 6c 65 28 20 2a 7a 20 29 7b  ;.  while( *z ){
3cb0: 0a 20 20 20 20 77 68 69 6c 65 28 20 69 73 73 70  .    while( issp
3cc0: 61 63 65 28 2a 7a 29 20 7c 7c 20 2a 7a 3d 3d 27  ace(*z) || *z=='
3cd0: 3b 27 20 29 7b 20 7a 2b 2b 3b 20 7d 0a 20 20 20  ;' ){ z++; }.   
3ce0: 20 69 66 28 20 2a 7a 3d 3d 27 22 27 20 26 26 20   if( *z=='"' && 
3cf0: 7a 5b 31 5d 20 29 7b 0a 20 20 20 20 20 20 2a 7a  z[1] ){.      *z
3d00: 20 3d 20 30 3b 0a 20 20 20 20 20 20 7a 2b 2b 3b   = 0;.      z++;
3d10: 0a 20 20 20 20 20 20 69 66 28 20 69 3c 6d 78 41  .      if( i<mxA
3d20: 72 67 2d 31 20 29 7b 20 61 7a 41 72 67 5b 69 2b  rg-1 ){ azArg[i+
3d30: 2b 5d 20 3d 20 7a 3b 20 7d 0a 20 20 20 20 20 20  +] = z; }.      
3d40: 77 68 69 6c 65 28 20 2a 7a 20 26 26 20 2a 7a 21  while( *z && *z!
3d50: 3d 27 22 27 20 29 7b 20 7a 2b 2b 3b 20 7d 0a 20  ='"' ){ z++; }. 
3d60: 20 20 20 20 20 69 66 28 20 2a 7a 3d 3d 30 20 29       if( *z==0 )
3d70: 20 62 72 65 61 6b 3b 0a 20 20 20 20 20 20 2a 7a   break;.      *z
3d80: 20 3d 20 30 3b 0a 20 20 20 20 20 20 7a 2b 2b 3b   = 0;.      z++;
3d90: 0a 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20  .    }else{.    
3da0: 20 20 69 66 28 20 69 3c 6d 78 41 72 67 2d 31 20    if( i<mxArg-1 
3db0: 29 7b 20 61 7a 41 72 67 5b 69 2b 2b 5d 20 3d 20  ){ azArg[i++] = 
3dc0: 7a 3b 20 7d 0a 20 20 20 20 20 20 77 68 69 6c 65  z; }.      while
3dd0: 28 20 2a 7a 20 26 26 20 21 69 73 73 70 61 63 65  ( *z && !isspace
3de0: 28 2a 7a 29 20 26 26 20 2a 7a 21 3d 27 3b 27 20  (*z) && *z!=';' 
3df0: 26 26 20 2a 7a 21 3d 27 22 27 20 29 7b 20 7a 2b  && *z!='"' ){ z+
3e00: 2b 3b 20 7d 0a 20 20 20 20 20 20 69 66 28 20 2a  +; }.      if( *
3e10: 7a 20 26 26 20 2a 7a 21 3d 27 22 27 20 29 7b 0a  z && *z!='"' ){.
3e20: 20 20 20 20 20 20 20 20 2a 7a 20 3d 20 30 3b 0a          *z = 0;.
3e30: 20 20 20 20 20 20 20 20 7a 2b 2b 3b 0a 20 20 20          z++;.   
3e40: 20 20 20 7d 0a 20 20 20 20 7d 0a 20 20 7d 0a 20     }.    }.  }. 
3e50: 20 61 7a 41 72 67 5b 69 5d 20 3d 20 30 3b 0a 20   azArg[i] = 0;. 
3e60: 20 72 65 74 75 72 6e 20 69 3b 0a 7d 0a 0a 2f 2a   return i;.}../*
3e70: 0a 2a 2a 20 53 63 61 6e 20 74 68 65 20 6d 75 6c  .** Scan the mul
3e80: 74 69 70 61 72 74 2d 66 6f 72 6d 20 63 6f 6e 74  tipart-form cont
3e90: 65 6e 74 20 61 6e 64 20 6d 61 6b 65 20 61 70 70  ent and make app
3ea0: 72 6f 70 72 69 61 74 65 20 65 6e 74 72 69 65 73  ropriate entries
3eb0: 0a 2a 2a 20 69 6e 74 6f 20 74 68 65 20 70 61 72  .** into the par
3ec0: 61 6d 65 74 65 72 20 74 61 62 6c 65 2e 0a 2a 2a  ameter table..**
3ed0: 0a 2a 2a 20 54 68 65 20 63 6f 6e 74 65 6e 74 20  .** The content 
3ee0: 73 74 72 69 6e 67 20 22 7a 22 20 69 73 20 6d 6f  string "z" is mo
3ef0: 64 69 66 69 65 64 20 62 79 20 74 68 69 73 20 72  dified by this r
3f00: 6f 75 74 69 6e 65 20 62 75 74 20 69 74 20 69 73  outine but it is
3f10: 0a 2a 2a 20 6e 6f 74 20 63 6f 70 69 65 64 2e 20  .** not copied. 
3f20: 20 54 68 65 20 63 61 6c 6c 69 6e 67 20 66 75 6e   The calling fun
3f30: 63 74 69 6f 6e 20 6d 75 73 74 20 6e 6f 74 20 64  ction must not d
3f40: 65 61 6c 6c 6f 63 61 74 65 20 6f 72 20 6d 6f 64  eallocate or mod
3f50: 69 66 79 0a 2a 2a 20 22 7a 22 20 61 66 74 65 72  ify.** "z" after
3f60: 20 74 68 69 73 20 72 6f 75 74 69 6e 65 20 66 69   this routine fi
3f70: 6e 69 73 68 65 73 20 6f 72 20 69 74 20 63 6f 75  nishes or it cou
3f80: 6c 64 20 63 6f 72 72 75 70 74 20 74 68 65 20 70  ld corrupt the p
3f90: 61 72 61 6d 65 74 65 72 0a 2a 2a 20 74 61 62 6c  arameter.** tabl
3fa0: 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69  e..*/.static voi
3fb0: 64 20 70 72 6f 63 65 73 73 5f 6d 75 6c 74 69 70  d process_multip
3fc0: 61 72 74 5f 66 6f 72 6d 5f 64 61 74 61 28 63 68  art_form_data(ch
3fd0: 61 72 20 2a 7a 2c 20 69 6e 74 20 6c 65 6e 29 7b  ar *z, int len){
3fe0: 0a 20 20 63 68 61 72 20 2a 7a 4c 69 6e 65 3b 0a  .  char *zLine;.
3ff0: 20 20 69 6e 74 20 6e 41 72 67 2c 20 69 3b 0a 20    int nArg, i;. 
4000: 20 63 68 61 72 20 2a 7a 42 6f 75 6e 64 72 79 3b   char *zBoundry;
4010: 0a 20 20 63 68 61 72 20 2a 7a 56 61 6c 75 65 3b  .  char *zValue;
4020: 0a 20 20 63 68 61 72 20 2a 7a 4e 61 6d 65 20 3d  .  char *zName =
4030: 20 30 3b 0a 20 20 69 6e 74 20 73 68 6f 77 42 79   0;.  int showBy
4040: 74 65 73 20 3d 20 30 3b 0a 20 20 63 68 61 72 20  tes = 0;.  char 
4050: 2a 61 7a 41 72 67 5b 35 30 5d 3b 0a 0a 20 20 7a  *azArg[50];..  z
4060: 42 6f 75 6e 64 72 79 20 3d 20 67 65 74 5f 6c 69  Boundry = get_li
4070: 6e 65 5f 66 72 6f 6d 5f 73 74 72 69 6e 67 28 26  ne_from_string(&
4080: 7a 2c 20 26 6c 65 6e 29 3b 0a 20 20 69 66 28 20  z, &len);.  if( 
4090: 7a 42 6f 75 6e 64 72 79 3d 3d 30 20 29 20 72 65  zBoundry==0 ) re
40a0: 74 75 72 6e 3b 0a 20 20 77 68 69 6c 65 28 20 28  turn;.  while( (
40b0: 7a 4c 69 6e 65 20 3d 20 67 65 74 5f 6c 69 6e 65  zLine = get_line
40c0: 5f 66 72 6f 6d 5f 73 74 72 69 6e 67 28 26 7a 2c  _from_string(&z,
40d0: 20 26 6c 65 6e 29 29 21 3d 30 20 29 7b 0a 20 20   &len))!=0 ){.  
40e0: 20 20 69 66 28 20 7a 4c 69 6e 65 5b 30 5d 3d 3d    if( zLine[0]==
40f0: 30 20 29 7b 0a 20 20 20 20 20 20 69 6e 74 20 6e  0 ){.      int n
4100: 43 6f 6e 74 65 6e 74 20 3d 20 30 3b 0a 20 20 20  Content = 0;.   
4110: 20 20 20 7a 56 61 6c 75 65 20 3d 20 67 65 74 5f     zValue = get_
4120: 62 6f 75 6e 64 65 64 5f 63 6f 6e 74 65 6e 74 28  bounded_content(
4130: 26 7a 2c 20 26 6c 65 6e 2c 20 7a 42 6f 75 6e 64  &z, &len, zBound
4140: 72 79 2c 20 26 6e 43 6f 6e 74 65 6e 74 29 3b 0a  ry, &nContent);.
4150: 20 20 20 20 20 20 69 66 28 20 7a 4e 61 6d 65 20        if( zName 
4160: 26 26 20 7a 56 61 6c 75 65 20 26 26 20 69 73 6c  && zValue && isl
4170: 6f 77 65 72 28 7a 4e 61 6d 65 5b 30 5d 29 20 29  ower(zName[0]) )
4180: 7b 0a 20 20 20 20 20 20 20 20 63 67 69 5f 73 65  {.        cgi_se
4190: 74 5f 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f  t_parameter_noco
41a0: 70 79 28 7a 4e 61 6d 65 2c 20 7a 56 61 6c 75 65  py(zName, zValue
41b0: 29 3b 0a 20 20 20 20 20 20 20 20 69 66 28 20 73  );.        if( s
41c0: 68 6f 77 42 79 74 65 73 20 29 7b 0a 20 20 20 20  howBytes ){.    
41d0: 20 20 20 20 20 20 63 67 69 5f 73 65 74 5f 70 61        cgi_set_pa
41e0: 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79 28 6d  rameter_nocopy(m
41f0: 70 72 69 6e 74 66 28 22 25 73 3a 62 79 74 65 73  printf("%s:bytes
4200: 22 2c 20 7a 4e 61 6d 65 29 2c 0a 20 20 20 20 20  ", zName),.     
4210: 20 20 20 20 20 20 20 20 20 20 6d 70 72 69 6e 74            mprint
4220: 66 28 22 25 64 22 2c 6e 43 6f 6e 74 65 6e 74 29  f("%d",nContent)
4230: 29 3b 0a 20 20 20 20 20 20 20 20 7d 0a 20 20 20  );.        }.   
4240: 20 20 20 7d 0a 20 20 20 20 20 20 7a 4e 61 6d 65     }.      zName
4250: 20 3d 20 30 3b 0a 20 20 20 20 20 20 73 68 6f 77   = 0;.      show
4260: 42 79 74 65 73 20 3d 20 30 3b 0a 20 20 20 20 7d  Bytes = 0;.    }
4270: 65 6c 73 65 7b 0a 20 20 20 20 20 20 6e 41 72 67  else{.      nArg
4280: 20 3d 20 74 6f 6b 65 6e 69 7a 65 5f 6c 69 6e 65   = tokenize_line
4290: 28 7a 4c 69 6e 65 2c 20 73 69 7a 65 6f 66 28 61  (zLine, sizeof(a
42a0: 7a 41 72 67 29 2f 73 69 7a 65 6f 66 28 61 7a 41  zArg)/sizeof(azA
42b0: 72 67 5b 30 5d 29 2c 20 61 7a 41 72 67 29 3b 0a  rg[0]), azArg);.
42c0: 20 20 20 20 20 20 66 6f 72 28 69 3d 30 3b 20 69        for(i=0; i
42d0: 3c 6e 41 72 67 3b 20 69 2b 2b 29 7b 0a 20 20 20  <nArg; i++){.   
42e0: 20 20 20 20 20 69 6e 74 20 63 20 3d 20 74 6f 6c       int c = tol
42f0: 6f 77 65 72 28 61 7a 41 72 67 5b 69 5d 5b 30 5d  ower(azArg[i][0]
4300: 29 3b 0a 20 20 20 20 20 20 20 20 69 66 28 20 63  );.        if( c
4310: 3d 3d 27 63 27 20 26 26 20 73 74 72 69 63 6d 70  =='c' && stricmp
4320: 28 61 7a 41 72 67 5b 69 5d 2c 22 63 6f 6e 74 65  (azArg[i],"conte
4330: 6e 74 2d 64 69 73 70 6f 73 69 74 69 6f 6e 3a 22  nt-disposition:"
4340: 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 20 20  )==0 ){.        
4350: 20 20 69 2b 2b 3b 0a 20 20 20 20 20 20 20 20 7d    i++;.        }
4360: 65 6c 73 65 20 69 66 28 20 63 3d 3d 27 6e 27 20  else if( c=='n' 
4370: 26 26 20 73 74 72 69 63 6d 70 28 61 7a 41 72 67  && stricmp(azArg
4380: 5b 69 5d 2c 22 6e 61 6d 65 3d 22 29 3d 3d 30 20  [i],"name=")==0 
4390: 29 7b 0a 20 20 20 20 20 20 20 20 20 20 7a 4e 61  ){.          zNa
43a0: 6d 65 20 3d 20 61 7a 41 72 67 5b 2b 2b 69 5d 3b  me = azArg[++i];
43b0: 0a 20 20 20 20 20 20 20 20 7d 65 6c 73 65 20 69  .        }else i
43c0: 66 28 20 63 3d 3d 27 66 27 20 26 26 20 73 74 72  f( c=='f' && str
43d0: 69 63 6d 70 28 61 7a 41 72 67 5b 69 5d 2c 22 66  icmp(azArg[i],"f
43e0: 69 6c 65 6e 61 6d 65 3d 22 29 3d 3d 30 20 29 7b  ilename=")==0 ){
43f0: 0a 20 20 20 20 20 20 20 20 20 20 63 68 61 72 20  .          char 
4400: 2a 7a 20 3d 20 61 7a 41 72 67 5b 2b 2b 69 5d 3b  *z = azArg[++i];
4410: 0a 20 20 20 20 20 20 20 20 20 20 69 66 28 20 7a  .          if( z
4420: 4e 61 6d 65 20 26 26 20 7a 20 26 26 20 69 73 6c  Name && z && isl
4430: 6f 77 65 72 28 7a 4e 61 6d 65 5b 30 5d 29 20 29  ower(zName[0]) )
4440: 7b 0a 20 20 20 20 20 20 20 20 20 20 20 20 63 67  {.            cg
4450: 69 5f 73 65 74 5f 70 61 72 61 6d 65 74 65 72 5f  i_set_parameter_
4460: 6e 6f 63 6f 70 79 28 6d 70 72 69 6e 74 66 28 22  nocopy(mprintf("
4470: 25 73 3a 66 69 6c 65 6e 61 6d 65 22 2c 7a 4e 61  %s:filename",zNa
4480: 6d 65 29 2c 20 7a 29 3b 0a 20 20 20 20 20 20 20  me), z);.       
4490: 20 20 20 7d 0a 20 20 20 20 20 20 20 20 20 20 73     }.          s
44a0: 68 6f 77 42 79 74 65 73 20 3d 20 31 3b 0a 20 20  howBytes = 1;.  
44b0: 20 20 20 20 20 20 7d 65 6c 73 65 20 69 66 28 20        }else if( 
44c0: 63 3d 3d 27 63 27 20 26 26 20 73 74 72 69 63 6d  c=='c' && stricm
44d0: 70 28 61 7a 41 72 67 5b 69 5d 2c 22 63 6f 6e 74  p(azArg[i],"cont
44e0: 65 6e 74 2d 74 79 70 65 3a 22 29 3d 3d 30 20 29  ent-type:")==0 )
44f0: 7b 0a 20 20 20 20 20 20 20 20 20 20 63 68 61 72  {.          char
4500: 20 2a 7a 20 3d 20 61 7a 41 72 67 5b 2b 2b 69 5d   *z = azArg[++i]
4510: 3b 0a 20 20 20 20 20 20 20 20 20 20 69 66 28 20  ;.          if( 
4520: 7a 4e 61 6d 65 20 26 26 20 7a 20 26 26 20 69 73  zName && z && is
4530: 6c 6f 77 65 72 28 7a 4e 61 6d 65 5b 30 5d 29 20  lower(zName[0]) 
4540: 29 7b 0a 20 20 20 20 20 20 20 20 20 20 20 20 63  ){.            c
4550: 67 69 5f 73 65 74 5f 70 61 72 61 6d 65 74 65 72  gi_set_parameter
4560: 5f 6e 6f 63 6f 70 79 28 6d 70 72 69 6e 74 66 28  _nocopy(mprintf(
4570: 22 25 73 3a 6d 69 6d 65 74 79 70 65 22 2c 7a 4e  "%s:mimetype",zN
4580: 61 6d 65 29 2c 20 7a 29 3b 0a 20 20 20 20 20 20  ame), z);.      
4590: 20 20 20 20 7d 0a 20 20 20 20 20 20 20 20 7d 0a      }.        }.
45a0: 20 20 20 20 20 20 7d 0a 20 20 20 20 7d 0a 20 20        }.    }.  
45b0: 7d 20 20 20 20 20 20 20 20 0a 7d 0a 0a 2f 2a 0a  }        .}../*.
45c0: 2a 2a 20 49 6e 69 74 69 61 6c 69 7a 65 20 74 68  ** Initialize th
45d0: 65 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65  e query paramete
45e0: 72 20 64 61 74 61 62 61 73 65 2e 20 20 49 6e 66  r database.  Inf
45f0: 6f 72 6d 61 74 69 6f 6e 20 69 73 20 70 75 6c 6c  ormation is pull
4600: 65 64 20 66 72 6f 6d 0a 2a 2a 20 74 68 65 20 51  ed from.** the Q
4610: 55 45 52 59 5f 53 54 52 49 4e 47 20 65 6e 76 69  UERY_STRING envi
4620: 72 6f 6e 6d 65 6e 74 20 76 61 72 69 61 62 6c 65  ronment variable
4630: 20 28 69 66 20 69 74 20 65 78 69 73 74 73 29 2c   (if it exists),
4640: 20 66 72 6f 6d 20 73 74 61 6e 64 61 72 64 0a 2a   from standard.*
4650: 2a 20 69 6e 70 75 74 20 69 66 20 74 68 65 72 65  * input if there
4660: 20 69 73 20 50 4f 53 54 20 64 61 74 61 2c 20 61   is POST data, a
4670: 6e 64 20 66 72 6f 6d 20 48 54 54 50 5f 43 4f 4f  nd from HTTP_COO
4680: 4b 49 45 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69  KIE..*/.void cgi
4690: 5f 69 6e 69 74 28 76 6f 69 64 29 7b 0a 20 20 63  _init(void){.  c
46a0: 68 61 72 20 2a 7a 3b 0a 20 20 63 6f 6e 73 74 20  har *z;.  const 
46b0: 63 68 61 72 20 2a 7a 54 79 70 65 3b 0a 20 20 69  char *zType;.  i
46c0: 6e 74 20 6c 65 6e 3b 0a 20 20 7a 20 3d 20 28 63  nt len;.  z = (c
46d0: 68 61 72 2a 29 50 28 22 51 55 45 52 59 5f 53 54  har*)P("QUERY_ST
46e0: 52 49 4e 47 22 29 3b 0a 20 20 69 66 28 20 7a 20  RING");.  if( z 
46f0: 29 7b 0a 20 20 20 20 7a 20 3d 20 6d 70 72 69 6e  ){.    z = mprin
4700: 74 66 28 22 25 73 22 2c 7a 29 3b 0a 20 20 20 20  tf("%s",z);.    
4710: 61 64 64 5f 70 61 72 61 6d 5f 6c 69 73 74 28 7a  add_param_list(z
4720: 2c 20 27 26 27 29 3b 0a 20 20 7d 0a 0a 20 20 6c  , '&');.  }..  l
4730: 65 6e 20 3d 20 61 74 6f 69 28 50 44 28 22 43 4f  en = atoi(PD("CO
4740: 4e 54 45 4e 54 5f 4c 45 4e 47 54 48 22 2c 20 22  NTENT_LENGTH", "
4750: 30 22 29 29 3b 0a 20 20 67 2e 7a 43 6f 6e 74 65  0"));.  g.zConte
4760: 6e 74 54 79 70 65 20 3d 20 7a 54 79 70 65 20 3d  ntType = zType =
4770: 20 50 28 22 43 4f 4e 54 45 4e 54 5f 54 59 50 45   P("CONTENT_TYPE
4780: 22 29 3b 0a 20 20 69 66 28 20 6c 65 6e 3e 30 20  ");.  if( len>0 
4790: 26 26 20 7a 54 79 70 65 20 29 7b 0a 20 20 20 20  && zType ){.    
47a0: 62 6c 6f 62 5f 7a 65 72 6f 28 26 67 2e 63 67 69  blob_zero(&g.cgi
47b0: 49 6e 29 3b 0a 20 20 20 20 69 66 28 20 73 74 72  In);.    if( str
47c0: 63 6d 70 28 7a 54 79 70 65 2c 22 61 70 70 6c 69  cmp(zType,"appli
47d0: 63 61 74 69 6f 6e 2f 78 2d 77 77 77 2d 66 6f 72  cation/x-www-for
47e0: 6d 2d 75 72 6c 65 6e 63 6f 64 65 64 22 29 3d 3d  m-urlencoded")==
47f0: 30 20 0a 20 20 20 20 20 20 20 20 20 7c 7c 20 73  0 .         || s
4800: 74 72 6e 63 6d 70 28 7a 54 79 70 65 2c 22 6d 75  trncmp(zType,"mu
4810: 6c 74 69 70 61 72 74 2f 66 6f 72 6d 2d 64 61 74  ltipart/form-dat
4820: 61 22 2c 31 39 29 3d 3d 30 20 29 7b 0a 20 20 20  a",19)==0 ){.   
4830: 20 20 20 7a 20 3d 20 6d 61 6c 6c 6f 63 28 20 6c     z = malloc( l
4840: 65 6e 2b 31 20 29 3b 0a 20 20 20 20 20 20 69 66  en+1 );.      if
4850: 28 20 7a 3d 3d 30 20 29 20 65 78 69 74 28 31 29  ( z==0 ) exit(1)
4860: 3b 0a 20 20 20 20 20 20 6c 65 6e 20 3d 20 66 72  ;.      len = fr
4870: 65 61 64 28 7a 2c 20 31 2c 20 6c 65 6e 2c 20 73  ead(z, 1, len, s
4880: 74 64 69 6e 29 3b 0a 20 20 20 20 20 20 7a 5b 6c  tdin);.      z[l
4890: 65 6e 5d 20 3d 20 30 3b 0a 20 20 20 20 20 20 69  en] = 0;.      i
48a0: 66 28 20 7a 54 79 70 65 5b 30 5d 3d 3d 27 61 27  f( zType[0]=='a'
48b0: 20 29 7b 0a 20 20 20 20 20 20 20 20 61 64 64 5f   ){.        add_
48c0: 70 61 72 61 6d 5f 6c 69 73 74 28 7a 2c 20 27 26  param_list(z, '&
48d0: 27 29 3b 0a 20 20 20 20 20 20 7d 65 6c 73 65 7b  ');.      }else{
48e0: 0a 20 20 20 20 20 20 20 20 70 72 6f 63 65 73 73  .        process
48f0: 5f 6d 75 6c 74 69 70 61 72 74 5f 66 6f 72 6d 5f  _multipart_form_
4900: 64 61 74 61 28 7a 2c 20 6c 65 6e 29 3b 0a 20 20  data(z, len);.  
4910: 20 20 20 20 7d 0a 20 20 20 20 7d 65 6c 73 65 20      }.    }else 
4920: 69 66 28 20 73 74 72 63 6d 70 28 7a 54 79 70 65  if( strcmp(zType
4930: 2c 20 22 61 70 70 6c 69 63 61 74 69 6f 6e 2f 78  , "application/x
4940: 2d 66 6f 73 73 69 6c 22 29 3d 3d 30 20 29 7b 0a  -fossil")==0 ){.
4950: 20 20 20 20 20 20 62 6c 6f 62 5f 72 65 61 64 5f        blob_read_
4960: 66 72 6f 6d 5f 63 68 61 6e 6e 65 6c 28 26 67 2e  from_channel(&g.
4970: 63 67 69 49 6e 2c 20 73 74 64 69 6e 2c 20 6c 65  cgiIn, stdin, le
4980: 6e 29 3b 0a 20 20 20 20 20 20 62 6c 6f 62 5f 75  n);.      blob_u
4990: 6e 63 6f 6d 70 72 65 73 73 28 26 67 2e 63 67 69  ncompress(&g.cgi
49a0: 49 6e 2c 20 26 67 2e 63 67 69 49 6e 29 3b 0a 20  In, &g.cgiIn);. 
49b0: 20 20 20 7d 65 6c 73 65 20 69 66 28 20 73 74 72     }else if( str
49c0: 63 6d 70 28 7a 54 79 70 65 2c 20 22 61 70 70 6c  cmp(zType, "appl
49d0: 69 63 61 74 69 6f 6e 2f 78 2d 66 6f 73 73 69 6c  ication/x-fossil
49e0: 2d 64 65 62 75 67 22 29 3d 3d 30 20 29 7b 0a 20  -debug")==0 ){. 
49f0: 20 20 20 20 20 62 6c 6f 62 5f 72 65 61 64 5f 66       blob_read_f
4a00: 72 6f 6d 5f 63 68 61 6e 6e 65 6c 28 26 67 2e 63  rom_channel(&g.c
4a10: 67 69 49 6e 2c 20 73 74 64 69 6e 2c 20 6c 65 6e  giIn, stdin, len
4a20: 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 0a 20 20  );.    }.  }..  
4a30: 7a 20 3d 20 28 63 68 61 72 2a 29 50 28 22 48 54  z = (char*)P("HT
4a40: 54 50 5f 43 4f 4f 4b 49 45 22 29 3b 0a 20 20 69  TP_COOKIE");.  i
4a50: 66 28 20 7a 20 29 7b 0a 20 20 20 20 7a 20 3d 20  f( z ){.    z = 
4a60: 6d 70 72 69 6e 74 66 28 22 25 73 22 2c 7a 29 3b  mprintf("%s",z);
4a70: 0a 20 20 20 20 61 64 64 5f 70 61 72 61 6d 5f 6c  .    add_param_l
4a80: 69 73 74 28 7a 2c 20 27 3b 27 29 3b 0a 20 20 7d  ist(z, ';');.  }
4a90: 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 69 73 20 69  .}../*.** This i
4aa0: 73 20 74 68 65 20 63 6f 6d 70 61 72 69 73 6f 6e  s the comparison
4ab0: 20 66 75 6e 63 74 69 6f 6e 20 75 73 65 64 20 74   function used t
4ac0: 6f 20 73 6f 72 74 20 74 68 65 20 61 50 61 72 61  o sort the aPara
4ad0: 6d 51 50 5b 5d 20 61 72 72 61 79 20 6f 66 0a 2a  mQP[] array of.*
4ae0: 2a 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65  * query paramete
4af0: 72 73 20 61 6e 64 20 63 6f 6f 6b 69 65 73 2e 0a  rs and cookies..
4b00: 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 71 70  */.static int qp
4b10: 61 72 61 6d 5f 63 6f 6d 70 61 72 65 28 63 6f 6e  aram_compare(con
4b20: 73 74 20 76 6f 69 64 20 2a 61 2c 20 63 6f 6e 73  st void *a, cons
4b30: 74 20 76 6f 69 64 20 2a 62 29 7b 0a 20 20 73 74  t void *b){.  st
4b40: 72 75 63 74 20 51 50 61 72 61 6d 20 2a 70 41 20  ruct QParam *pA 
4b50: 3d 20 28 73 74 72 75 63 74 20 51 50 61 72 61 6d  = (struct QParam
4b60: 2a 29 61 3b 0a 20 20 73 74 72 75 63 74 20 51 50  *)a;.  struct QP
4b70: 61 72 61 6d 20 2a 70 42 20 3d 20 28 73 74 72 75  aram *pB = (stru
4b80: 63 74 20 51 50 61 72 61 6d 2a 29 62 3b 0a 20 20  ct QParam*)b;.  
4b90: 69 6e 74 20 63 3b 0a 20 20 63 20 3d 20 73 74 72  int c;.  c = str
4ba0: 63 6d 70 28 70 41 2d 3e 7a 4e 61 6d 65 2c 20 70  cmp(pA->zName, p
4bb0: 42 2d 3e 7a 4e 61 6d 65 29 3b 0a 20 20 69 66 28  B->zName);.  if(
4bc0: 20 63 3d 3d 30 20 29 7b 0a 20 20 20 20 63 20 3d   c==0 ){.    c =
4bd0: 20 70 41 2d 3e 73 65 71 20 2d 20 70 42 2d 3e 73   pA->seq - pB->s
4be0: 65 71 3b 0a 20 20 7d 0a 20 20 72 65 74 75 72 6e  eq;.  }.  return
4bf0: 20 63 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 52 65 74   c;.}../*.** Ret
4c00: 75 72 6e 20 74 68 65 20 76 61 6c 75 65 20 6f 66  urn the value of
4c10: 20 61 20 71 75 65 72 79 20 70 61 72 61 6d 65 74   a query paramet
4c20: 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 77 68 6f  er or cookie who
4c30: 73 65 20 6e 61 6d 65 20 69 73 20 7a 4e 61 6d 65  se name is zName
4c40: 2e 0a 2a 2a 20 49 66 20 74 68 65 72 65 20 69 73  ..** If there is
4c50: 20 6e 6f 20 71 75 65 72 79 20 70 61 72 61 6d 65   no query parame
4c60: 74 65 72 20 6f 72 20 63 6f 6f 6b 69 65 20 6e 61  ter or cookie na
4c70: 6d 65 64 20 7a 4e 61 6d 65 20 61 6e 64 20 74 68  med zName and th
4c80: 65 20 66 69 72 73 74 0a 2a 2a 20 63 68 61 72 61  e first.** chara
4c90: 63 74 65 72 20 6f 66 20 7a 4e 61 6d 65 20 69 73  cter of zName is
4ca0: 20 75 70 70 65 72 63 61 73 65 2c 20 74 68 65 6e   uppercase, then
4cb0: 20 63 68 65 63 6b 20 74 6f 20 73 65 65 20 69 66   check to see if
4cc0: 20 74 68 65 72 65 20 69 73 20 61 6e 0a 2a 2a 20   there is an.** 
4cd0: 65 6e 76 69 72 6f 6e 6d 65 6e 74 20 76 61 72 69  environment vari
4ce0: 61 62 6c 65 20 62 79 20 74 68 61 74 20 6e 61 6d  able by that nam
4cf0: 65 20 61 6e 64 20 72 65 74 75 72 6e 20 69 74 20  e and return it 
4d00: 69 66 20 74 68 65 72 65 20 69 73 2e 20 20 41 73  if there is.  As
4d10: 0a 2a 2a 20 61 20 6c 61 73 74 20 72 65 73 6f 72  .** a last resor
4d20: 74 20 77 68 65 6e 20 6e 6f 74 68 69 6e 67 20 65  t when nothing e
4d30: 6c 73 65 20 6d 61 74 63 68 65 73 2c 20 72 65 74  lse matches, ret
4d40: 75 72 6e 20 7a 44 65 66 61 75 6c 74 2e 0a 2a 2f  urn zDefault..*/
4d50: 0a 63 6f 6e 73 74 20 63 68 61 72 20 2a 63 67 69  .const char *cgi
4d60: 5f 70 61 72 61 6d 65 74 65 72 28 63 6f 6e 73 74  _parameter(const
4d70: 20 63 68 61 72 20 2a 7a 4e 61 6d 65 2c 20 63 6f   char *zName, co
4d80: 6e 73 74 20 63 68 61 72 20 2a 7a 44 65 66 61 75  nst char *zDefau
4d90: 6c 74 29 7b 0a 20 20 69 6e 74 20 6c 6f 2c 20 68  lt){.  int lo, h
4da0: 69 2c 20 6d 69 64 2c 20 63 3b 0a 0a 20 20 2f 2a  i, mid, c;..  /*
4db0: 20 54 68 65 20 73 6f 72 74 51 50 20 66 6c 61 67   The sortQP flag
4dc0: 20 69 73 20 73 65 74 20 77 68 65 6e 65 76 65 72   is set whenever
4dd0: 20 61 20 6e 65 77 20 71 75 65 72 79 20 70 61 72   a new query par
4de0: 61 6d 65 74 65 72 20 69 73 20 69 6e 73 65 72 74  ameter is insert
4df0: 65 64 2e 0a 20 20 2a 2a 20 49 74 20 69 6e 64 69  ed..  ** It indi
4e00: 63 61 74 65 73 20 74 68 61 74 20 77 65 20 6e 65  cates that we ne
4e10: 65 64 20 74 6f 20 72 65 73 6f 72 74 20 74 68 65  ed to resort the
4e20: 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72   query parameter
4e30: 73 2e 0a 20 20 2a 2f 0a 20 20 69 66 28 20 73 6f  s..  */.  if( so
4e40: 72 74 51 50 20 29 7b 0a 20 20 20 20 69 6e 74 20  rtQP ){.    int 
4e50: 69 2c 20 6a 3b 0a 20 20 20 20 71 73 6f 72 74 28  i, j;.    qsort(
4e60: 61 50 61 72 61 6d 51 50 2c 20 6e 55 73 65 64 51  aParamQP, nUsedQ
4e70: 50 2c 20 73 69 7a 65 6f 66 28 61 50 61 72 61 6d  P, sizeof(aParam
4e80: 51 50 5b 30 5d 29 2c 20 71 70 61 72 61 6d 5f 63  QP[0]), qparam_c
4e90: 6f 6d 70 61 72 65 29 3b 0a 20 20 20 20 73 6f 72  ompare);.    sor
4ea0: 74 51 50 20 3d 20 30 3b 0a 20 20 20 20 2f 2a 20  tQP = 0;.    /* 
4eb0: 41 66 74 65 72 20 73 6f 72 74 69 6e 67 2c 20 72  After sorting, r
4ec0: 65 6d 6f 76 65 20 64 75 70 6c 69 63 61 74 65 20  emove duplicate 
4ed0: 70 61 72 61 6d 65 74 65 72 73 2e 20 20 54 68 65  parameters.  The
4ee0: 20 73 65 63 6f 6e 64 61 72 79 20 73 6f 72 74 0a   secondary sort.
4ef0: 20 20 20 20 2a 2a 20 6b 65 79 20 69 73 20 61 50      ** key is aP
4f00: 61 72 61 6d 51 50 5b 5d 2e 73 65 71 20 61 6e 64  aramQP[].seq and
4f10: 20 77 65 20 6b 65 65 70 20 74 68 65 20 66 69 72   we keep the fir
4f20: 73 74 20 65 6e 74 72 79 2e 20 20 54 68 61 74 20  st entry.  That 
4f30: 6d 65 61 6e 73 0a 20 20 20 20 2a 2a 20 77 69 74  means.    ** wit
4f40: 68 20 64 75 70 6c 69 63 61 74 65 20 63 61 6c 6c  h duplicate call
4f50: 73 20 74 6f 20 63 67 69 5f 73 65 74 5f 70 61 72  s to cgi_set_par
4f60: 61 6d 65 74 65 72 28 29 20 74 68 65 20 73 65 63  ameter() the sec
4f70: 6f 6e 64 20 61 6e 64 0a 20 20 20 20 2a 2a 20 73  ond and.    ** s
4f80: 75 62 73 65 71 75 65 6e 74 20 63 61 6c 6c 73 20  ubsequent calls 
4f90: 61 72 65 20 65 66 66 65 63 74 69 76 65 6c 79 20  are effectively 
4fa0: 6e 6f 2d 6f 70 73 2e 20 2a 2f 0a 20 20 20 20 66  no-ops. */.    f
4fb0: 6f 72 28 69 3d 6a 3d 31 3b 20 69 3c 6e 55 73 65  or(i=j=1; i<nUse
4fc0: 64 51 50 3b 20 69 2b 2b 29 7b 0a 20 20 20 20 20  dQP; i++){.     
4fd0: 20 69 66 28 20 73 74 72 63 6d 70 28 61 50 61 72   if( strcmp(aPar
4fe0: 61 6d 51 50 5b 69 5d 2e 7a 4e 61 6d 65 2c 61 50  amQP[i].zName,aP
4ff0: 61 72 61 6d 51 50 5b 69 2d 31 5d 2e 7a 4e 61 6d  aramQP[i-1].zNam
5000: 65 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 20  e)==0 ){.       
5010: 20 63 6f 6e 74 69 6e 75 65 3b 0a 20 20 20 20 20   continue;.     
5020: 20 7d 0a 20 20 20 20 20 20 69 66 28 20 6a 3c 69   }.      if( j<i
5030: 20 29 7b 0a 20 20 20 20 20 20 20 20 6d 65 6d 63   ){.        memc
5040: 70 79 28 26 61 50 61 72 61 6d 51 50 5b 6a 5d 2c  py(&aParamQP[j],
5050: 20 26 61 50 61 72 61 6d 51 50 5b 69 5d 2c 20 73   &aParamQP[i], s
5060: 69 7a 65 6f 66 28 61 50 61 72 61 6d 51 50 5b 6a  izeof(aParamQP[j
5070: 5d 29 29 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20  ]));.      }.   
5080: 20 20 20 6a 2b 2b 3b 0a 20 20 20 20 7d 0a 20 20     j++;.    }.  
5090: 20 20 6e 55 73 65 64 51 50 20 3d 20 6a 3b 0a 20    nUsedQP = j;. 
50a0: 20 7d 0a 0a 20 20 2f 2a 20 44 6f 20 61 20 62 69   }..  /* Do a bi
50b0: 6e 61 72 79 20 73 65 61 72 63 68 20 66 6f 72 20  nary search for 
50c0: 61 20 6d 61 74 63 68 69 6e 67 20 71 75 65 72 79  a matching query
50d0: 20 70 61 72 61 6d 65 74 65 72 20 2a 2f 0a 20 20   parameter */.  
50e0: 6c 6f 20 3d 20 30 3b 0a 20 20 68 69 20 3d 20 6e  lo = 0;.  hi = n
50f0: 55 73 65 64 51 50 2d 31 3b 0a 20 20 77 68 69 6c  UsedQP-1;.  whil
5100: 65 28 20 6c 6f 3c 3d 68 69 20 29 7b 0a 20 20 20  e( lo<=hi ){.   
5110: 20 6d 69 64 20 3d 20 28 6c 6f 2b 68 69 29 2f 32   mid = (lo+hi)/2
5120: 3b 0a 20 20 20 20 63 20 3d 20 73 74 72 63 6d 70  ;.    c = strcmp
5130: 28 61 50 61 72 61 6d 51 50 5b 6d 69 64 5d 2e 7a  (aParamQP[mid].z
5140: 4e 61 6d 65 2c 20 7a 4e 61 6d 65 29 3b 0a 20 20  Name, zName);.  
5150: 20 20 69 66 28 20 63 3d 3d 30 20 29 7b 0a 20 20    if( c==0 ){.  
5160: 20 20 20 20 43 47 49 44 45 42 55 47 28 28 22 6d      CGIDEBUG(("m
5170: 65 6d 2d 6d 61 74 63 68 20 5b 25 73 5d 20 3d 20  em-match [%s] = 
5180: 5b 25 73 5d 5c 6e 22 2c 20 7a 4e 61 6d 65 2c 20  [%s]\n", zName, 
5190: 61 50 61 72 61 6d 51 50 5b 6d 69 64 5d 2e 7a 56  aParamQP[mid].zV
51a0: 61 6c 75 65 29 29 3b 0a 20 20 20 20 20 20 72 65  alue));.      re
51b0: 74 75 72 6e 20 61 50 61 72 61 6d 51 50 5b 6d 69  turn aParamQP[mi
51c0: 64 5d 2e 7a 56 61 6c 75 65 3b 0a 20 20 20 20 7d  d].zValue;.    }
51d0: 65 6c 73 65 20 69 66 28 20 63 3e 30 20 29 7b 0a  else if( c>0 ){.
51e0: 20 20 20 20 20 20 68 69 20 3d 20 6d 69 64 2d 31        hi = mid-1
51f0: 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20  ;.    }else{.   
5200: 20 20 20 6c 6f 20 3d 20 6d 69 64 2b 31 3b 0a 20     lo = mid+1;. 
5210: 20 20 20 7d 0a 20 20 7d 0a 0a 20 20 2f 2a 20 49     }.  }..  /* I
5220: 66 20 6e 6f 20 6d 61 74 63 68 20 69 73 20 66 6f  f no match is fo
5230: 75 6e 64 20 61 6e 64 20 74 68 65 20 6e 61 6d 65  und and the name
5240: 20 62 65 67 69 6e 73 20 77 69 74 68 20 61 6e 20   begins with an 
5250: 75 70 70 65 72 2d 63 61 73 65 0a 20 20 2a 2a 20  upper-case.  ** 
5260: 6c 65 74 74 65 72 2c 20 74 68 65 6e 20 63 68 65  letter, then che
5270: 63 6b 20 74 6f 20 73 65 65 20 69 66 20 74 68 65  ck to see if the
5280: 72 65 20 69 73 20 61 6e 20 65 6e 76 69 72 6f 6e  re is an environ
5290: 6d 65 6e 74 20 76 61 72 69 61 62 6c 65 0a 20 20  ment variable.  
52a0: 2a 2a 20 77 69 74 68 20 74 68 65 20 67 69 76 65  ** with the give
52b0: 6e 20 6e 61 6d 65 2e 0a 20 20 2a 2f 0a 20 20 69  n name..  */.  i
52c0: 66 28 20 69 73 75 70 70 65 72 28 7a 4e 61 6d 65  f( isupper(zName
52d0: 5b 30 5d 29 20 29 7b 0a 20 20 20 20 63 6f 6e 73  [0]) ){.    cons
52e0: 74 20 63 68 61 72 20 2a 7a 56 61 6c 75 65 20 3d  t char *zValue =
52f0: 20 67 65 74 65 6e 76 28 7a 4e 61 6d 65 29 3b 0a   getenv(zName);.
5300: 20 20 20 20 69 66 28 20 7a 56 61 6c 75 65 20 29      if( zValue )
5310: 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 65 74 5f  {.      cgi_set_
5320: 70 61 72 61 6d 65 74 65 72 5f 6e 6f 63 6f 70 79  parameter_nocopy
5330: 28 7a 4e 61 6d 65 2c 20 7a 56 61 6c 75 65 29 3b  (zName, zValue);
5340: 0a 20 20 20 20 20 20 43 47 49 44 45 42 55 47 28  .      CGIDEBUG(
5350: 28 22 65 6e 76 2d 6d 61 74 63 68 20 5b 25 73 5d  ("env-match [%s]
5360: 20 3d 20 5b 25 73 5d 5c 6e 22 2c 20 7a 4e 61 6d   = [%s]\n", zNam
5370: 65 2c 20 7a 56 61 6c 75 65 29 29 3b 0a 20 20 20  e, zValue));.   
5380: 20 20 20 72 65 74 75 72 6e 20 7a 56 61 6c 75 65     return zValue
5390: 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 43 47  ;.    }.  }.  CG
53a0: 49 44 45 42 55 47 28 28 22 6e 6f 2d 6d 61 74 63  IDEBUG(("no-matc
53b0: 68 20 5b 25 73 5d 5c 6e 22 2c 20 7a 4e 61 6d 65  h [%s]\n", zName
53c0: 29 29 3b 0a 20 20 72 65 74 75 72 6e 20 7a 44 65  ));.  return zDe
53d0: 66 61 75 6c 74 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20  fault;.}../*.** 
53e0: 52 65 74 75 72 6e 20 74 68 65 20 6e 61 6d 65 20  Return the name 
53f0: 6f 66 20 74 68 65 20 69 2d 74 68 20 43 47 49 20  of the i-th CGI 
5400: 70 61 72 61 6d 65 74 65 72 2e 20 20 52 65 74 75  parameter.  Retu
5410: 72 6e 20 4e 55 4c 4c 20 69 66 20 74 68 65 72 65  rn NULL if there
5420: 0a 2a 2a 20 61 72 65 20 66 65 77 65 72 20 74 68  .** are fewer th
5430: 61 6e 20 69 20 72 65 67 69 73 74 65 72 65 64 20  an i registered 
5440: 43 47 49 20 70 61 72 6d 61 65 74 65 72 73 2e 0a  CGI parmaeters..
5450: 2a 2f 0a 63 6f 6e 73 74 20 63 68 61 72 20 2a 63  */.const char *c
5460: 67 69 5f 70 61 72 61 6d 65 74 65 72 5f 6e 61 6d  gi_parameter_nam
5470: 65 28 69 6e 74 20 69 29 7b 0a 20 20 69 66 28 20  e(int i){.  if( 
5480: 69 3e 3d 30 20 26 26 20 69 3c 6e 55 73 65 64 51  i>=0 && i<nUsedQ
5490: 50 20 29 7b 0a 20 20 20 20 72 65 74 75 72 6e 20  P ){.    return 
54a0: 61 50 61 72 61 6d 51 50 5b 69 5d 2e 7a 4e 61 6d  aParamQP[i].zNam
54b0: 65 3b 0a 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20  e;.  }else{.    
54c0: 72 65 74 75 72 6e 20 30 3b 0a 20 20 7d 0a 7d 0a  return 0;.  }.}.
54d0: 0a 2f 2a 0a 2a 2a 20 50 72 69 6e 74 20 43 47 49  ./*.** Print CGI
54e0: 20 64 65 62 75 67 67 69 6e 67 20 6d 65 73 73 61   debugging messa
54f0: 67 65 73 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69  ges..*/.void cgi
5500: 5f 64 65 62 75 67 28 63 6f 6e 73 74 20 63 68 61  _debug(const cha
5510: 72 20 2a 7a 46 6f 72 6d 61 74 2c 20 2e 2e 2e 29  r *zFormat, ...)
5520: 7b 0a 20 20 76 61 5f 6c 69 73 74 20 61 70 3b 0a  {.  va_list ap;.
5530: 20 20 69 66 28 20 67 2e 66 44 65 62 75 67 20 29    if( g.fDebug )
5540: 7b 0a 20 20 20 20 76 61 5f 73 74 61 72 74 28 61  {.    va_start(a
5550: 70 2c 20 7a 46 6f 72 6d 61 74 29 3b 0a 20 20 20  p, zFormat);.   
5560: 20 76 66 70 72 69 6e 74 66 28 67 2e 66 44 65 62   vfprintf(g.fDeb
5570: 75 67 2c 20 7a 46 6f 72 6d 61 74 2c 20 61 70 29  ug, zFormat, ap)
5580: 3b 0a 20 20 20 20 76 61 5f 65 6e 64 28 61 70 29  ;.    va_end(ap)
5590: 3b 0a 20 20 20 20 66 66 6c 75 73 68 28 67 2e 66  ;.    fflush(g.f
55a0: 44 65 62 75 67 29 3b 0a 20 20 7d 0a 7d 0a 0a 2f  Debug);.  }.}../
55b0: 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 74 72 75 65  *.** Return true
55c0: 20 69 66 20 61 6e 79 20 6f 66 20 74 68 65 20 71   if any of the q
55d0: 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 73 20  uery parameters 
55e0: 69 6e 20 74 68 65 20 61 72 67 75 6d 65 6e 74 0a  in the argument.
55f0: 2a 2a 20 6c 69 73 74 20 61 72 65 20 64 65 66 69  ** list are defi
5600: 6e 65 64 2e 0a 2a 2f 0a 69 6e 74 20 63 67 69 5f  ned..*/.int cgi_
5610: 61 6e 79 28 63 6f 6e 73 74 20 63 68 61 72 20 2a  any(const char *
5620: 7a 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f 6c 69  z, ...){.  va_li
5630: 73 74 20 61 70 3b 0a 20 20 63 68 61 72 20 2a 7a  st ap;.  char *z
5640: 32 3b 0a 20 20 69 66 28 20 63 67 69 5f 70 61 72  2;.  if( cgi_par
5650: 61 6d 65 74 65 72 28 7a 2c 30 29 21 3d 30 20 29  ameter(z,0)!=0 )
5660: 20 72 65 74 75 72 6e 20 31 3b 0a 20 20 76 61 5f   return 1;.  va_
5670: 73 74 61 72 74 28 61 70 2c 20 7a 29 3b 0a 20 20  start(ap, z);.  
5680: 77 68 69 6c 65 28 20 28 7a 32 20 3d 20 76 61 5f  while( (z2 = va_
5690: 61 72 67 28 61 70 2c 20 63 68 61 72 2a 29 29 21  arg(ap, char*))!
56a0: 3d 30 20 29 7b 0a 20 20 20 20 69 66 28 20 63 67  =0 ){.    if( cg
56b0: 69 5f 70 61 72 61 6d 65 74 65 72 28 7a 32 2c 30  i_parameter(z2,0
56c0: 29 21 3d 30 20 29 20 72 65 74 75 72 6e 20 31 3b  )!=0 ) return 1;
56d0: 0a 20 20 7d 0a 20 20 76 61 5f 65 6e 64 28 61 70  .  }.  va_end(ap
56e0: 29 3b 0a 20 20 72 65 74 75 72 6e 20 30 3b 0a 7d  );.  return 0;.}
56f0: 0a 0a 2f 2a 0a 2a 2a 20 52 65 74 75 72 6e 20 74  ../*.** Return t
5700: 72 75 65 20 69 66 20 61 6c 6c 20 6f 66 20 74 68  rue if all of th
5710: 65 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65  e query paramete
5720: 72 73 20 69 6e 20 74 68 65 20 61 72 67 75 6d 65  rs in the argume
5730: 6e 74 20 6c 69 73 74 0a 2a 2a 20 61 72 65 20 64  nt list.** are d
5740: 65 66 69 6e 65 64 2e 0a 2a 2f 0a 69 6e 74 20 63  efined..*/.int c
5750: 67 69 5f 61 6c 6c 28 63 6f 6e 73 74 20 63 68 61  gi_all(const cha
5760: 72 20 2a 7a 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61  r *z, ...){.  va
5770: 5f 6c 69 73 74 20 61 70 3b 0a 20 20 63 68 61 72  _list ap;.  char
5780: 20 2a 7a 32 3b 0a 20 20 69 66 28 20 63 67 69 5f   *z2;.  if( cgi_
5790: 70 61 72 61 6d 65 74 65 72 28 7a 2c 30 29 3d 3d  parameter(z,0)==
57a0: 30 20 29 20 72 65 74 75 72 6e 20 30 3b 0a 20 20  0 ) return 0;.  
57b0: 76 61 5f 73 74 61 72 74 28 61 70 2c 20 7a 29 3b  va_start(ap, z);
57c0: 0a 20 20 77 68 69 6c 65 28 20 28 7a 32 20 3d 20  .  while( (z2 = 
57d0: 76 61 5f 61 72 67 28 61 70 2c 20 63 68 61 72 2a  va_arg(ap, char*
57e0: 29 29 3d 3d 30 20 29 7b 0a 20 20 20 20 69 66 28  ))==0 ){.    if(
57f0: 20 63 67 69 5f 70 61 72 61 6d 65 74 65 72 28 7a   cgi_parameter(z
5800: 32 2c 30 29 3d 3d 30 20 29 20 72 65 74 75 72 6e  2,0)==0 ) return
5810: 20 30 3b 0a 20 20 7d 0a 20 20 76 61 5f 65 6e 64   0;.  }.  va_end
5820: 28 61 70 29 3b 0a 20 20 72 65 74 75 72 6e 20 31  (ap);.  return 1
5830: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 50 72 69 6e 74  ;.}../*.** Print
5840: 20 61 6c 6c 20 71 75 65 72 79 20 70 61 72 61 6d   all query param
5850: 65 74 65 72 73 20 6f 6e 20 73 74 61 6e 64 61 72  eters on standar
5860: 64 20 6f 75 74 70 75 74 2e 20 20 46 6f 72 6d 61  d output.  Forma
5870: 74 20 74 68 65 0a 2a 2a 20 70 61 72 61 6d 65 74  t the.** paramet
5880: 65 72 73 20 61 73 20 48 54 4d 4c 2e 20 20 54 68  ers as HTML.  Th
5890: 69 73 20 69 73 20 75 73 65 64 20 66 6f 72 20 74  is is used for t
58a0: 65 73 74 69 6e 67 20 61 6e 64 20 64 65 62 75 67  esting and debug
58b0: 67 69 6e 67 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67  ging..*/.void cg
58c0: 69 5f 70 72 69 6e 74 5f 61 6c 6c 28 76 6f 69 64  i_print_all(void
58d0: 29 7b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 63 67  ){.  int i;.  cg
58e0: 69 5f 70 61 72 61 6d 65 74 65 72 28 22 22 2c 22  i_parameter("","
58f0: 22 29 3b 20 20 2f 2a 20 46 6f 72 63 65 20 74 68  ");  /* Force th
5900: 65 20 70 61 72 61 6d 65 74 65 72 73 20 69 6e 74  e parameters int
5910: 6f 20 73 6f 72 74 65 64 20 6f 72 64 65 72 20 2a  o sorted order *
5920: 2f 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69 3c 6e  /.  for(i=0; i<n
5930: 55 73 65 64 51 50 3b 20 69 2b 2b 29 7b 0a 20 20  UsedQP; i++){.  
5940: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 73    cgi_printf("%s
5950: 20 3d 20 25 73 20 20 3c 62 72 20 2f 3e 5c 6e 22   = %s  <br />\n"
5960: 2c 0a 20 20 20 20 20 20 20 68 74 6d 6c 69 7a 65  ,.       htmlize
5970: 28 61 50 61 72 61 6d 51 50 5b 69 5d 2e 7a 4e 61  (aParamQP[i].zNa
5980: 6d 65 2c 20 2d 31 29 2c 20 68 74 6d 6c 69 7a 65  me, -1), htmlize
5990: 28 61 50 61 72 61 6d 51 50 5b 69 5d 2e 7a 56 61  (aParamQP[i].zVa
59a0: 6c 75 65 2c 20 2d 31 29 29 3b 0a 20 20 7d 0a 7d  lue, -1));.  }.}
59b0: 0a 0a 2f 2a 0a 2a 2a 20 57 72 69 74 65 20 48 54  ../*.** Write HT
59c0: 4d 4c 20 74 65 78 74 20 66 6f 72 20 61 6e 20 6f  ML text for an o
59d0: 70 74 69 6f 6e 20 6d 65 6e 75 20 74 6f 20 73 74  ption menu to st
59e0: 61 6e 64 61 72 64 20 6f 75 74 70 75 74 2e 20 20  andard output.  
59f0: 7a 50 61 72 61 6d 0a 2a 2a 20 69 73 20 74 68 65  zParam.** is the
5a00: 20 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72   query parameter
5a10: 20 74 68 61 74 20 74 68 65 20 6f 70 74 69 6f 6e   that the option
5a20: 20 6d 65 6e 75 20 73 65 74 73 2e 20 20 7a 44 66   menu sets.  zDf
5a30: 6c 74 20 69 73 20 74 68 65 0a 2a 2a 20 69 6e 69  lt is the.** ini
5a40: 74 69 61 6c 20 76 61 6c 75 65 20 6f 66 20 74 68  tial value of th
5a50: 65 20 6f 70 74 69 6f 6e 20 6d 65 6e 75 2e 20 20  e option menu.  
5a60: 41 64 64 69 74 69 6f 6e 20 61 72 67 75 6d 65 6e  Addition argumen
5a70: 74 73 20 61 72 65 20 6e 61 6d 65 2f 76 61 6c 75  ts are name/valu
5a80: 65 0a 2a 2a 20 70 61 69 72 73 20 74 68 61 74 20  e.** pairs that 
5a90: 64 65 66 69 6e 65 20 76 61 6c 75 65 73 20 6f 6e  define values on
5aa0: 20 74 68 65 20 6d 65 6e 75 2e 20 20 54 68 65 20   the menu.  The 
5ab0: 6c 69 73 74 20 69 73 20 74 65 72 6d 69 6e 61 74  list is terminat
5ac0: 65 64 20 77 69 74 68 0a 2a 2a 20 61 20 73 69 6e  ed with.** a sin
5ad0: 67 6c 65 20 4e 55 4c 4c 20 61 72 67 75 6d 65 6e  gle NULL argumen
5ae0: 74 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f 6f  t..*/.void cgi_o
5af0: 70 74 69 6f 6e 6d 65 6e 75 28 69 6e 74 20 69 6e  ptionmenu(int in
5b00: 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 50  , const char *zP
5b10: 2c 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 44  , const char *zD
5b20: 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f 6c 69 73  , ...){.  va_lis
5b30: 74 20 61 70 3b 0a 20 20 63 68 61 72 20 2a 7a 4e  t ap;.  char *zN
5b40: 61 6d 65 2c 20 2a 7a 56 61 6c 3b 0a 20 20 69 6e  ame, *zVal;.  in
5b50: 74 20 64 66 6c 74 53 65 65 6e 20 3d 20 30 3b 0a  t dfltSeen = 0;.
5b60: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a    cgi_printf("%*
5b70: 73 3c 73 65 6c 65 63 74 20 73 69 7a 65 3d 31 20  s<select size=1 
5b80: 6e 61 6d 65 3d 5c 22 25 73 5c 22 3e 5c 6e 22 2c  name=\"%s\">\n",
5b90: 20 69 6e 2c 20 22 22 2c 20 7a 50 29 3b 0a 20 20   in, "", zP);.  
5ba0: 76 61 5f 73 74 61 72 74 28 61 70 2c 20 7a 44 29  va_start(ap, zD)
5bb0: 3b 0a 20 20 77 68 69 6c 65 28 20 28 7a 4e 61 6d  ;.  while( (zNam
5bc0: 65 20 3d 20 76 61 5f 61 72 67 28 61 70 2c 20 63  e = va_arg(ap, c
5bd0: 68 61 72 2a 29 29 21 3d 30 20 26 26 20 28 7a 56  har*))!=0 && (zV
5be0: 61 6c 20 3d 20 76 61 5f 61 72 67 28 61 70 2c 20  al = va_arg(ap, 
5bf0: 63 68 61 72 2a 29 29 21 3d 30 20 29 7b 0a 20 20  char*))!=0 ){.  
5c00: 20 20 69 66 28 20 73 74 72 63 6d 70 28 7a 56 61    if( strcmp(zVa
5c10: 6c 2c 7a 44 29 3d 3d 30 20 29 7b 20 64 66 6c 74  l,zD)==0 ){ dflt
5c20: 53 65 65 6e 20 3d 20 31 3b 20 62 72 65 61 6b 3b  Seen = 1; break;
5c30: 20 7d 0a 20 20 7d 0a 20 20 76 61 5f 65 6e 64 28   }.  }.  va_end(
5c40: 61 70 29 3b 0a 20 20 69 66 28 20 21 64 66 6c 74  ap);.  if( !dflt
5c50: 53 65 65 6e 20 29 7b 0a 20 20 20 20 69 66 28 20  Seen ){.    if( 
5c60: 7a 44 5b 30 5d 20 29 7b 0a 20 20 20 20 20 20 63  zD[0] ){.      c
5c70: 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f  gi_printf("%*s<o
5c80: 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c 22 25 68  ption value=\"%h
5c90: 5c 22 20 73 65 6c 65 63 74 65 64 3e 25 68 3c 2f  \" selected>%h</
5ca0: 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20  option>\n",.    
5cb0: 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c 20 7a 44      in+2, "", zD
5cc0: 2c 20 7a 44 29 3b 0a 20 20 20 20 7d 65 6c 73 65  , zD);.    }else
5cd0: 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69 6e  {.      cgi_prin
5ce0: 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20 76  tf("%*s<option v
5cf0: 61 6c 75 65 3d 5c 22 5c 22 20 73 65 6c 65 63 74  alue=\"\" select
5d00: 65 64 3e 26 6e 62 73 70 3b 3c 2f 6f 70 74 69 6f  ed>&nbsp;</optio
5d10: 6e 3e 5c 6e 22 2c 20 69 6e 2b 32 2c 20 22 22 29  n>\n", in+2, "")
5d20: 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 76 61  ;.    }.  }.  va
5d30: 5f 73 74 61 72 74 28 61 70 2c 20 7a 44 29 3b 0a  _start(ap, zD);.
5d40: 20 20 77 68 69 6c 65 28 20 28 7a 4e 61 6d 65 20    while( (zName 
5d50: 3d 20 76 61 5f 61 72 67 28 61 70 2c 20 63 68 61  = va_arg(ap, cha
5d60: 72 2a 29 29 21 3d 30 20 26 26 20 28 7a 56 61 6c  r*))!=0 && (zVal
5d70: 20 3d 20 76 61 5f 61 72 67 28 61 70 2c 20 63 68   = va_arg(ap, ch
5d80: 61 72 2a 29 29 21 3d 30 20 29 7b 0a 20 20 20 20  ar*))!=0 ){.    
5d90: 69 66 28 20 7a 4e 61 6d 65 5b 30 5d 20 29 7b 0a  if( zName[0] ){.
5da0: 20 20 20 20 20 20 63 67 69 5f 70 72 69 6e 74 66        cgi_printf
5db0: 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 6c  ("%*s<option val
5dc0: 75 65 3d 5c 22 25 68 5c 22 25 73 3e 25 68 3c 2f  ue=\"%h\"%s>%h</
5dd0: 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20  option>\n",.    
5de0: 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c 0a 20 20      in+2, "",.  
5df0: 20 20 20 20 20 20 7a 56 61 6c 2c 0a 20 20 20 20        zVal,.    
5e00: 20 20 20 20 73 74 72 63 6d 70 28 7a 56 61 6c 2c      strcmp(zVal,
5e10: 20 7a 44 29 20 3f 20 22 22 20 3a 20 22 20 73 65   zD) ? "" : " se
5e20: 6c 65 63 74 65 64 22 2c 0a 20 20 20 20 20 20 20  lected",.       
5e30: 20 7a 4e 61 6d 65 0a 20 20 20 20 20 20 29 3b 0a   zName.      );.
5e40: 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20      }else{.     
5e50: 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73   cgi_printf("%*s
5e60: 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c 22  <option value=\"
5e70: 5c 22 25 73 3e 26 6e 62 73 70 3b 3c 2f 6f 70 74  \"%s>&nbsp;</opt
5e80: 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 20 20 20  ion>\n",.       
5e90: 20 69 6e 2b 32 2c 20 22 22 2c 0a 20 20 20 20 20   in+2, "",.     
5ea0: 20 20 20 73 74 72 63 6d 70 28 7a 56 61 6c 2c 20     strcmp(zVal, 
5eb0: 7a 44 29 20 3f 20 22 22 20 3a 20 22 20 73 65 6c  zD) ? "" : " sel
5ec0: 65 63 74 65 64 22 0a 20 20 20 20 20 20 29 3b 0a  ected".      );.
5ed0: 20 20 20 20 7d 0a 20 20 7d 0a 20 20 76 61 5f 65      }.  }.  va_e
5ee0: 6e 64 28 61 70 29 3b 0a 20 20 63 67 69 5f 70 72  nd(ap);.  cgi_pr
5ef0: 69 6e 74 66 28 22 25 2a 73 3c 2f 73 65 6c 65 63  intf("%*s</selec
5f00: 74 3e 5c 6e 22 2c 20 69 6e 2c 20 22 22 29 3b 0a  t>\n", in, "");.
5f10: 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 69 73 20 72 6f  }../*.** This ro
5f20: 75 74 69 6e 65 20 77 6f 72 6b 73 20 61 20 6c 6f  utine works a lo
5f30: 74 20 6c 69 6b 65 20 63 67 69 5f 6f 70 74 69 6f  t like cgi_optio
5f40: 6e 6d 65 6e 75 28 29 20 65 78 63 65 70 74 20 74  nmenu() except t
5f50: 68 61 74 20 74 68 65 20 6c 69 73 74 20 6f 66 0a  hat the list of.
5f60: 2a 2a 20 76 61 6c 75 65 73 20 69 73 20 63 6f 6e  ** values is con
5f70: 74 61 69 6e 65 64 20 69 6e 20 61 6e 20 61 72 72  tained in an arr
5f80: 61 79 2e 20 20 41 6c 73 6f 2c 20 74 68 65 20 76  ay.  Also, the v
5f90: 61 6c 75 65 73 20 61 72 65 20 6a 75 73 74 20 76  alues are just v
5fa0: 61 6c 75 65 73 2c 20 6e 6f 74 0a 2a 2a 20 6e 61  alues, not.** na
5fb0: 6d 65 2f 76 61 6c 75 65 20 70 61 69 72 73 20 61  me/value pairs a
5fc0: 73 20 69 6e 20 63 67 69 5f 6f 70 74 69 6f 6e 6d  s in cgi_optionm
5fd0: 65 6e 75 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69  enu..*/.void cgi
5fe0: 5f 76 5f 6f 70 74 69 6f 6e 6d 65 6e 75 28 0a 20  _v_optionmenu(. 
5ff0: 20 69 6e 74 20 69 6e 2c 20 20 20 20 20 20 20 20   int in,        
6000: 20 20 20 20 20 20 2f 2a 20 49 6e 64 65 6e 74 20        /* Indent 
6010: 62 79 20 74 68 69 73 20 61 6d 6f 75 6e 74 20 2a  by this amount *
6020: 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a  /.  const char *
6030: 7a 50 2c 20 20 20 20 20 20 2f 2a 20 54 68 65 20  zP,      /* The 
6040: 71 75 65 72 79 20 70 61 72 61 6d 65 74 65 72 20  query parameter 
6050: 6e 61 6d 65 20 2a 2f 0a 20 20 63 6f 6e 73 74 20  name */.  const 
6060: 63 68 61 72 20 2a 7a 44 2c 20 20 20 20 20 20 2f  char *zD,      /
6070: 2a 20 44 65 66 61 75 6c 74 20 76 61 6c 75 65 20  * Default value 
6080: 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20  */.  const char 
6090: 2a 2a 61 7a 20 20 20 20 20 20 2f 2a 20 4e 55 4c  **az      /* NUL
60a0: 4c 2d 74 65 72 6d 69 6e 61 74 65 64 20 6c 69 73  L-terminated lis
60b0: 74 20 6f 66 20 61 6c 6c 6f 77 65 64 20 76 61 6c  t of allowed val
60c0: 75 65 73 20 2a 2f 0a 29 7b 0a 20 20 63 6f 6e 73  ues */.){.  cons
60d0: 74 20 63 68 61 72 20 2a 7a 56 61 6c 3b 0a 20 20  t char *zVal;.  
60e0: 69 6e 74 20 69 3b 0a 20 20 63 67 69 5f 70 72 69  int i;.  cgi_pri
60f0: 6e 74 66 28 22 25 2a 73 3c 73 65 6c 65 63 74 20  ntf("%*s<select 
6100: 73 69 7a 65 3d 31 20 6e 61 6d 65 3d 5c 22 25 73  size=1 name=\"%s
6110: 5c 22 3e 5c 6e 22 2c 20 69 6e 2c 20 22 22 2c 20  \">\n", in, "", 
6120: 7a 50 29 3b 0a 20 20 66 6f 72 28 69 3d 30 3b 20  zP);.  for(i=0; 
6130: 61 7a 5b 69 5d 3b 20 69 2b 2b 29 7b 0a 20 20 20  az[i]; i++){.   
6140: 20 69 66 28 20 73 74 72 63 6d 70 28 61 7a 5b 69   if( strcmp(az[i
6150: 5d 2c 7a 44 29 3d 3d 30 20 29 20 62 72 65 61 6b  ],zD)==0 ) break
6160: 3b 0a 20 20 7d 0a 20 20 69 66 28 20 61 7a 5b 69  ;.  }.  if( az[i
6170: 5d 3d 3d 30 20 29 7b 0a 20 20 20 20 69 66 28 20  ]==0 ){.    if( 
6180: 7a 44 5b 30 5d 3d 3d 30 20 29 7b 0a 20 20 20 20  zD[0]==0 ){.    
6190: 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a    cgi_printf("%*
61a0: 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c  s<option value=\
61b0: 22 5c 22 20 73 65 6c 65 63 74 65 64 3e 26 6e 62  "\" selected>&nb
61c0: 73 70 3b 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c  sp;</option>\n",
61d0: 0a 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22  .       in+2, ""
61e0: 29 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20  );.    }else{.  
61f0: 20 20 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22      cgi_printf("
6200: 25 2a 73 3c 6f 70 74 69 6f 6e 20 76 61 6c 75 65  %*s<option value
6210: 3d 5c 22 25 68 5c 22 20 73 65 6c 65 63 74 65 64  =\"%h\" selected
6220: 3e 25 68 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c  >%h</option>\n",
6230: 0a 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22  .       in+2, ""
6240: 2c 20 7a 44 2c 20 7a 44 29 3b 0a 20 20 20 20 7d  , zD, zD);.    }
6250: 0a 20 20 7d 0a 20 20 77 68 69 6c 65 28 20 28 7a  .  }.  while( (z
6260: 56 61 6c 20 3d 20 2a 28 61 7a 2b 2b 29 29 21 3d  Val = *(az++))!=
6270: 30 20 20 29 7b 0a 20 20 20 20 69 66 28 20 7a 56  0  ){.    if( zV
6280: 61 6c 5b 30 5d 20 29 7b 0a 20 20 20 20 20 20 63  al[0] ){.      c
6290: 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f  gi_printf("%*s<o
62a0: 70 74 69 6f 6e 20 76 61 6c 75 65 3d 5c 22 25 68  ption value=\"%h
62b0: 5c 22 25 73 3e 25 68 3c 2f 6f 70 74 69 6f 6e 3e  \"%s>%h</option>
62c0: 5c 6e 22 2c 0a 20 20 20 20 20 20 20 20 69 6e 2b  \n",.        in+
62d0: 32 2c 20 22 22 2c 0a 20 20 20 20 20 20 20 20 7a  2, "",.        z
62e0: 56 61 6c 2c 0a 20 20 20 20 20 20 20 20 73 74 72  Val,.        str
62f0: 63 6d 70 28 7a 56 61 6c 2c 20 7a 44 29 20 3f 20  cmp(zVal, zD) ? 
6300: 22 22 20 3a 20 22 20 73 65 6c 65 63 74 65 64 22  "" : " selected"
6310: 2c 0a 20 20 20 20 20 20 20 20 7a 56 61 6c 0a 20  ,.        zVal. 
6320: 20 20 20 20 20 29 3b 0a 20 20 20 20 7d 65 6c 73       );.    }els
6330: 65 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69  e{.      cgi_pri
6340: 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20  ntf("%*s<option 
6350: 76 61 6c 75 65 3d 5c 22 5c 22 25 73 3e 26 6e 62  value=\"\"%s>&nb
6360: 73 70 3b 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c  sp;</option>\n",
6370: 0a 20 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22  .        in+2, "
6380: 22 2c 0a 20 20 20 20 20 20 20 20 73 74 72 63 6d  ",.        strcm
6390: 70 28 7a 56 61 6c 2c 20 7a 44 29 20 3f 20 22 22  p(zVal, zD) ? ""
63a0: 20 3a 20 22 20 73 65 6c 65 63 74 65 64 22 0a 20   : " selected". 
63b0: 20 20 20 20 20 29 3b 0a 20 20 20 20 7d 0a 20 20       );.    }.  
63c0: 7d 0a 20 20 63 67 69 5f 70 72 69 6e 74 66 28 22  }.  cgi_printf("
63d0: 25 2a 73 3c 2f 73 65 6c 65 63 74 3e 5c 6e 22 2c  %*s</select>\n",
63e0: 20 69 6e 2c 20 22 22 29 3b 0a 7d 0a 0a 2f 2a 0a   in, "");.}../*.
63f0: 2a 2a 20 54 68 69 73 20 72 6f 75 74 69 6e 65 20  ** This routine 
6400: 77 6f 72 6b 73 20 61 20 6c 6f 74 20 6c 69 6b 65  works a lot like
6410: 20 63 67 69 5f 76 5f 6f 70 74 69 6f 6e 6d 65 6e   cgi_v_optionmen
6420: 75 28 29 20 65 78 63 65 70 74 20 74 68 61 74 20  u() except that 
6430: 74 68 65 20 6c 69 73 74 0a 2a 2a 20 69 73 20 61  the list.** is a
6440: 20 6c 69 73 74 20 6f 66 20 70 61 69 72 73 2e 20   list of pairs. 
6450: 20 54 68 65 20 66 69 72 73 74 20 65 6c 65 6d 65   The first eleme
6460: 6e 74 20 6f 66 20 65 61 63 68 20 70 61 69 72 20  nt of each pair 
6470: 69 73 20 74 68 65 20 76 61 6c 75 65 20 75 73 65  is the value use
6480: 64 0a 2a 2a 20 69 6e 74 65 72 6e 61 6c 6c 79 20  d.** internally 
6490: 61 6e 64 20 74 68 65 20 73 65 63 6f 6e 64 20 65  and the second e
64a0: 6c 65 6d 65 6e 74 20 69 73 20 74 68 65 20 76 61  lement is the va
64b0: 6c 75 65 20 64 69 73 70 6c 61 79 65 64 20 74 6f  lue displayed to
64c0: 20 74 68 65 20 75 73 65 72 2e 0a 2a 2f 0a 76 6f   the user..*/.vo
64d0: 69 64 20 63 67 69 5f 76 5f 6f 70 74 69 6f 6e 6d  id cgi_v_optionm
64e0: 65 6e 75 32 28 0a 20 20 69 6e 74 20 69 6e 2c 20  enu2(.  int in, 
64f0: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
6500: 49 6e 64 65 6e 74 20 62 79 20 74 68 69 73 20 61  Indent by this a
6510: 6d 6f 75 6e 74 20 2a 2f 0a 20 20 63 6f 6e 73 74  mount */.  const
6520: 20 63 68 61 72 20 2a 7a 50 2c 20 20 20 20 20 20   char *zP,      
6530: 2f 2a 20 54 68 65 20 71 75 65 72 79 20 70 61 72  /* The query par
6540: 61 6d 65 74 65 72 20 6e 61 6d 65 20 2a 2f 0a 20  ameter name */. 
6550: 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 44 2c   const char *zD,
6560: 20 20 20 20 20 20 2f 2a 20 44 65 66 61 75 6c 74        /* Default
6570: 20 76 61 6c 75 65 20 2a 2f 0a 20 20 63 6f 6e 73   value */.  cons
6580: 74 20 63 68 61 72 20 2a 2a 61 7a 20 20 20 20 20  t char **az     
6590: 20 2f 2a 20 4e 55 4c 4c 2d 74 65 72 6d 69 6e 61   /* NULL-termina
65a0: 74 65 64 20 6c 69 73 74 20 6f 66 20 61 6c 6c 6f  ted list of allo
65b0: 77 65 64 20 76 61 6c 75 65 73 20 2a 2f 0a 29 7b  wed values */.){
65c0: 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a  .  const char *z
65d0: 56 61 6c 3b 0a 20 20 69 6e 74 20 69 3b 0a 20 20  Val;.  int i;.  
65e0: 63 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c  cgi_printf("%*s<
65f0: 73 65 6c 65 63 74 20 73 69 7a 65 3d 31 20 6e 61  select size=1 na
6600: 6d 65 3d 5c 22 25 73 5c 22 3e 5c 6e 22 2c 20 69  me=\"%s\">\n", i
6610: 6e 2c 20 22 22 2c 20 7a 50 29 3b 0a 20 20 66 6f  n, "", zP);.  fo
6620: 72 28 69 3d 30 3b 20 61 7a 5b 69 5d 3b 20 69 2b  r(i=0; az[i]; i+
6630: 3d 32 29 7b 0a 20 20 20 20 69 66 28 20 73 74 72  =2){.    if( str
6640: 63 6d 70 28 61 7a 5b 69 5d 2c 7a 44 29 3d 3d 30  cmp(az[i],zD)==0
6650: 20 29 20 62 72 65 61 6b 3b 0a 20 20 7d 0a 20 20   ) break;.  }.  
6660: 69 66 28 20 61 7a 5b 69 5d 3d 3d 30 20 29 7b 0a  if( az[i]==0 ){.
6670: 20 20 20 20 69 66 28 20 7a 44 5b 30 5d 3d 3d 30      if( zD[0]==0
6680: 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72   ){.      cgi_pr
6690: 69 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e  intf("%*s<option
66a0: 20 76 61 6c 75 65 3d 5c 22 5c 22 20 73 65 6c 65   value=\"\" sele
66b0: 63 74 65 64 3e 26 6e 62 73 70 3b 3c 2f 6f 70 74  cted>&nbsp;</opt
66c0: 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 20 20 20  ion>\n",.       
66d0: 69 6e 2b 32 2c 20 22 22 29 3b 0a 20 20 20 20 7d  in+2, "");.    }
66e0: 65 6c 73 65 7b 0a 20 20 20 20 20 20 63 67 69 5f  else{.      cgi_
66f0: 70 72 69 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69  printf("%*s<opti
6700: 6f 6e 20 76 61 6c 75 65 3d 5c 22 25 68 5c 22 20  on value=\"%h\" 
6710: 73 65 6c 65 63 74 65 64 3e 25 68 3c 2f 6f 70 74  selected>%h</opt
6720: 69 6f 6e 3e 5c 6e 22 2c 0a 20 20 20 20 20 20 20  ion>\n",.       
6730: 69 6e 2b 32 2c 20 22 22 2c 20 7a 44 2c 20 7a 44  in+2, "", zD, zD
6740: 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 77  );.    }.  }.  w
6750: 68 69 6c 65 28 20 28 7a 56 61 6c 20 3d 20 2a 28  hile( (zVal = *(
6760: 61 7a 2b 2b 29 29 21 3d 30 20 20 29 7b 0a 20 20  az++))!=0  ){.  
6770: 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e    const char *zN
6780: 61 6d 65 20 3d 20 2a 28 61 7a 2b 2b 29 3b 0a 20  ame = *(az++);. 
6790: 20 20 20 69 66 28 20 7a 4e 61 6d 65 5b 30 5d 20     if( zName[0] 
67a0: 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 70 72 69  ){.      cgi_pri
67b0: 6e 74 66 28 22 25 2a 73 3c 6f 70 74 69 6f 6e 20  ntf("%*s<option 
67c0: 76 61 6c 75 65 3d 5c 22 25 68 5c 22 25 73 3e 25  value=\"%h\"%s>%
67d0: 68 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20  h</option>\n",. 
67e0: 20 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c         in+2, "",
67f0: 0a 20 20 20 20 20 20 20 20 7a 56 61 6c 2c 0a 20  .        zVal,. 
6800: 20 20 20 20 20 20 20 73 74 72 63 6d 70 28 7a 56         strcmp(zV
6810: 61 6c 2c 20 7a 44 29 20 3f 20 22 22 20 3a 20 22  al, zD) ? "" : "
6820: 20 73 65 6c 65 63 74 65 64 22 2c 0a 20 20 20 20   selected",.    
6830: 20 20 20 20 7a 4e 61 6d 65 0a 20 20 20 20 20 20      zName.      
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 25 73 3e 26 6e 62 73 70 3b  =\"%h\"%s>&nbsp;
6880: 3c 2f 6f 70 74 69 6f 6e 3e 5c 6e 22 2c 0a 20 20  </option>\n",.  
6890: 20 20 20 20 20 20 69 6e 2b 32 2c 20 22 22 2c 0a        in+2, "",.
68a0: 20 20 20 20 20 20 20 20 7a 56 61 6c 2c 0a 20 20          zVal,.  
68b0: 20 20 20 20 20 20 73 74 72 63 6d 70 28 7a 56 61        strcmp(zVa
68c0: 6c 2c 20 7a 44 29 20 3f 20 22 22 20 3a 20 22 20  l, zD) ? "" : " 
68d0: 73 65 6c 65 63 74 65 64 22 0a 20 20 20 20 20 20  selected".      
68e0: 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20 63  );.    }.  }.  c
68f0: 67 69 5f 70 72 69 6e 74 66 28 22 25 2a 73 3c 2f  gi_printf("%*s</
6900: 73 65 6c 65 63 74 3e 5c 6e 22 2c 20 69 6e 2c 20  select>\n", in, 
6910: 22 22 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68  "");.}../*.** Th
6920: 69 73 20 72 6f 75 74 69 6e 65 20 77 6f 72 6b 73  is routine works
6930: 20 6c 69 6b 65 20 22 70 72 69 6e 74 66 22 20 65   like "printf" e
6940: 78 63 65 70 74 20 74 68 61 74 20 69 74 20 68 61  xcept that it ha
6950: 73 20 74 68 65 0a 2a 2a 20 65 78 74 72 61 20 66  s the.** extra f
6960: 6f 72 6d 61 74 74 69 6e 67 20 63 61 70 61 62 69  ormatting capabi
6970: 6c 69 74 69 65 73 20 73 75 63 68 20 61 73 20 25  lities such as %
6980: 68 20 61 6e 64 20 25 74 2e 0a 2a 2f 0a 76 6f 69  h and %t..*/.voi
6990: 64 20 63 67 69 5f 70 72 69 6e 74 66 28 63 6f 6e  d cgi_printf(con
69a0: 73 74 20 63 68 61 72 20 2a 7a 46 6f 72 6d 61 74  st char *zFormat
69b0: 2c 20 2e 2e 2e 29 7b 0a 20 20 76 61 5f 6c 69 73  , ...){.  va_lis
69c0: 74 20 61 70 3b 0a 20 20 76 61 5f 73 74 61 72 74  t ap;.  va_start
69d0: 28 61 70 2c 7a 46 6f 72 6d 61 74 29 3b 0a 20 20  (ap,zFormat);.  
69e0: 76 78 70 72 69 6e 74 66 28 26 63 67 69 43 6f 6e  vxprintf(&cgiCon
69f0: 74 65 6e 74 2c 7a 46 6f 72 6d 61 74 2c 61 70 29  tent,zFormat,ap)
6a00: 3b 0a 20 20 76 61 5f 65 6e 64 28 61 70 29 3b 0a  ;.  va_end(ap);.
6a10: 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 69 73 20 72 6f  }../*.** This ro
6a20: 75 74 69 6e 65 20 77 6f 72 6b 73 20 6c 69 6b 65  utine works like
6a30: 20 22 76 70 72 69 6e 74 66 22 20 65 78 63 65 70   "vprintf" excep
6a40: 74 20 74 68 61 74 20 69 74 20 68 61 73 20 74 68  t that it has th
6a50: 65 0a 2a 2a 20 65 78 74 72 61 20 66 6f 72 6d 61  e.** extra forma
6a60: 74 74 69 6e 67 20 63 61 70 61 62 69 6c 69 74 69  tting capabiliti
6a70: 65 73 20 73 75 63 68 20 61 73 20 25 68 20 61 6e  es such as %h an
6a80: 64 20 25 74 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67  d %t..*/.void cg
6a90: 69 5f 76 70 72 69 6e 74 66 28 63 6f 6e 73 74 20  i_vprintf(const 
6aa0: 63 68 61 72 20 2a 7a 46 6f 72 6d 61 74 2c 20 76  char *zFormat, v
6ab0: 61 5f 6c 69 73 74 20 61 70 29 7b 0a 20 20 76 78  a_list ap){.  vx
6ac0: 70 72 69 6e 74 66 28 26 63 67 69 43 6f 6e 74 65  printf(&cgiConte
6ad0: 6e 74 2c 7a 46 6f 72 6d 61 74 2c 61 70 29 3b 0a  nt,zFormat,ap);.
6ae0: 7d 0a 0a 0a 2f 2a 0a 2a 2a 20 53 65 6e 64 20 61  }.../*.** Send a
6af0: 20 72 65 70 6c 79 20 69 6e 64 69 63 61 74 69 6e   reply indicatin
6b00: 67 20 74 68 61 74 20 74 68 65 20 48 54 54 50 20  g that the HTTP 
6b10: 72 65 71 75 65 73 74 20 77 61 73 20 6d 61 6c 66  request was malf
6b20: 6f 72 6d 65 64 0a 2a 2f 0a 73 74 61 74 69 63 20  ormed.*/.static 
6b30: 76 6f 69 64 20 6d 61 6c 66 6f 72 6d 65 64 5f 72  void malformed_r
6b40: 65 71 75 65 73 74 28 76 6f 69 64 29 7b 0a 20 20  equest(void){.  
6b50: 63 67 69 5f 73 65 74 5f 73 74 61 74 75 73 28 35  cgi_set_status(5
6b60: 30 31 2c 20 22 4e 6f 74 20 49 6d 70 6c 65 6d 65  01, "Not Impleme
6b70: 6e 74 65 64 22 29 3b 0a 20 20 63 67 69 5f 70 72  nted");.  cgi_pr
6b80: 69 6e 74 66 28 0a 20 20 20 20 22 3c 68 74 6d 6c  intf(.    "<html
6b90: 3e 3c 62 6f 64 79 3e 55 6e 72 65 63 6f 67 6e 69  ><body>Unrecogni
6ba0: 7a 65 64 20 48 54 54 50 20 52 65 71 75 65 73 74  zed HTTP Request
6bb0: 3c 2f 62 6f 64 79 3e 3c 2f 68 74 6d 6c 3e 5c 6e  </body></html>\n
6bc0: 22 0a 20 20 29 3b 0a 20 20 63 67 69 5f 72 65 70  ".  );.  cgi_rep
6bd0: 6c 79 28 29 3b 0a 20 20 65 78 69 74 28 30 29 3b  ly();.  exit(0);
6be0: 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 50 61 6e 69 63 20  .}../*.** Panic 
6bf0: 61 6e 64 20 64 69 65 20 77 68 69 6c 65 20 70 72  and die while pr
6c00: 6f 63 65 73 73 69 6e 67 20 61 20 77 65 62 70 61  ocessing a webpa
6c10: 67 65 2e 0a 2a 2f 0a 76 6f 69 64 20 63 67 69 5f  ge..*/.void cgi_
6c20: 70 61 6e 69 63 28 63 6f 6e 73 74 20 63 68 61 72  panic(const char
6c30: 20 2a 7a 46 6f 72 6d 61 74 2c 20 2e 2e 2e 29 7b   *zFormat, ...){
6c40: 0a 20 20 76 61 5f 6c 69 73 74 20 61 70 3b 0a 20  .  va_list ap;. 
6c50: 20 63 67 69 5f 72 65 73 65 74 5f 63 6f 6e 74 65   cgi_reset_conte
6c60: 6e 74 28 29 3b 0a 20 20 63 67 69 5f 73 65 74 5f  nt();.  cgi_set_
6c70: 73 74 61 74 75 73 28 35 30 30 2c 20 22 49 6e 74  status(500, "Int
6c80: 65 72 6e 61 6c 20 53 65 72 76 65 72 20 45 72 72  ernal Server Err
6c90: 6f 72 22 29 3b 0a 20 20 63 67 69 5f 70 72 69 6e  or");.  cgi_prin
6ca0: 74 66 28 0a 20 20 20 20 22 3c 68 74 6d 6c 3e 3c  tf(.    "<html><
6cb0: 62 6f 64 79 3e 3c 68 31 3e 49 6e 74 65 72 6e 61  body><h1>Interna
6cc0: 6c 20 53 65 72 76 65 72 20 45 72 72 6f 72 3c 2f  l Server Error</
6cd0: 68 31 3e 5c 6e 22 0a 20 20 20 20 22 3c 70 6c 61  h1>\n".    "<pla
6ce0: 69 6e 74 65 78 74 3e 22 0a 20 20 29 3b 0a 20 20  intext>".  );.  
6cf0: 76 61 5f 73 74 61 72 74 28 61 70 2c 20 7a 46 6f  va_start(ap, zFo
6d00: 72 6d 61 74 29 3b 0a 20 20 76 78 70 72 69 6e 74  rmat);.  vxprint
6d10: 66 28 26 63 67 69 43 6f 6e 74 65 6e 74 2c 7a 46  f(&cgiContent,zF
6d20: 6f 72 6d 61 74 2c 61 70 29 3b 0a 20 20 76 61 5f  ormat,ap);.  va_
6d30: 65 6e 64 28 61 70 29 3b 0a 20 20 63 67 69 5f 72  end(ap);.  cgi_r
6d40: 65 70 6c 79 28 29 3b 0a 20 20 65 78 69 74 28 31  eply();.  exit(1
6d50: 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 52 65 6d 6f  );.}../*.** Remo
6d60: 76 65 20 74 68 65 20 66 69 72 73 74 20 73 70 61  ve the first spa
6d70: 63 65 2d 64 65 6c 69 6d 69 74 65 64 20 74 6f 6b  ce-delimited tok
6d80: 65 6e 20 66 72 6f 6d 20 61 20 73 74 72 69 6e 67  en from a string
6d90: 20 61 6e 64 20 72 65 74 75 72 6e 0a 2a 2a 20 61   and return.** a
6da0: 20 70 6f 69 6e 74 65 72 20 74 6f 20 69 74 2e 20   pointer to it. 
6db0: 20 41 64 64 20 61 20 4e 55 4c 4c 20 74 6f 20 74   Add a NULL to t
6dc0: 68 65 20 73 74 72 69 6e 67 20 74 6f 20 74 65 72  he string to ter
6dd0: 6d 69 6e 61 74 65 20 74 68 65 20 74 6f 6b 65 6e  minate the token
6de0: 2e 0a 2a 2a 20 4d 61 6b 65 20 2a 7a 4c 65 66 74  ..** Make *zLeft
6df0: 4f 76 65 72 20 70 6f 69 6e 74 20 74 6f 20 74 68  Over point to th
6e00: 65 20 73 74 61 72 74 20 6f 66 20 74 68 65 20 6e  e start of the n
6e10: 65 78 74 20 74 6f 6b 65 6e 2e 0a 2a 2f 0a 73 74  ext token..*/.st
6e20: 61 74 69 63 20 63 68 61 72 20 2a 65 78 74 72 61  atic char *extra
6e30: 63 74 5f 74 6f 6b 65 6e 28 63 68 61 72 20 2a 7a  ct_token(char *z
6e40: 49 6e 70 75 74 2c 20 63 68 61 72 20 2a 2a 7a 4c  Input, char **zL
6e50: 65 66 74 4f 76 65 72 29 7b 0a 20 20 63 68 61 72  eftOver){.  char
6e60: 20 2a 7a 52 65 73 75 6c 74 20 3d 20 30 3b 0a 20   *zResult = 0;. 
6e70: 20 69 66 28 20 7a 49 6e 70 75 74 3d 3d 30 20 29   if( zInput==0 )
6e80: 7b 0a 20 20 20 20 69 66 28 20 7a 4c 65 66 74 4f  {.    if( zLeftO
6e90: 76 65 72 20 29 20 2a 7a 4c 65 66 74 4f 76 65 72  ver ) *zLeftOver
6ea0: 20 3d 20 30 3b 0a 20 20 20 20 72 65 74 75 72 6e   = 0;.    return
6eb0: 20 30 3b 0a 20 20 7d 0a 20 20 77 68 69 6c 65 28   0;.  }.  while(
6ec0: 20 69 73 73 70 61 63 65 28 2a 7a 49 6e 70 75 74   isspace(*zInput
6ed0: 29 20 29 7b 20 7a 49 6e 70 75 74 2b 2b 3b 20 7d  ) ){ zInput++; }
6ee0: 0a 20 20 7a 52 65 73 75 6c 74 20 3d 20 7a 49 6e  .  zResult = zIn
6ef0: 70 75 74 3b 0a 20 20 77 68 69 6c 65 28 20 2a 7a  put;.  while( *z
6f00: 49 6e 70 75 74 20 26 26 20 21 69 73 73 70 61 63  Input && !isspac
6f10: 65 28 2a 7a 49 6e 70 75 74 29 20 29 7b 20 7a 49  e(*zInput) ){ zI
6f20: 6e 70 75 74 2b 2b 3b 20 7d 0a 20 20 69 66 28 20  nput++; }.  if( 
6f30: 2a 7a 49 6e 70 75 74 20 29 7b 0a 20 20 20 20 2a  *zInput ){.    *
6f40: 7a 49 6e 70 75 74 20 3d 20 30 3b 0a 20 20 20 20  zInput = 0;.    
6f50: 7a 49 6e 70 75 74 2b 2b 3b 0a 20 20 20 20 77 68  zInput++;.    wh
6f60: 69 6c 65 28 20 69 73 73 70 61 63 65 28 2a 7a 49  ile( isspace(*zI
6f70: 6e 70 75 74 29 20 29 7b 20 7a 49 6e 70 75 74 2b  nput) ){ zInput+
6f80: 2b 3b 20 7d 0a 20 20 7d 0a 20 20 69 66 28 20 7a  +; }.  }.  if( z
6f90: 4c 65 66 74 4f 76 65 72 20 29 7b 20 2a 7a 4c 65  LeftOver ){ *zLe
6fa0: 66 74 4f 76 65 72 20 3d 20 7a 49 6e 70 75 74 3b  ftOver = zInput;
6fb0: 20 7d 0a 20 20 72 65 74 75 72 6e 20 7a 52 65 73   }.  return zRes
6fc0: 75 6c 74 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68  ult;.}../*.** Th
6fd0: 69 73 20 72 6f 75 74 69 6e 65 20 68 61 6e 64 6c  is routine handl
6fe0: 65 73 20 61 20 73 69 6e 67 6c 65 20 48 54 54 50  es a single HTTP
6ff0: 20 72 65 71 75 65 73 74 20 77 68 69 63 68 20 69   request which i
7000: 73 20 63 6f 6d 69 6e 67 20 69 6e 20 6f 6e 0a 2a  s coming in on.*
7010: 2a 20 73 74 61 6e 64 61 72 64 20 69 6e 70 75 74  * standard input
7020: 20 61 6e 64 20 77 68 69 63 68 20 72 65 70 6c 69   and which repli
7030: 65 73 20 6f 6e 20 73 74 61 6e 64 61 72 64 20 6f  es on standard o
7040: 75 74 70 75 74 2e 0a 2a 2a 0a 2a 2a 20 54 68 65  utput..**.** The
7050: 20 48 54 54 50 20 72 65 71 75 65 73 74 20 69 73   HTTP request is
7060: 20 72 65 61 64 20 66 72 6f 6d 20 73 74 61 6e 64   read from stand
7070: 61 72 64 20 69 6e 70 75 74 20 61 6e 64 20 69 73  ard input and is
7080: 20 75 73 65 64 20 74 6f 20 69 6e 69 74 69 61 6c   used to initial
7090: 69 7a 65 0a 2a 2a 20 65 6e 76 69 72 6f 6e 6d 65  ize.** environme
70a0: 6e 74 20 76 61 72 69 61 62 6c 65 73 20 61 73 20  nt variables as 
70b0: 70 65 72 20 43 47 49 2e 20 20 54 68 65 20 63 67  per CGI.  The cg
70c0: 69 5f 69 6e 69 74 28 29 20 72 6f 75 74 69 6e 65  i_init() routine
70d0: 20 74 6f 20 63 6f 6d 70 6c 65 74 65 0a 2a 2a 20   to complete.** 
70e0: 74 68 65 20 73 65 74 75 70 2e 20 20 4f 6e 63 65  the setup.  Once
70f0: 20 61 6c 6c 20 74 68 65 20 73 65 74 75 70 20 69   all the setup i
7100: 73 20 66 69 6e 69 73 68 65 64 2c 20 74 68 69 73  s finished, this
7110: 20 70 72 6f 63 65 64 75 72 65 20 72 65 74 75 72   procedure retur
7120: 6e 73 0a 2a 2a 20 61 6e 64 20 73 75 62 73 65 71  ns.** and subseq
7130: 75 65 6e 74 20 63 6f 64 65 20 68 61 6e 64 6c 65  uent code handle
7140: 73 20 74 68 65 20 61 63 74 75 61 6c 20 67 65 6e  s the actual gen
7150: 65 72 61 74 69 6f 6e 20 6f 66 20 74 68 65 20 77  eration of the w
7160: 65 62 70 61 67 65 2e 0a 2a 2f 0a 76 6f 69 64 20  ebpage..*/.void 
7170: 63 67 69 5f 68 61 6e 64 6c 65 5f 68 74 74 70 5f  cgi_handle_http_
7180: 72 65 71 75 65 73 74 28 76 6f 69 64 29 7b 0a 20  request(void){. 
7190: 20 63 68 61 72 20 2a 7a 2c 20 2a 7a 54 6f 6b 65   char *z, *zToke
71a0: 6e 3b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 73 74  n;.  int i;.  st
71b0: 72 75 63 74 20 73 6f 63 6b 61 64 64 72 5f 69 6e  ruct sockaddr_in
71c0: 20 72 65 6d 6f 74 65 4e 61 6d 65 3b 0a 20 20 73   remoteName;.  s
71d0: 69 7a 65 5f 74 20 73 69 7a 65 20 3d 20 73 69 7a  ize_t size = siz
71e0: 65 6f 66 28 73 74 72 75 63 74 20 73 6f 63 6b 61  eof(struct socka
71f0: 64 64 72 5f 69 6e 29 3b 0a 20 20 63 68 61 72 20  ddr_in);.  char 
7200: 7a 4c 69 6e 65 5b 32 30 30 30 5d 3b 20 20 20 20  zLine[2000];    
7210: 20 2f 2a 20 41 20 73 69 6e 67 6c 65 20 6c 69 6e   /* A single lin
7220: 65 20 6f 66 20 69 6e 70 75 74 2e 20 2a 2f 0a 0a  e of input. */..
7230: 20 20 66 75 6c 6c 48 74 74 70 52 65 70 6c 79 20    fullHttpReply 
7240: 3d 20 31 3b 0a 20 20 69 66 28 20 66 67 65 74 73  = 1;.  if( fgets
7250: 28 7a 4c 69 6e 65 2c 20 73 69 7a 65 6f 66 28 7a  (zLine, sizeof(z
7260: 4c 69 6e 65 29 2c 20 73 74 64 69 6e 29 3d 3d 30  Line), stdin)==0
7270: 20 29 7b 0a 20 20 20 20 6d 61 6c 66 6f 72 6d 65   ){.    malforme
7280: 64 5f 72 65 71 75 65 73 74 28 29 3b 0a 20 20 7d  d_request();.  }
7290: 0a 20 20 7a 54 6f 6b 65 6e 20 3d 20 65 78 74 72  .  zToken = extr
72a0: 61 63 74 5f 74 6f 6b 65 6e 28 7a 4c 69 6e 65 2c  act_token(zLine,
72b0: 20 26 7a 29 3b 0a 20 20 69 66 28 20 7a 54 6f 6b   &z);.  if( zTok
72c0: 65 6e 3d 3d 30 20 29 7b 0a 20 20 20 20 6d 61 6c  en==0 ){.    mal
72d0: 66 6f 72 6d 65 64 5f 72 65 71 75 65 73 74 28 29  formed_request()
72e0: 3b 0a 20 20 7d 0a 20 20 69 66 28 20 73 74 72 63  ;.  }.  if( strc
72f0: 6d 70 28 7a 54 6f 6b 65 6e 2c 22 47 45 54 22 29  mp(zToken,"GET")
7300: 21 3d 30 20 26 26 20 73 74 72 63 6d 70 28 7a 54  !=0 && strcmp(zT
7310: 6f 6b 65 6e 2c 22 50 4f 53 54 22 29 21 3d 30 0a  oken,"POST")!=0.
7320: 20 20 20 20 20 20 26 26 20 73 74 72 63 6d 70 28        && strcmp(
7330: 7a 54 6f 6b 65 6e 2c 22 48 45 41 44 22 29 21 3d  zToken,"HEAD")!=
7340: 30 20 29 7b 0a 20 20 20 20 6d 61 6c 66 6f 72 6d  0 ){.    malform
7350: 65 64 5f 72 65 71 75 65 73 74 28 29 3b 0a 20 20  ed_request();.  
7360: 7d 0a 20 20 63 67 69 5f 73 65 74 65 6e 76 28 22  }.  cgi_setenv("
7370: 47 41 54 45 57 41 59 5f 49 4e 54 45 52 46 41 43  GATEWAY_INTERFAC
7380: 45 22 2c 22 43 47 49 2f 31 2e 30 22 29 3b 0a 20  E","CGI/1.0");. 
7390: 20 63 67 69 5f 73 65 74 65 6e 76 28 22 52 45 51   cgi_setenv("REQ
73a0: 55 45 53 54 5f 4d 45 54 48 4f 44 22 2c 7a 54 6f  UEST_METHOD",zTo
73b0: 6b 65 6e 29 3b 0a 20 20 7a 54 6f 6b 65 6e 20 3d  ken);.  zToken =
73c0: 20 65 78 74 72 61 63 74 5f 74 6f 6b 65 6e 28 7a   extract_token(z
73d0: 2c 20 26 7a 29 3b 0a 20 20 69 66 28 20 7a 54 6f  , &z);.  if( zTo
73e0: 6b 65 6e 3d 3d 30 20 29 7b 0a 20 20 20 20 6d 61  ken==0 ){.    ma
73f0: 6c 66 6f 72 6d 65 64 5f 72 65 71 75 65 73 74 28  lformed_request(
7400: 29 3b 0a 20 20 7d 0a 20 20 63 67 69 5f 73 65 74  );.  }.  cgi_set
7410: 65 6e 76 28 22 52 45 51 55 45 53 54 5f 55 52 49  env("REQUEST_URI
7420: 22 2c 20 7a 54 6f 6b 65 6e 29 3b 0a 20 20 66 6f  ", zToken);.  fo
7430: 72 28 69 3d 30 3b 20 7a 54 6f 6b 65 6e 5b 69 5d  r(i=0; zToken[i]
7440: 20 26 26 20 7a 54 6f 6b 65 6e 5b 69 5d 21 3d 27   && zToken[i]!='
7450: 3f 27 3b 20 69 2b 2b 29 7b 7d 0a 20 20 69 66 28  ?'; i++){}.  if(
7460: 20 7a 54 6f 6b 65 6e 5b 69 5d 20 29 20 7a 54 6f   zToken[i] ) zTo
7470: 6b 65 6e 5b 69 2b 2b 5d 20 3d 20 30 3b 0a 20 20  ken[i++] = 0;.  
7480: 63 67 69 5f 73 65 74 65 6e 76 28 22 50 41 54 48  cgi_setenv("PATH
7490: 5f 49 4e 46 4f 22 2c 20 7a 54 6f 6b 65 6e 29 3b  _INFO", zToken);
74a0: 0a 20 20 63 67 69 5f 73 65 74 65 6e 76 28 22 51  .  cgi_setenv("Q
74b0: 55 45 52 59 5f 53 54 52 49 4e 47 22 2c 20 26 7a  UERY_STRING", &z
74c0: 54 6f 6b 65 6e 5b 69 5d 29 3b 0a 20 20 69 66 28  Token[i]);.  if(
74d0: 20 67 65 74 70 65 65 72 6e 61 6d 65 28 66 69 6c   getpeername(fil
74e0: 65 6e 6f 28 73 74 64 69 6e 29 2c 20 28 73 74 72  eno(stdin), (str
74f0: 75 63 74 20 73 6f 63 6b 61 64 64 72 2a 29 26 72  uct sockaddr*)&r
7500: 65 6d 6f 74 65 4e 61 6d 65 2c 20 28 73 6f 63 6b  emoteName, (sock
7510: 6c 65 6e 5f 74 2a 29 26 73 69 7a 65 29 3e 3d 30  len_t*)&size)>=0
7520: 20 29 7b 0a 20 20 20 20 63 68 61 72 20 2a 7a 49   ){.    char *zI
7530: 70 41 64 64 72 20 3d 20 69 6e 65 74 5f 6e 74 6f  pAddr = inet_nto
7540: 61 28 72 65 6d 6f 74 65 4e 61 6d 65 2e 73 69 6e  a(remoteName.sin
7550: 5f 61 64 64 72 29 3b 0a 20 20 20 20 63 67 69 5f  _addr);.    cgi_
7560: 73 65 74 65 6e 76 28 22 52 45 4d 4f 54 45 5f 41  setenv("REMOTE_A
7570: 44 44 52 22 2c 20 7a 49 70 41 64 64 72 29 3b 0a  DDR", zIpAddr);.
7580: 0a 20 20 20 20 2f 2a 20 53 65 74 20 74 68 65 20  .    /* Set the 
7590: 47 6c 6f 62 61 6c 2e 7a 49 70 41 64 64 72 20 76  Global.zIpAddr v
75a0: 61 72 69 61 62 6c 65 20 74 6f 20 74 68 65 20 73  ariable to the s
75b0: 65 72 76 65 72 20 77 65 20 61 72 65 20 74 61 6c  erver we are tal
75c0: 6b 69 6e 67 20 74 6f 2e 0a 20 20 20 20 2a 2a 20  king to..    ** 
75d0: 54 68 69 73 20 69 73 20 75 73 65 64 20 74 6f 20  This is used to 
75e0: 70 6f 70 75 6c 61 74 65 20 74 68 65 20 69 70 61  populate the ipa
75f0: 64 64 72 20 63 6f 6c 75 6d 6e 20 6f 66 20 74 68  ddr column of th
7600: 65 20 72 63 76 66 72 6f 6d 20 74 61 62 6c 65 2c  e rcvfrom table,
7610: 0a 20 20 20 20 2a 2a 20 69 66 20 61 6e 79 20 66  .    ** if any f
7620: 69 6c 65 73 20 61 72 65 20 72 65 63 65 69 76 65  iles are receive
7630: 64 20 66 72 6f 6d 20 74 68 65 20 63 6f 6e 6e 65  d from the conne
7640: 63 74 65 64 20 63 6c 69 65 6e 74 2e 0a 20 20 20  cted client..   
7650: 20 2a 2f 0a 20 20 20 20 67 2e 7a 49 70 41 64 64   */.    g.zIpAdd
7660: 72 20 3d 20 6d 70 72 69 6e 74 66 28 22 25 73 22  r = mprintf("%s"
7670: 2c 20 7a 49 70 41 64 64 72 29 3b 0a 20 20 7d 0a  , zIpAddr);.  }.
7680: 20 0a 20 20 2f 2a 20 47 65 74 20 61 6c 6c 20 74   .  /* Get all t
7690: 68 65 20 6f 70 74 69 6f 6e 61 6c 20 66 69 65 6c  he optional fiel
76a0: 64 73 20 74 68 61 74 20 66 6f 6c 6c 6f 77 20 74  ds that follow t
76b0: 68 65 20 66 69 72 73 74 20 6c 69 6e 65 2e 0a 20  he first line.. 
76c0: 20 2a 2f 0a 20 20 77 68 69 6c 65 28 20 66 67 65   */.  while( fge
76d0: 74 73 28 7a 4c 69 6e 65 2c 73 69 7a 65 6f 66 28  ts(zLine,sizeof(
76e0: 7a 4c 69 6e 65 29 2c 73 74 64 69 6e 29 20 29 7b  zLine),stdin) ){
76f0: 0a 20 20 20 20 63 68 61 72 20 2a 7a 46 69 65 6c  .    char *zFiel
7700: 64 4e 61 6d 65 3b 0a 20 20 20 20 63 68 61 72 20  dName;.    char 
7710: 2a 7a 56 61 6c 3b 0a 0a 20 20 20 20 7a 46 69 65  *zVal;..    zFie
7720: 6c 64 4e 61 6d 65 20 3d 20 65 78 74 72 61 63 74  ldName = extract
7730: 5f 74 6f 6b 65 6e 28 7a 4c 69 6e 65 2c 26 7a 56  _token(zLine,&zV
7740: 61 6c 29 3b 0a 20 20 20 20 69 66 28 20 7a 46 69  al);.    if( zFi
7750: 65 6c 64 4e 61 6d 65 3d 3d 30 20 7c 7c 20 2a 7a  eldName==0 || *z
7760: 46 69 65 6c 64 4e 61 6d 65 3d 3d 30 20 29 20 62  FieldName==0 ) b
7770: 72 65 61 6b 3b 0a 20 20 20 20 77 68 69 6c 65 28  reak;.    while(
7780: 20 69 73 73 70 61 63 65 28 2a 7a 56 61 6c 29 20   isspace(*zVal) 
7790: 29 7b 20 7a 56 61 6c 2b 2b 3b 20 7d 0a 20 20 20  ){ zVal++; }.   
77a0: 20 69 20 3d 20 73 74 72 6c 65 6e 28 7a 56 61 6c   i = strlen(zVal
77b0: 29 3b 0a 20 20 20 20 77 68 69 6c 65 28 20 69 3e  );.    while( i>
77c0: 30 20 26 26 20 69 73 73 70 61 63 65 28 7a 56 61  0 && isspace(zVa
77d0: 6c 5b 69 2d 31 5d 29 20 29 7b 20 69 2d 2d 3b 20  l[i-1]) ){ i--; 
77e0: 7d 0a 20 20 20 20 7a 56 61 6c 5b 69 5d 20 3d 20  }.    zVal[i] = 
77f0: 30 3b 0a 20 20 20 20 66 6f 72 28 69 3d 30 3b 20  0;.    for(i=0; 
7800: 7a 46 69 65 6c 64 4e 61 6d 65 5b 69 5d 3b 20 69  zFieldName[i]; i
7810: 2b 2b 29 7b 20 7a 46 69 65 6c 64 4e 61 6d 65 5b  ++){ zFieldName[
7820: 69 5d 20 3d 20 74 6f 6c 6f 77 65 72 28 7a 46 69  i] = tolower(zFi
7830: 65 6c 64 4e 61 6d 65 5b 69 5d 29 3b 20 7d 0a 20  eldName[i]); }. 
7840: 20 20 20 69 66 28 20 73 74 72 63 6d 70 28 7a 46     if( strcmp(zF
7850: 69 65 6c 64 4e 61 6d 65 2c 22 75 73 65 72 2d 61  ieldName,"user-a
7860: 67 65 6e 74 3a 22 29 3d 3d 30 20 29 7b 0a 20 20  gent:")==0 ){.  
7870: 20 20 20 20 63 67 69 5f 73 65 74 65 6e 76 28 22      cgi_setenv("
7880: 48 54 54 50 5f 55 53 45 52 5f 41 47 45 4e 54 22  HTTP_USER_AGENT"
7890: 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d 65 6c  , zVal);.    }el
78a0: 73 65 20 69 66 28 20 73 74 72 63 6d 70 28 7a 46  se if( strcmp(zF
78b0: 69 65 6c 64 4e 61 6d 65 2c 22 63 6f 6e 74 65 6e  ieldName,"conten
78c0: 74 2d 6c 65 6e 67 74 68 3a 22 29 3d 3d 30 20 29  t-length:")==0 )
78d0: 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 65 74 65  {.      cgi_sete
78e0: 6e 76 28 22 43 4f 4e 54 45 4e 54 5f 4c 45 4e 47  nv("CONTENT_LENG
78f0: 54 48 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20  TH", zVal);.    
7900: 7d 65 6c 73 65 20 69 66 28 20 73 74 72 63 6d 70  }else if( strcmp
7910: 28 7a 46 69 65 6c 64 4e 61 6d 65 2c 22 72 65 66  (zFieldName,"ref
7920: 65 72 65 72 3a 22 29 3d 3d 30 20 29 7b 0a 20 20  erer:")==0 ){.  
7930: 20 20 20 20 63 67 69 5f 73 65 74 65 6e 76 28 22      cgi_setenv("
7940: 48 54 54 50 5f 52 45 46 45 52 45 52 22 2c 20 7a  HTTP_REFERER", z
7950: 56 61 6c 29 3b 0a 20 20 20 20 7d 65 6c 73 65 20  Val);.    }else 
7960: 69 66 28 20 73 74 72 63 6d 70 28 7a 46 69 65 6c  if( strcmp(zFiel
7970: 64 4e 61 6d 65 2c 22 68 6f 73 74 3a 22 29 3d 3d  dName,"host:")==
7980: 30 20 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73  0 ){.      cgi_s
7990: 65 74 65 6e 76 28 22 48 54 54 50 5f 48 4f 53 54  etenv("HTTP_HOST
79a0: 22 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d 65  ", zVal);.    }e
79b0: 6c 73 65 20 69 66 28 20 73 74 72 63 6d 70 28 7a  lse if( strcmp(z
79c0: 46 69 65 6c 64 4e 61 6d 65 2c 22 63 6f 6e 74 65  FieldName,"conte
79d0: 6e 74 2d 74 79 70 65 3a 22 29 3d 3d 30 20 29 7b  nt-type:")==0 ){
79e0: 0a 20 20 20 20 20 20 63 67 69 5f 73 65 74 65 6e  .      cgi_seten
79f0: 76 28 22 43 4f 4e 54 45 4e 54 5f 54 59 50 45 22  v("CONTENT_TYPE"
7a00: 2c 20 7a 56 61 6c 29 3b 0a 20 20 20 20 7d 65 6c  , zVal);.    }el
7a10: 73 65 20 69 66 28 20 73 74 72 63 6d 70 28 7a 46  se if( strcmp(zF
7a20: 69 65 6c 64 4e 61 6d 65 2c 22 63 6f 6f 6b 69 65  ieldName,"cookie
7a30: 3a 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20  :")==0 ){.      
7a40: 63 67 69 5f 73 65 74 65 6e 76 28 22 48 54 54 50  cgi_setenv("HTTP
7a50: 5f 43 4f 4f 4b 49 45 22 2c 20 7a 56 61 6c 29 3b  _COOKIE", zVal);
7a60: 0a 20 20 20 20 7d 65 6c 73 65 20 69 66 28 20 73  .    }else if( s
7a70: 74 72 63 6d 70 28 7a 46 69 65 6c 64 4e 61 6d 65  trcmp(zFieldName
7a80: 2c 22 69 66 2d 6e 6f 6e 65 2d 6d 61 74 63 68 3a  ,"if-none-match:
7a90: 22 29 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 63  ")==0 ){.      c
7aa0: 67 69 5f 73 65 74 65 6e 76 28 22 48 54 54 50 5f  gi_setenv("HTTP_
7ab0: 49 46 5f 4e 4f 4e 45 5f 4d 41 54 43 48 22 2c 20  IF_NONE_MATCH", 
7ac0: 7a 56 61 6c 29 3b 0a 20 20 20 20 7d 65 6c 73 65  zVal);.    }else
7ad0: 20 69 66 28 20 73 74 72 63 6d 70 28 7a 46 69 65   if( strcmp(zFie
7ae0: 6c 64 4e 61 6d 65 2c 22 69 66 2d 6d 6f 64 69 66  ldName,"if-modif
7af0: 69 65 64 2d 73 69 6e 63 65 3a 22 29 3d 3d 30 20  ied-since:")==0 
7b00: 29 7b 0a 20 20 20 20 20 20 63 67 69 5f 73 65 74  ){.      cgi_set
7b10: 65 6e 76 28 22 48 54 54 50 5f 49 46 5f 4d 4f 44  env("HTTP_IF_MOD
7b20: 49 46 49 45 44 5f 53 49 4e 43 45 22 2c 20 7a 56  IFIED_SINCE", zV
7b30: 61 6c 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 0a  al);.    }.  }..
7b40: 20 20 63 67 69 5f 69 6e 69 74 28 29 3b 0a 7d 0a    cgi_init();.}.
7b50: 0a 2f 2a 0a 2a 2a 20 4d 61 78 69 6d 75 6d 20 6e  ./*.** Maximum n
7b60: 75 6d 62 65 72 20 6f 66 20 63 68 69 6c 64 20 70  umber of child p
7b70: 72 6f 63 65 73 73 65 73 20 74 68 61 74 20 77 65  rocesses that we
7b80: 20 63 61 6e 20 68 61 76 65 20 72 75 6e 6e 69 6e   can have runnin
7b90: 67 0a 2a 2a 20 61 74 20 6f 6e 65 20 74 69 6d 65  g.** at one time
7ba0: 20 62 65 66 6f 72 65 20 77 65 20 73 74 61 72 74   before we start
7bb0: 20 73 6c 6f 77 69 6e 67 20 74 68 69 6e 67 73 20   slowing things 
7bc0: 64 6f 77 6e 2e 0a 2a 2f 0a 23 64 65 66 69 6e 65  down..*/.#define
7bd0: 20 4d 41 58 5f 50 41 52 41 4c 4c 45 4c 20 32 0a   MAX_PARALLEL 2.
7be0: 0a 2f 2a 0a 2a 2a 20 49 6d 70 6c 65 6d 65 6e 74  ./*.** Implement
7bf0: 20 61 6e 20 48 54 54 50 20 73 65 72 76 65 72 20   an HTTP server 
7c00: 64 61 65 6d 6f 6e 20 6c 69 73 74 65 6e 69 6e 67  daemon listening
7c10: 20 6f 6e 20 70 6f 72 74 20 69 50 6f 72 74 2e 0a   on port iPort..
7c20: 2a 2a 0a 2a 2a 20 41 73 20 6e 65 77 20 63 6f 6e  **.** As new con
7c30: 6e 65 63 74 69 6f 6e 73 20 61 72 72 69 76 65 2c  nections arrive,
7c40: 20 66 6f 72 6b 20 61 20 63 68 69 6c 64 20 61 6e   fork a child an
7c50: 64 20 6c 65 74 20 63 68 69 6c 64 20 72 65 74 75  d let child retu
7c60: 72 6e 0a 2a 2a 20 6f 75 74 20 6f 66 20 74 68 69  rn.** out of thi
7c70: 73 20 70 72 6f 63 65 64 75 72 65 20 63 61 6c 6c  s procedure call
7c80: 2e 20 20 54 68 65 20 63 68 69 6c 64 20 77 69 6c  .  The child wil
7c90: 6c 20 68 61 6e 64 6c 65 20 74 68 65 20 72 65 71  l handle the req
7ca0: 75 65 73 74 2e 0a 2a 2a 20 54 68 65 20 70 61 72  uest..** The par
7cb0: 65 6e 74 20 6e 65 76 65 72 20 72 65 74 75 72 6e  ent never return
7cc0: 73 20 66 72 6f 6d 20 74 68 69 73 20 70 72 6f 63  s from this proc
7cd0: 65 64 75 72 65 2e 0a 2a 2f 0a 76 6f 69 64 20 63  edure..*/.void c
7ce0: 67 69 5f 68 74 74 70 5f 73 65 72 76 65 72 28 69  gi_http_server(i
7cf0: 6e 74 20 69 50 6f 72 74 29 7b 0a 23 69 66 64 65  nt iPort){.#ifde
7d00: 66 20 5f 5f 4d 49 4e 47 57 33 32 5f 5f 0a 20 20  f __MINGW32__.  
7d10: 66 70 72 69 6e 74 66 28 73 74 64 65 72 72 2c 22  fprintf(stderr,"
7d20: 73 65 72 76 65 72 20 6e 6f 74 20 79 65 74 20 61  server not yet a
7d30: 76 61 69 6c 61 62 6c 65 20 69 6e 20 77 69 6e 64  vailable in wind
7d40: 6f 77 73 20 76 65 72 73 69 6f 6e 20 6f 66 20 66  ows version of f
7d50: 6f 73 73 69 6c 5c 6e 22 29 3b 0a 20 20 65 78 69  ossil\n");.  exi
7d60: 74 28 31 29 3b 0a 23 65 6c 73 65 0a 20 20 69 6e  t(1);.#else.  in
7d70: 74 20 6c 69 73 74 65 6e 65 72 3b 20 20 20 20 20  t listener;     
7d80: 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 54 68             /* Th
7d90: 65 20 73 65 72 76 65 72 20 73 6f 63 6b 65 74 20  e server socket 
7da0: 2a 2f 0a 20 20 69 6e 74 20 63 6f 6e 6e 65 63 74  */.  int connect
7db0: 69 6f 6e 3b 20 20 20 20 20 20 20 20 20 20 20 20  ion;            
7dc0: 20 20 2f 2a 20 41 20 73 6f 63 6b 65 74 20 66 6f    /* A socket fo
7dd0: 72 20 65 61 63 68 20 69 6e 64 69 76 69 64 75 61  r each individua
7de0: 6c 20 63 6f 6e 6e 65 63 74 69 6f 6e 20 2a 2f 0a  l connection */.
7df0: 20 20 66 64 5f 73 65 74 20 72 65 61 64 66 64 73    fd_set readfds
7e00: 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20 2f  ;              /
7e10: 2a 20 53 65 74 20 6f 66 20 66 69 6c 65 20 64 65  * Set of file de
7e20: 73 63 72 69 70 74 6f 72 73 20 66 6f 72 20 73 65  scriptors for se
7e30: 6c 65 63 74 28 29 20 2a 2f 0a 20 20 73 69 7a 65  lect() */.  size
7e40: 5f 74 20 6c 65 6e 61 64 64 72 3b 20 20 20 20 20  _t lenaddr;     
7e50: 20 20 20 20 20 20 20 20 20 2f 2a 20 4c 65 6e 67           /* Leng
7e60: 74 68 20 6f 66 20 74 68 65 20 69 6e 61 64 64 72  th of the inaddr
7e70: 20 73 74 72 75 63 74 75 72 65 20 2a 2f 0a 20 20   structure */.  
7e80: 69 6e 74 20 63 68 69 6c 64 3b 20 20 20 20 20 20  int child;      
7e90: 20 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20               /* 
7ea0: 50 49 44 20 6f 66 20 74 68 65 20 63 68 69 6c 64  PID of the child
7eb0: 20 70 72 6f 63 65 73 73 20 2a 2f 0a 20 20 69 6e   process */.  in
7ec0: 74 20 6e 63 68 69 6c 64 72 65 6e 20 3d 20 30 3b  t nchildren = 0;
7ed0: 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4e 75             /* Nu
7ee0: 6d 62 65 72 20 6f 66 20 63 68 69 6c 64 20 70 72  mber of child pr
7ef0: 6f 63 65 73 73 65 73 20 2a 2f 0a 20 20 73 74 72  ocesses */.  str
7f00: 75 63 74 20 74 69 6d 65 76 61 6c 20 64 65 6c 61  uct timeval dela
7f10: 79 3b 20 20 20 20 20 20 20 20 2f 2a 20 48 6f 77  y;        /* How
7f20: 20 6c 6f 6e 67 20 74 6f 20 77 61 69 74 20 69 6e   long to wait in
7f30: 73 69 64 65 20 73 65 6c 65 63 74 28 29 20 2a 2f  side select() */
7f40: 0a 20 20 73 74 72 75 63 74 20 73 6f 63 6b 61 64  .  struct sockad
7f50: 64 72 5f 69 6e 20 69 6e 61 64 64 72 3b 20 20 20  dr_in inaddr;   
7f60: 2f 2a 20 54 68 65 20 73 6f 63 6b 65 74 20 61 64  /* The socket ad
7f70: 64 72 65 73 73 20 2a 2f 0a 20 20 69 6e 74 20 6f  dress */.  int o
7f80: 70 74 20 3d 20 31 3b 20 20 20 20 20 20 20 20 20  pt = 1;         
7f90: 20 20 20 20 20 20 20 20 2f 2a 20 73 65 74 73 6f          /* setso
7fa0: 63 6b 6f 70 74 20 66 6c 61 67 20 2a 2f 0a 0a 20  ckopt flag */.. 
7fb0: 20 6d 65 6d 73 65 74 28 26 69 6e 61 64 64 72 2c   memset(&inaddr,
7fc0: 20 30 2c 20 73 69 7a 65 6f 66 28 69 6e 61 64 64   0, sizeof(inadd
7fd0: 72 29 29 3b 0a 20 20 69 6e 61 64 64 72 2e 73 69  r));.  inaddr.si
7fe0: 6e 5f 66 61 6d 69 6c 79 20 3d 20 41 46 5f 49 4e  n_family = AF_IN
7ff0: 45 54 3b 0a 20 20 69 6e 61 64 64 72 2e 73 69 6e  ET;.  inaddr.sin
8000: 5f 61 64 64 72 2e 73 5f 61 64 64 72 20 3d 20 49  _addr.s_addr = I
8010: 4e 41 44 44 52 5f 41 4e 59 3b 0a 20 20 69 6e 61  NADDR_ANY;.  ina
8020: 64 64 72 2e 73 69 6e 5f 70 6f 72 74 20 3d 20 68  ddr.sin_port = h
8030: 74 6f 6e 73 28 69 50 6f 72 74 29 3b 0a 20 20 6c  tons(iPort);.  l
8040: 69 73 74 65 6e 65 72 20 3d 20 73 6f 63 6b 65 74  istener = socket
8050: 28 41 46 5f 49 4e 45 54 2c 20 53 4f 43 4b 5f 53  (AF_INET, SOCK_S
8060: 54 52 45 41 4d 2c 20 30 29 3b 0a 20 20 69 66 28  TREAM, 0);.  if(
8070: 20 6c 69 73 74 65 6e 65 72 3c 30 20 29 7b 0a 20   listener<0 ){. 
8080: 20 20 20 66 70 72 69 6e 74 66 28 73 74 64 65 72     fprintf(stder
8090: 72 2c 22 43 61 6e 27 74 20 63 72 65 61 74 65 20  r,"Can't create 
80a0: 61 20 73 6f 63 6b 65 74 5c 6e 22 29 3b 0a 20 20  a socket\n");.  
80b0: 20 20 65 78 69 74 28 31 29 3b 0a 20 20 7d 0a 0a    exit(1);.  }..
80c0: 20 20 2f 2a 20 69 66 20 77 65 20 63 61 6e 27 74    /* if we can't
80d0: 20 74 65 72 6d 69 6e 61 74 65 20 6e 69 63 65 6c   terminate nicel
80e0: 79 2c 20 61 74 20 6c 65 61 73 74 20 61 6c 6c 6f  y, at least allo
80f0: 77 20 74 68 65 20 73 6f 63 6b 65 74 20 74 6f 20  w the socket to 
8100: 62 65 20 72 65 75 73 65 64 20 2a 2f 0a 20 20 73  be reused */.  s
8110: 65 74 73 6f 63 6b 6f 70 74 28 6c 69 73 74 65 6e  etsockopt(listen
8120: 65 72 2c 53 4f 4c 5f 53 4f 43 4b 45 54 2c 53 4f  er,SOL_SOCKET,SO
8130: 5f 52 45 55 53 45 41 44 44 52 2c 26 6f 70 74 2c  _REUSEADDR,&opt,
8140: 73 69 7a 65 6f 66 28 6f 70 74 29 29 3b 0a 0a 20  sizeof(opt));.. 
8150: 20 69 66 28 20 62 69 6e 64 28 6c 69 73 74 65 6e   if( bind(listen
8160: 65 72 2c 20 28 73 74 72 75 63 74 20 73 6f 63 6b  er, (struct sock
8170: 61 64 64 72 2a 29 26 69 6e 61 64 64 72 2c 20 73  addr*)&inaddr, s
8180: 69 7a 65 6f 66 28 69 6e 61 64 64 72 29 29 3c 30  izeof(inaddr))<0
8190: 20 29 7b 0a 20 20 20 20 66 70 72 69 6e 74 66 28   ){.    fprintf(
81a0: 73 74 64 65 72 72 2c 22 43 61 6e 27 74 20 62 69  stderr,"Can't bi
81b0: 6e 64 20 74 6f 20 70 6f 72 74 20 25 64 5c 6e 22  nd to port %d\n"
81c0: 2c 20 69 50 6f 72 74 29 3b 0a 20 20 20 20 65 78  , iPort);.    ex
81d0: 69 74 28 31 29 3b 0a 20 20 7d 0a 20 20 6c 69 73  it(1);.  }.  lis
81e0: 74 65 6e 28 6c 69 73 74 65 6e 65 72 2c 31 30 29  ten(listener,10)
81f0: 3b 0a 20 20 77 68 69 6c 65 28 20 31 20 29 7b 0a  ;.  while( 1 ){.
8200: 20 20 20 20 69 66 28 20 6e 63 68 69 6c 64 72 65      if( nchildre
8210: 6e 3e 4d 41 58 5f 50 41 52 41 4c 4c 45 4c 20 29  n>MAX_PARALLEL )
8220: 7b 0a 20 20 20 20 20 20 2f 2a 20 53 6c 6f 77 20  {.      /* Slow 
8230: 64 6f 77 6e 20 69 66 20 63 6f 6e 6e 65 63 74 69  down if connecti
8240: 6f 6e 73 20 61 72 65 20 61 72 72 69 76 69 6e 67  ons are arriving
8250: 20 74 6f 6f 20 66 61 73 74 20 2a 2f 0a 20 20 20   too fast */.   
8260: 20 20 20 73 6c 65 65 70 28 20 6e 63 68 69 6c 64     sleep( nchild
8270: 72 65 6e 2d 4d 41 58 5f 50 41 52 41 4c 4c 45 4c  ren-MAX_PARALLEL
8280: 20 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20 64 65   );.    }.    de
8290: 6c 61 79 2e 74 76 5f 73 65 63 20 3d 20 36 30 3b  lay.tv_sec = 60;
82a0: 0a 20 20 20 20 64 65 6c 61 79 2e 74 76 5f 75 73  .    delay.tv_us
82b0: 65 63 20 3d 20 30 3b 0a 20 20 20 20 46 44 5f 5a  ec = 0;.    FD_Z
82c0: 45 52 4f 28 26 72 65 61 64 66 64 73 29 3b 0a 20  ERO(&readfds);. 
82d0: 20 20 20 46 44 5f 53 45 54 28 20 6c 69 73 74 65     FD_SET( liste
82e0: 6e 65 72 2c 20 26 72 65 61 64 66 64 73 29 3b 0a  ner, &readfds);.
82f0: 20 20 20 20 69 66 28 20 73 65 6c 65 63 74 28 20      if( select( 
8300: 6c 69 73 74 65 6e 65 72 2b 31 2c 20 26 72 65 61  listener+1, &rea
8310: 64 66 64 73 2c 20 30 2c 20 30 2c 20 26 64 65 6c  dfds, 0, 0, &del
8320: 61 79 29 20 29 7b 0a 20 20 20 20 20 20 6c 65 6e  ay) ){.      len
8330: 61 64 64 72 20 3d 20 73 69 7a 65 6f 66 28 69 6e  addr = sizeof(in
8340: 61 64 64 72 29 3b 0a 20 20 20 20 20 20 63 6f 6e  addr);.      con
8350: 6e 65 63 74 69 6f 6e 20 3d 20 61 63 63 65 70 74  nection = accept
8360: 28 6c 69 73 74 65 6e 65 72 2c 20 28 73 74 72 75  (listener, (stru
8370: 63 74 20 73 6f 63 6b 61 64 64 72 2a 29 26 69 6e  ct sockaddr*)&in
8380: 61 64 64 72 2c 20 28 73 6f 63 6b 6c 65 6e 5f 74  addr, (socklen_t
8390: 2a 29 20 26 6c 65 6e 61 64 64 72 29 3b 0a 20 20  *) &lenaddr);.  
83a0: 20 20 20 20 69 66 28 20 63 6f 6e 6e 65 63 74 69      if( connecti
83b0: 6f 6e 3e 3d 30 20 29 7b 0a 20 20 20 20 20 20 20  on>=0 ){.       
83c0: 20 63 68 69 6c 64 20 3d 20 66 6f 72 6b 28 29 3b   child = fork();
83d0: 0a 20 20 20 20 20 20 20 20 69 66 28 20 63 68 69  .        if( chi
83e0: 6c 64 21 3d 30 20 29 7b 0a 20 20 20 20 20 20 20  ld!=0 ){.       
83f0: 20 20 20 69 66 28 20 63 68 69 6c 64 3e 30 20 29     if( child>0 )
8400: 20 6e 63 68 69 6c 64 72 65 6e 2b 2b 3b 0a 20 20   nchildren++;.  
8410: 20 20 20 20 20 20 20 20 63 6c 6f 73 65 28 63 6f          close(co
8420: 6e 6e 65 63 74 69 6f 6e 29 3b 0a 20 20 20 20 20  nnection);.     
8430: 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20 20     }else{.      
8440: 20 20 20 20 63 6c 6f 73 65 28 30 29 3b 0a 20 20      close(0);.  
8450: 20 20 20 20 20 20 20 20 64 75 70 28 63 6f 6e 6e          dup(conn
8460: 65 63 74 69 6f 6e 29 3b 0a 20 20 20 20 20 20 20  ection);.       
8470: 20 20 20 63 6c 6f 73 65 28 31 29 3b 0a 20 20 20     close(1);.   
8480: 20 20 20 20 20 20 20 64 75 70 28 63 6f 6e 6e 65         dup(conne
8490: 63 74 69 6f 6e 29 3b 0a 20 20 20 20 20 20 20 20  ction);.        
84a0: 20 20 69 66 28 20 21 67 2e 66 48 74 74 70 54 72    if( !g.fHttpTr
84b0: 61 63 65 20 29 7b 0a 20 20 20 20 20 20 20 20 20  ace ){.         
84c0: 20 20 20 63 6c 6f 73 65 28 32 29 3b 0a 20 20 20     close(2);.   
84d0: 20 20 20 20 20 20 20 20 20 64 75 70 28 63 6f 6e           dup(con
84e0: 6e 65 63 74 69 6f 6e 29 3b 0a 20 20 20 20 20 20  nection);.      
84f0: 20 20 20 20 7d 0a 20 20 20 20 20 20 20 20 20 20      }.          
8500: 63 6c 6f 73 65 28 63 6f 6e 6e 65 63 74 69 6f 6e  close(connection
8510: 29 3b 0a 20 20 20 20 20 20 20 20 20 20 72 65 74  );.          ret
8520: 75 72 6e 3b 0a 20 20 20 20 20 20 20 20 7d 0a 20  urn;.        }. 
8530: 20 20 20 20 20 7d 0a 20 20 20 20 7d 0a 20 20 20       }.    }.   
8540: 20 2f 2a 20 42 75 72 79 20 64 65 61 64 20 63 68   /* Bury dead ch
8550: 69 6c 64 72 65 6e 20 2a 2f 0a 20 20 20 20 77 68  ildren */.    wh
8560: 69 6c 65 28 20 77 61 69 74 70 69 64 28 30 2c 20  ile( waitpid(0, 
8570: 30 2c 20 57 4e 4f 48 41 4e 47 29 3e 30 20 29 7b  0, WNOHANG)>0 ){
8580: 0a 20 20 20 20 20 20 6e 63 68 69 6c 64 72 65 6e  .      nchildren
8590: 2d 2d 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20  --;.    }.  }.  
85a0: 2f 2a 20 4e 4f 54 20 52 45 41 43 48 45 44 20 2a  /* NOT REACHED *
85b0: 2f 20 20 0a 20 20 65 78 69 74 28 31 29 3b 0a 23  /  .  exit(1);.#
85c0: 65 6e 64 69 66 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 4e  endif.}../*.** N
85d0: 61 6d 65 20 6f 66 20 64 61 79 73 20 61 6e 64 20  ame of days and 
85e0: 6d 6f 6e 74 68 73 2e 0a 2a 2f 0a 73 74 61 74 69  months..*/.stati
85f0: 63 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 61 7a  c const char *az
8600: 44 61 79 73 5b 5d 20 3d 0a 20 20 20 20 7b 22 53  Days[] =.    {"S
8610: 75 6e 22 2c 20 22 4d 6f 6e 22 2c 20 22 54 75 65  un", "Mon", "Tue
8620: 22 2c 20 22 57 65 64 22 2c 20 22 54 68 75 22 2c  ", "Wed", "Thu",
8630: 20 22 46 72 69 22 2c 20 22 53 61 74 22 2c 20 30   "Fri", "Sat", 0
8640: 7d 3b 0a 73 74 61 74 69 63 20 63 6f 6e 73 74 20  };.static const 
8650: 63 68 61 72 20 2a 61 7a 4d 6f 6e 74 68 73 5b 5d  char *azMonths[]
8660: 20 3d 0a 20 20 20 20 7b 22 4a 61 6e 22 2c 20 22   =.    {"Jan", "
8670: 46 65 62 22 2c 20 22 4d 61 72 22 2c 20 22 41 70  Feb", "Mar", "Ap
8680: 72 22 2c 20 22 4d 61 79 22 2c 20 22 4a 75 6e 22  r", "May", "Jun"
8690: 2c 0a 20 20 20 20 20 22 4a 75 6c 22 2c 20 22 41  ,.     "Jul", "A
86a0: 75 67 22 2c 20 22 53 65 70 22 2c 20 22 4f 63 74  ug", "Sep", "Oct
86b0: 22 2c 20 22 4e 6f 76 22 2c 20 22 44 65 63 22 2c  ", "Nov", "Dec",
86c0: 20 30 7d 3b 0a 0a 0a 2f 2a 0a 2a 2a 20 52 65 74   0};.../*.** Ret
86d0: 75 72 6e 73 20 61 6e 20 52 46 43 38 32 32 2d 66  urns an RFC822-f
86e0: 6f 72 6d 61 74 74 65 64 20 74 69 6d 65 20 73 74  ormatted time st
86f0: 72 69 6e 67 20 73 75 69 74 61 62 6c 65 20 66 6f  ring suitable fo
8700: 72 20 48 54 54 50 20 68 65 61 64 65 72 73 2c 20  r HTTP headers, 
8710: 61 6d 6f 6e 67 0a 2a 2a 20 6f 74 68 65 72 20 74  among.** other t
8720: 68 69 6e 67 73 2e 0a 2a 2a 20 52 65 74 75 72 6e  hings..** Return
8730: 65 64 20 74 69 6d 65 7a 6f 6e 65 20 69 73 20 61  ed timezone is a
8740: 6c 77 61 79 73 20 47 4d 54 20 61 73 20 72 65 71  lways GMT as req
8750: 75 69 72 65 64 20 62 79 20 48 54 54 50 2f 31 2e  uired by HTTP/1.
8760: 31 20 73 70 65 63 69 66 69 63 61 74 69 6f 6e 2e  1 specification.
8770: 0a 2a 2a 0a 2a 2a 20 53 65 65 20 68 74 74 70 3a  .**.** See http:
8780: 2f 2f 77 77 77 2e 66 61 71 73 2e 6f 72 67 2f 72  //www.faqs.org/r
8790: 66 63 73 2f 72 66 63 38 32 32 2e 68 74 6d 6c 2c  fcs/rfc822.html,
87a0: 20 73 65 63 74 69 6f 6e 20 35 0a 2a 2a 20 61 6e   section 5.** an
87b0: 64 20 68 74 74 70 3a 2f 2f 77 77 77 2e 66 61 71  d http://www.faq
87c0: 73 2e 6f 72 67 2f 72 66 63 73 2f 72 66 63 32 36  s.org/rfcs/rfc26
87d0: 31 36 2e 68 74 6d 6c 2c 20 73 65 63 74 69 6f 6e  16.html, section
87e0: 20 33 2e 33 2e 0a 2a 2f 0a 63 68 61 72 20 2a 63   3.3..*/.char *c
87f0: 67 69 5f 72 66 63 38 32 32 5f 64 61 74 65 73 74  gi_rfc822_datest
8800: 61 6d 70 28 74 69 6d 65 5f 74 20 6e 6f 77 29 7b  amp(time_t now){
8810: 0a 20 20 73 74 72 75 63 74 20 74 6d 20 2a 70 54  .  struct tm *pT
8820: 6d 3b 0a 20 20 70 54 6d 20 3d 20 67 6d 74 69 6d  m;.  pTm = gmtim
8830: 65 28 26 6e 6f 77 29 3b 0a 20 20 69 66 28 20 70  e(&now);.  if( p
8840: 54 6d 3d 3d 30 20 29 20 72 65 74 75 72 6e 20 22  Tm==0 ) return "
8850: 22 3b 0a 20 20 72 65 74 75 72 6e 20 6d 70 72 69  ";.  return mpri
8860: 6e 74 66 28 22 25 73 2c 20 25 64 20 25 73 20 25  ntf("%s, %d %s %
8870: 30 32 64 20 25 30 32 64 3a 25 30 32 64 3a 25 30  02d %02d:%02d:%0
8880: 32 64 20 47 4d 54 22 2c 0a 20 20 20 20 20 20 20  2d GMT",.       
8890: 20 20 20 20 20 20 20 20 20 20 61 7a 44 61 79 73            azDays
88a0: 5b 70 54 6d 2d 3e 74 6d 5f 77 64 61 79 5d 2c 20  [pTm->tm_wday], 
88b0: 70 54 6d 2d 3e 74 6d 5f 6d 64 61 79 2c 20 61 7a  pTm->tm_mday, az
88c0: 4d 6f 6e 74 68 73 5b 70 54 6d 2d 3e 74 6d 5f 6d  Months[pTm->tm_m
88d0: 6f 6e 5d 2c 0a 20 20 20 20 20 20 20 20 20 20 20  on],.           
88e0: 20 20 20 20 20 20 70 54 6d 2d 3e 74 6d 5f 79 65        pTm->tm_ye
88f0: 61 72 2b 31 39 30 30 2c 20 70 54 6d 2d 3e 74 6d  ar+1900, pTm->tm
8900: 5f 68 6f 75 72 2c 20 70 54 6d 2d 3e 74 6d 5f 6d  _hour, pTm->tm_m
8910: 69 6e 2c 20 70 54 6d 2d 3e 74 6d 5f 73 65 63 29  in, pTm->tm_sec)
8920: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 50 61 72 73 65  ;.}../*.** Parse
8930: 20 61 6e 20 52 46 43 38 32 32 2d 66 6f 72 6d 61   an RFC822-forma
8940: 74 74 65 64 20 74 69 6d 65 73 74 61 6d 70 20 61  tted timestamp a
8950: 73 20 77 65 27 64 20 65 78 70 65 63 74 20 66 72  s we'd expect fr
8960: 6f 6d 20 48 54 54 50 20 61 6e 64 20 72 65 74 75  om HTTP and retu
8970: 72 6e 0a 2a 2a 20 61 20 55 6e 69 78 20 65 70 6f  rn.** a Unix epo
8980: 63 68 20 74 69 6d 65 2e 20 3c 3d 20 7a 65 72 6f  ch time. <= zero
8990: 20 69 73 20 72 65 74 75 72 6e 65 64 20 6f 6e 20   is returned on 
89a0: 66 61 69 6c 75 72 65 2e 0a 2a 2a 0a 2a 2a 20 4e  failure..**.** N
89b0: 6f 74 65 20 74 68 61 74 20 74 68 69 73 20 77 6f  ote that this wo
89c0: 6e 27 74 20 68 61 6e 64 6c 65 20 61 6c 6c 20 74  n't handle all t
89d0: 68 65 20 5f 61 6c 6c 6f 77 65 64 5f 20 48 54 54  he _allowed_ HTT
89e0: 50 20 66 6f 72 6d 61 74 73 2c 20 6a 75 73 74 20  P formats, just 
89f0: 74 68 65 0a 2a 2a 20 6d 6f 73 74 20 70 6f 70 75  the.** most popu
8a00: 6c 61 72 20 6f 6e 65 20 28 74 68 65 20 6f 6e 65  lar one (the one
8a10: 20 67 65 6e 65 72 61 74 65 64 20 62 79 20 63 67   generated by cg
8a20: 69 5f 72 66 63 38 32 32 5f 64 61 74 65 73 74 61  i_rfc822_datesta
8a30: 6d 70 28 29 2c 20 61 63 74 75 61 6c 6c 79 29 2e  mp(), actually).
8a40: 0a 2a 2f 0a 74 69 6d 65 5f 74 20 63 67 69 5f 72  .*/.time_t cgi_r
8a50: 66 63 38 32 32 5f 70 61 72 73 65 64 61 74 65 28  fc822_parsedate(
8a60: 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 44 61 74  const char *zDat
8a70: 65 29 7b 0a 20 20 73 74 72 75 63 74 20 74 6d 20  e){.  struct tm 
8a80: 74 3b 0a 20 20 63 68 61 72 20 7a 49 67 6e 6f 72  t;.  char zIgnor
8a90: 65 5b 31 36 5d 3b 0a 20 20 63 68 61 72 20 7a 4d  e[16];.  char zM
8aa0: 6f 6e 74 68 5b 31 36 5d 3b 0a 0a 20 20 6d 65 6d  onth[16];..  mem
8ab0: 73 65 74 28 26 74 2c 20 30 2c 20 73 69 7a 65 6f  set(&t, 0, sizeo
8ac0: 66 28 74 29 29 3b 0a 20 20 69 66 28 20 37 3d 3d  f(t));.  if( 7==
8ad0: 73 73 63 61 6e 66 28 7a 44 61 74 65 2c 20 22 25  sscanf(zDate, "%
8ae0: 31 32 5b 41 2d 5a 61 2d 7a 2c 5d 20 25 64 20 25  12[A-Za-z,] %d %
8af0: 31 32 5b 41 2d 5a 61 2d 7a 5d 20 25 64 20 25 64  12[A-Za-z] %d %d
8b00: 3a 25 64 3a 25 64 22 2c 20 7a 49 67 6e 6f 72 65  :%d:%d", zIgnore
8b10: 2c 0a 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ,.              
8b20: 20 20 20 20 20 20 20 20 20 26 74 2e 74 6d 5f 6d           &t.tm_m
8b30: 64 61 79 2c 20 7a 4d 6f 6e 74 68 2c 20 26 74 2e  day, zMonth, &t.
8b40: 74 6d 5f 79 65 61 72 2c 20 26 74 2e 74 6d 5f 68  tm_year, &t.tm_h
8b50: 6f 75 72 2c 20 26 74 2e 74 6d 5f 6d 69 6e 2c 0a  our, &t.tm_min,.
8b60: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
8b70: 20 20 20 20 20 20 20 26 74 2e 74 6d 5f 73 65 63         &t.tm_sec
8b80: 29 29 7b 0a 0a 20 20 20 20 69 66 28 20 74 2e 74  )){..    if( t.t
8b90: 6d 5f 79 65 61 72 20 3e 20 31 39 30 30 20 29 20  m_year > 1900 ) 
8ba0: 74 2e 74 6d 5f 79 65 61 72 20 2d 3d 20 31 39 30  t.tm_year -= 190
8bb0: 30 3b 0a 20 20 20 20 66 6f 72 28 74 2e 74 6d 5f  0;.    for(t.tm_
8bc0: 6d 6f 6e 3d 30 3b 20 61 7a 4d 6f 6e 74 68 73 5b  mon=0; azMonths[
8bd0: 74 2e 74 6d 5f 6d 6f 6e 5d 3b 20 74 2e 74 6d 5f  t.tm_mon]; t.tm_
8be0: 6d 6f 6e 2b 2b 29 7b 0a 20 20 20 20 20 20 69 66  mon++){.      if
8bf0: 28 20 21 73 74 72 6e 63 61 73 65 63 6d 70 28 20  ( !strncasecmp( 
8c00: 61 7a 4d 6f 6e 74 68 73 5b 74 2e 74 6d 5f 6d 6f  azMonths[t.tm_mo
8c10: 6e 5d 2c 20 7a 4d 6f 6e 74 68 2c 20 33 20 29 29  n], zMonth, 3 ))
8c20: 7b 0a 20 20 20 20 20 20 20 20 72 65 74 75 72 6e  {.        return
8c30: 20 6d 6b 67 6d 74 69 6d 65 28 26 74 29 3b 0a 20   mkgmtime(&t);. 
8c40: 20 20 20 20 20 7d 0a 20 20 20 20 7d 0a 20 20 7d       }.    }.  }
8c50: 0a 0a 20 20 72 65 74 75 72 6e 20 30 3b 0a 7d 0a  ..  return 0;.}.
8c60: 0a 2f 2a 0a 2a 2a 20 43 6f 6e 76 65 72 74 20 61  ./*.** Convert a
8c70: 20 73 74 72 75 63 74 20 74 6d 2a 20 74 68 61 74   struct tm* that
8c80: 20 72 65 70 72 65 73 65 6e 74 73 20 61 20 6d 6f   represents a mo
8c90: 6d 65 6e 74 20 69 6e 20 55 54 43 20 69 6e 74 6f  ment in UTC into
8ca0: 20 74 68 65 20 6e 75 6d 62 65 72 0a 2a 2a 20 6f   the number.** o
8cb0: 66 20 73 65 63 6f 6e 64 73 20 69 6e 20 31 39 37  f seconds in 197
8cc0: 30 2c 20 55 54 43 2e 0a 2a 2f 0a 74 69 6d 65 5f  0, UTC..*/.time_
8cd0: 74 20 6d 6b 67 6d 74 69 6d 65 28 73 74 72 75 63  t mkgmtime(struc
8ce0: 74 20 74 6d 20 2a 70 29 7b 0a 20 20 74 69 6d 65  t tm *p){.  time
8cf0: 5f 74 20 74 3b 0a 20 20 69 6e 74 20 6e 44 61 79  _t t;.  int nDay
8d00: 3b 0a 20 20 69 6e 74 20 69 73 4c 65 61 70 59 72  ;.  int isLeapYr
8d10: 3b 0a 20 20 2f 2a 20 44 61 79 73 20 69 6e 20 65  ;.  /* Days in e
8d20: 61 63 68 20 6d 6f 6e 74 68 3a 20 20 20 20 20 20  ach month:      
8d30: 20 33 31 2c 20 32 38 2c 20 33 31 2c 20 33 30 2c   31, 28, 31, 30,
8d40: 20 33 31 2c 20 33 30 2c 20 33 31 2c 20 33 31 2c   31, 30, 31, 31,
8d50: 20 33 30 2c 20 33 31 2c 20 33 30 2c 20 33 31 20   30, 31, 30, 31 
8d60: 2a 2f 0a 20 20 73 74 61 74 69 63 20 69 6e 74 20  */.  static int 
8d70: 70 72 69 6f 72 44 61 79 73 5b 5d 20 20 20 3d 20  priorDays[]   = 
8d80: 7b 20 20 30 2c 20 33 31 2c 20 35 39 2c 20 39 30  {  0, 31, 59, 90
8d90: 2c 31 32 30 2c 31 35 31 2c 31 38 31 2c 32 31 32  ,120,151,181,212
8da0: 2c 32 34 33 2c 32 37 33 2c 33 30 34 2c 33 33 34  ,243,273,304,334
8db0: 20 7d 3b 0a 20 20 69 66 28 20 70 2d 3e 74 6d 5f   };.  if( p->tm_
8dc0: 6d 6f 6e 3c 30 20 29 7b 0a 20 20 20 20 69 6e 74  mon<0 ){.    int
8dd0: 20 6e 59 65 61 72 20 3d 20 28 31 31 20 2d 20 70   nYear = (11 - p
8de0: 2d 3e 74 6d 5f 6d 6f 6e 29 2f 31 32 3b 0a 20 20  ->tm_mon)/12;.  
8df0: 20 20 70 2d 3e 74 6d 5f 79 65 61 72 20 2d 3d 20    p->tm_year -= 
8e00: 6e 59 65 61 72 3b 0a 20 20 20 20 70 2d 3e 74 6d  nYear;.    p->tm
8e10: 5f 6d 6f 6e 20 2b 3d 20 6e 59 65 61 72 2a 31 32  _mon += nYear*12
8e20: 3b 0a 20 20 7d 65 6c 73 65 20 69 66 28 20 70 2d  ;.  }else if( p-
8e30: 3e 74 6d 5f 6d 6f 6e 3e 31 31 20 29 7b 0a 20 20  >tm_mon>11 ){.  
8e40: 20 20 70 2d 3e 74 6d 5f 79 65 61 72 20 2b 3d 20    p->tm_year += 
8e50: 70 2d 3e 74 6d 5f 6d 6f 6e 2f 31 32 3b 0a 20 20  p->tm_mon/12;.  
8e60: 20 20 70 2d 3e 74 6d 5f 6d 6f 6e 20 25 3d 20 31    p->tm_mon %= 1
8e70: 32 3b 0a 20 20 7d 0a 20 20 69 73 4c 65 61 70 59  2;.  }.  isLeapY
8e80: 72 20 3d 20 70 2d 3e 74 6d 5f 79 65 61 72 25 34  r = p->tm_year%4
8e90: 3d 3d 30 20 26 26 20 28 70 2d 3e 74 6d 5f 79 65  ==0 && (p->tm_ye
8ea0: 61 72 25 31 30 30 21 3d 30 20 7c 7c 20 28 70 2d  ar%100!=0 || (p-
8eb0: 3e 74 6d 5f 79 65 61 72 2b 33 30 30 29 25 34 30  >tm_year+300)%40
8ec0: 30 3d 3d 30 29 3b 0a 20 20 70 2d 3e 74 6d 5f 79  0==0);.  p->tm_y
8ed0: 64 61 79 20 3d 20 70 72 69 6f 72 44 61 79 73 5b  day = priorDays[
8ee0: 70 2d 3e 74 6d 5f 6d 6f 6e 5d 20 2b 20 70 2d 3e  p->tm_mon] + p->
8ef0: 74 6d 5f 6d 64 61 79 20 2d 20 31 3b 0a 20 20 69  tm_mday - 1;.  i
8f00: 66 28 20 69 73 4c 65 61 70 59 72 20 26 26 20 70  f( isLeapYr && p
8f10: 2d 3e 74 6d 5f 6d 6f 6e 3e 31 20 29 20 70 2d 3e  ->tm_mon>1 ) p->
8f20: 74 6d 5f 79 64 61 79 2b 2b 3b 0a 20 20 6e 44 61  tm_yday++;.  nDa
8f30: 79 20 3d 20 28 70 2d 3e 74 6d 5f 79 65 61 72 2d  y = (p->tm_year-
8f40: 37 30 29 2a 33 36 35 20 2b 20 28 70 2d 3e 74 6d  70)*365 + (p->tm
8f50: 5f 79 65 61 72 2d 36 39 29 2f 34 20 2d 70 2d 3e  _year-69)/4 -p->
8f60: 74 6d 5f 79 65 61 72 2f 31 30 30 20 2b 20 0a 20  tm_year/100 + . 
8f70: 20 20 20 20 20 20 20 20 28 70 2d 3e 74 6d 5f 79          (p->tm_y
8f80: 65 61 72 2b 33 30 30 29 2f 34 30 30 20 2b 20 70  ear+300)/400 + p
8f90: 2d 3e 74 6d 5f 79 64 61 79 3b 0a 20 20 74 20 3d  ->tm_yday;.  t =
8fa0: 20 28 28 6e 44 61 79 2a 32 34 20 2b 20 70 2d 3e   ((nDay*24 + p->
8fb0: 74 6d 5f 68 6f 75 72 29 2a 36 30 20 2b 20 70 2d  tm_hour)*60 + p-
8fc0: 3e 74 6d 5f 6d 69 6e 29 2a 36 30 20 2b 20 70 2d  >tm_min)*60 + p-
8fd0: 3e 74 6d 5f 73 65 63 3b 0a 20 20 72 65 74 75 72  >tm_sec;.  retur
8fe0: 6e 20 74 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 43 68  n t;.}../*.** Ch
8ff0: 65 63 6b 20 74 68 65 20 6f 62 6a 65 63 74 54 69  eck the objectTi
9000: 6d 65 20 61 67 61 69 6e 73 74 20 74 68 65 20 49  me against the I
9010: 66 2d 4d 6f 64 69 66 69 65 64 2d 53 69 6e 63 65  f-Modified-Since
9020: 20 72 65 71 75 65 73 74 20 68 65 61 64 65 72 2e   request header.
9030: 20 49 66 20 74 68 65 0a 2a 2a 20 6f 62 6a 65 63   If the.** objec
9040: 74 20 74 69 6d 65 20 69 73 6e 27 74 20 61 6e 79  t time isn't any
9050: 20 6e 65 77 65 72 20 74 68 61 6e 20 74 68 65 20   newer than the 
9060: 68 65 61 64 65 72 2c 20 77 65 20 69 6d 6d 65 64  header, we immed
9070: 69 61 74 65 6c 79 20 73 65 6e 64 20 62 61 63 6b  iately send back
9080: 0a 2a 2a 20 61 20 33 30 34 20 72 65 70 6c 79 20  .** a 304 reply 
9090: 61 6e 64 20 65 78 69 74 2e 0a 2a 2f 0a 76 6f 69  and exit..*/.voi
90a0: 64 20 63 67 69 5f 6d 6f 64 69 66 69 65 64 5f 73  d cgi_modified_s
90b0: 69 6e 63 65 28 74 69 6d 65 5f 74 20 6f 62 6a 65  ince(time_t obje
90c0: 63 74 54 69 6d 65 29 7b 0a 20 20 63 6f 6e 73 74  ctTime){.  const
90d0: 20 63 68 61 72 20 2a 7a 49 66 20 3d 20 50 28 22   char *zIf = P("
90e0: 48 54 54 50 5f 49 46 5f 4d 4f 44 49 46 49 45 44  HTTP_IF_MODIFIED
90f0: 5f 53 49 4e 43 45 22 29 3b 0a 20 20 69 66 28 20  _SINCE");.  if( 
9100: 7a 49 66 3d 3d 30 20 29 20 72 65 74 75 72 6e 3b  zIf==0 ) return;
9110: 0a 20 20 69 66 28 20 6f 62 6a 65 63 74 54 69 6d  .  if( objectTim
9120: 65 20 3e 20 63 67 69 5f 72 66 63 38 32 32 5f 70  e > cgi_rfc822_p
9130: 61 72 73 65 64 61 74 65 28 7a 49 66 29 20 29 20  arsedate(zIf) ) 
9140: 72 65 74 75 72 6e 3b 0a 20 20 63 67 69 5f 73 65  return;.  cgi_se
9150: 74 5f 73 74 61 74 75 73 28 33 30 34 2c 22 4e 6f  t_status(304,"No
9160: 74 20 4d 6f 64 69 66 69 65 64 22 29 3b 0a 20 20  t Modified");.  
9170: 63 67 69 5f 72 65 73 65 74 5f 63 6f 6e 74 65 6e  cgi_reset_conten
9180: 74 28 29 3b 0a 20 20 63 67 69 5f 72 65 70 6c 79  t();.  cgi_reply
9190: 28 29 3b 0a 20 20 65 78 69 74 28 30 29 3b 0a 7d  ();.  exit(0);.}
91a0: 0a                                               .