Hex Artifact Content
Not logged in

Artifact 0e2cb48261dc9fd1f9a8735ead105100a762c95a:

File src/xfer.c part of check-in [3cb547be2c] - Code to randomize messages from client to the server. by drh on 2008-07-26 14:02:33.

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 37 20 44 2e 20 52 69 63 68  (c) 2007 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 63 6f 64 65 20 74   contains code t
0380: 6f 20 69 6d 70 6c 65 6d 65 6e 74 20 74 68 65 20  o implement the 
0390: 66 69 6c 65 20 74 72 61 6e 73 66 65 72 20 70 72  file transfer pr
03a0: 6f 74 6f 63 6f 6c 2e 0a 2a 2f 0a 23 69 6e 63 6c  otocol..*/.#incl
03b0: 75 64 65 20 22 63 6f 6e 66 69 67 2e 68 22 0a 23  ude "config.h".#
03c0: 69 6e 63 6c 75 64 65 20 22 78 66 65 72 2e 68 22  include "xfer.h"
03d0: 0a 0a 2f 2a 0a 2a 2a 20 54 68 69 73 20 73 74 72  ../*.** This str
03e0: 75 63 74 75 72 65 20 68 6f 6c 64 73 20 69 6e 66  ucture holds inf
03f0: 6f 72 6d 61 74 69 6f 6e 20 61 62 6f 75 74 20 74  ormation about t
0400: 68 65 20 63 75 72 72 65 6e 74 20 73 74 61 74 65  he current state
0410: 20 6f 66 20 65 69 74 68 65 72 0a 2a 2a 20 61 20   of either.** a 
0420: 63 6c 69 65 6e 74 20 6f 72 20 61 20 73 65 72 76  client or a serv
0430: 65 72 20 74 68 61 74 20 69 73 20 70 61 72 74 69  er that is parti
0440: 63 69 70 61 74 69 6e 67 20 69 6e 20 78 66 65 72  cipating in xfer
0450: 2e 0a 2a 2f 0a 74 79 70 65 64 65 66 20 73 74 72  ..*/.typedef str
0460: 75 63 74 20 58 66 65 72 20 58 66 65 72 3b 0a 73  uct Xfer Xfer;.s
0470: 74 72 75 63 74 20 58 66 65 72 20 7b 0a 20 20 42  truct Xfer {.  B
0480: 6c 6f 62 20 2a 70 49 6e 3b 20 20 20 20 20 20 20  lob *pIn;       
0490: 20 20 20 2f 2a 20 49 6e 70 75 74 20 74 65 78 74     /* Input text
04a0: 20 66 72 6f 6d 20 74 68 65 20 6f 74 68 65 72 20   from the other 
04b0: 73 69 64 65 20 2a 2f 0a 20 20 42 6c 6f 62 20 2a  side */.  Blob *
04c0: 70 4f 75 74 3b 20 20 20 20 20 20 20 20 20 2f 2a  pOut;         /*
04d0: 20 43 6f 6d 70 6f 73 65 20 6f 75 72 20 72 65 70   Compose our rep
04e0: 6c 79 20 68 65 72 65 20 2a 2f 0a 20 20 42 6c 6f  ly here */.  Blo
04f0: 62 20 6c 69 6e 65 3b 20 20 20 20 20 20 20 20 20  b line;         
0500: 20 2f 2a 20 54 68 65 20 63 75 72 72 65 6e 74 20   /* The current 
0510: 6c 69 6e 65 20 6f 66 20 69 6e 70 75 74 20 2a 2f  line of input */
0520: 0a 20 20 42 6c 6f 62 20 61 54 6f 6b 65 6e 5b 35  .  Blob aToken[5
0530: 5d 3b 20 20 20 20 20 2f 2a 20 54 6f 6b 65 6e 69  ];     /* Tokeni
0540: 7a 65 64 20 76 65 72 73 69 6f 6e 20 6f 66 20 6c  zed version of l
0550: 69 6e 65 20 2a 2f 0a 20 20 42 6c 6f 62 20 65 72  ine */.  Blob er
0560: 72 3b 20 20 20 20 20 20 20 20 20 20 20 2f 2a 20  r;           /* 
0570: 45 72 72 6f 72 20 6d 65 73 73 61 67 65 20 74 65  Error message te
0580: 78 74 20 2a 2f 0a 20 20 69 6e 74 20 6e 54 6f 6b  xt */.  int nTok
0590: 65 6e 3b 20 20 20 20 20 20 20 20 20 2f 2a 20 4e  en;         /* N
05a0: 75 6d 62 65 72 20 6f 66 20 74 6f 6b 65 6e 73 20  umber of tokens 
05b0: 69 6e 20 6c 69 6e 65 20 2a 2f 0a 20 20 69 6e 74  in line */.  int
05c0: 20 6e 49 47 6f 74 53 65 6e 74 3b 20 20 20 20 20   nIGotSent;     
05d0: 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 22 69   /* Number of "i
05e0: 67 6f 74 22 20 63 61 72 64 73 20 73 65 6e 74 20  got" cards sent 
05f0: 2a 2f 0a 20 20 69 6e 74 20 6e 47 69 6d 6d 65 53  */.  int nGimmeS
0600: 65 6e 74 3b 20 20 20 20 20 2f 2a 20 4e 75 6d 62  ent;     /* Numb
0610: 65 72 20 6f 66 20 67 69 6d 6d 65 20 63 61 72 64  er of gimme card
0620: 73 20 73 65 6e 74 20 2a 2f 0a 20 20 69 6e 74 20  s sent */.  int 
0630: 6e 46 69 6c 65 53 65 6e 74 3b 20 20 20 20 20 20  nFileSent;      
0640: 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 66 69 6c  /* Number of fil
0650: 65 73 20 73 65 6e 74 20 2a 2f 0a 20 20 69 6e 74  es sent */.  int
0660: 20 6e 44 65 6c 74 61 53 65 6e 74 3b 20 20 20 20   nDeltaSent;    
0670: 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20 64 65   /* Number of de
0680: 6c 74 61 73 20 73 65 6e 74 20 2a 2f 0a 20 20 69  ltas sent */.  i
0690: 6e 74 20 6e 46 69 6c 65 52 63 76 64 3b 20 20 20  nt nFileRcvd;   
06a0: 20 20 20 2f 2a 20 4e 75 6d 62 65 72 20 6f 66 20     /* Number of 
06b0: 66 69 6c 65 73 20 72 65 63 65 69 76 65 64 20 2a  files received *
06c0: 2f 0a 20 20 69 6e 74 20 6e 44 65 6c 74 61 52 63  /.  int nDeltaRc
06d0: 76 64 3b 20 20 20 20 20 2f 2a 20 4e 75 6d 62 65  vd;     /* Numbe
06e0: 72 20 6f 66 20 64 65 6c 74 61 73 20 72 65 63 65  r of deltas rece
06f0: 69 76 65 64 20 2a 2f 0a 20 20 69 6e 74 20 6e 44  ived */.  int nD
0700: 61 6e 67 6c 69 6e 67 46 69 6c 65 3b 20 20 2f 2a  anglingFile;  /*
0710: 20 4e 75 6d 62 65 72 20 6f 66 20 64 61 6e 67 6c   Number of dangl
0720: 69 6e 67 20 64 65 6c 74 61 73 20 72 65 63 65 69  ing deltas recei
0730: 76 65 64 20 2a 2f 0a 20 20 69 6e 74 20 6d 78 53  ved */.  int mxS
0740: 65 6e 64 3b 20 20 20 20 20 20 20 20 20 2f 2a 20  end;         /* 
0750: 53 74 6f 70 20 73 65 6e 64 69 6e 67 20 22 66 69  Stop sending "fi
0760: 6c 65 22 20 77 69 74 68 20 70 4f 75 74 20 72 65  le" with pOut re
0770: 61 63 68 65 73 20 74 68 69 73 20 73 69 7a 65 20  aches this size 
0780: 2a 2f 0a 7d 3b 0a 0a 0a 2f 2a 0a 2a 2a 20 54 68  */.};.../*.** Th
0790: 65 20 69 6e 70 75 74 20 62 6c 6f 62 20 63 6f 6e  e input blob con
07a0: 74 61 69 6e 73 20 61 20 55 55 49 44 2e 20 20 43  tains a UUID.  C
07b0: 6f 6e 76 65 72 74 20 69 74 20 69 6e 74 6f 20 61  onvert it into a
07c0: 20 72 65 63 6f 72 64 20 49 44 2e 0a 2a 2a 20 43   record ID..** C
07d0: 72 65 61 74 65 20 61 20 70 68 61 6e 74 6f 6d 20  reate a phantom 
07e0: 72 65 63 6f 72 64 20 69 66 20 6e 6f 20 70 72 69  record if no pri
07f0: 6f 72 20 72 65 63 6f 72 64 20 65 78 69 73 74 73  or record exists
0800: 20 61 6e 64 0a 2a 2a 20 70 68 61 6e 74 6f 6d 69   and.** phantomi
0810: 7a 65 20 69 73 20 74 72 75 65 2e 0a 2a 2a 0a 2a  ze is true..**.*
0820: 2a 20 43 6f 6d 70 61 72 65 20 74 6f 20 75 75 69  * Compare to uui
0830: 64 5f 74 6f 5f 72 69 64 28 29 2e 20 20 54 68 69  d_to_rid().  Thi
0840: 73 20 72 6f 75 74 69 6e 65 20 74 61 6b 65 73 20  s routine takes 
0850: 61 20 62 6c 6f 62 20 61 72 67 75 6d 65 6e 74 0a  a blob argument.
0860: 2a 2a 20 61 6e 64 20 64 6f 65 73 20 6c 65 73 73  ** and does less
0870: 20 65 72 72 6f 72 20 63 68 65 63 6b 69 6e 67 2e   error checking.
0880: 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74 20 72  .*/.static int r
0890: 69 64 5f 66 72 6f 6d 5f 75 75 69 64 28 42 6c 6f  id_from_uuid(Blo
08a0: 62 20 2a 70 55 75 69 64 2c 20 69 6e 74 20 70 68  b *pUuid, int ph
08b0: 61 6e 74 6f 6d 69 7a 65 29 7b 0a 20 20 73 74 61  antomize){.  sta
08c0: 74 69 63 20 53 74 6d 74 20 71 3b 0a 20 20 69 6e  tic Stmt q;.  in
08d0: 74 20 72 69 64 3b 0a 0a 20 20 64 62 5f 73 74 61  t rid;..  db_sta
08e0: 74 69 63 5f 70 72 65 70 61 72 65 28 26 71 2c 20  tic_prepare(&q, 
08f0: 22 53 45 4c 45 43 54 20 72 69 64 20 46 52 4f 4d  "SELECT rid FROM
0900: 20 62 6c 6f 62 20 57 48 45 52 45 20 75 75 69 64   blob WHERE uuid
0910: 3d 3a 75 75 69 64 22 29 3b 0a 20 20 64 62 5f 62  =:uuid");.  db_b
0920: 69 6e 64 5f 73 74 72 28 26 71 2c 20 22 3a 75 75  ind_str(&q, ":uu
0930: 69 64 22 2c 20 70 55 75 69 64 29 3b 0a 20 20 69  id", pUuid);.  i
0940: 66 28 20 64 62 5f 73 74 65 70 28 26 71 29 3d 3d  f( db_step(&q)==
0950: 53 51 4c 49 54 45 5f 52 4f 57 20 29 7b 0a 20 20  SQLITE_ROW ){.  
0960: 20 20 72 69 64 20 3d 20 64 62 5f 63 6f 6c 75 6d    rid = db_colum
0970: 6e 5f 69 6e 74 28 26 71 2c 20 30 29 3b 0a 20 20  n_int(&q, 0);.  
0980: 7d 65 6c 73 65 7b 0a 20 20 20 20 72 69 64 20 3d  }else{.    rid =
0990: 20 30 3b 0a 20 20 7d 0a 20 20 64 62 5f 72 65 73   0;.  }.  db_res
09a0: 65 74 28 26 71 29 3b 0a 20 20 69 66 28 20 72 69  et(&q);.  if( ri
09b0: 64 3d 3d 30 20 26 26 20 70 68 61 6e 74 6f 6d 69  d==0 && phantomi
09c0: 7a 65 20 29 7b 0a 20 20 20 20 72 69 64 20 3d 20  ze ){.    rid = 
09d0: 63 6f 6e 74 65 6e 74 5f 6e 65 77 28 62 6c 6f 62  content_new(blob
09e0: 5f 73 74 72 28 70 55 75 69 64 29 29 3b 0a 20 20  _str(pUuid));.  
09f0: 7d 0a 20 20 72 65 74 75 72 6e 20 72 69 64 3b 0a  }.  return rid;.
0a00: 7d 0a 0a 2f 2a 0a 2a 2a 20 52 65 6d 65 6d 62 65  }../*.** Remembe
0a10: 72 20 74 68 61 74 20 74 68 65 20 6f 74 68 65 72  r that the other
0a20: 20 73 69 64 65 20 6f 66 20 74 68 65 20 63 6f 6e   side of the con
0a30: 6e 65 63 74 69 6f 6e 20 61 6c 72 65 61 64 79 20  nection already 
0a40: 68 61 73 20 61 20 63 6f 70 79 0a 2a 2a 20 6f 66  has a copy.** of
0a50: 20 74 68 65 20 66 69 6c 65 20 72 69 64 2e 0a 2a   the file rid..*
0a60: 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20 72 65  /.static void re
0a70: 6d 6f 74 65 5f 68 61 73 28 69 6e 74 20 72 69 64  mote_has(int rid
0a80: 29 7b 0a 20 20 69 66 28 20 72 69 64 20 29 20 64  ){.  if( rid ) d
0a90: 62 5f 6d 75 6c 74 69 5f 65 78 65 63 28 22 49 4e  b_multi_exec("IN
0aa0: 53 45 52 54 20 4f 52 20 49 47 4e 4f 52 45 20 49  SERT OR IGNORE I
0ab0: 4e 54 4f 20 6f 6e 72 65 6d 6f 74 65 20 56 41 4c  NTO onremote VAL
0ac0: 55 45 53 28 25 64 29 22 2c 20 72 69 64 29 3b 0a  UES(%d)", rid);.
0ad0: 7d 0a 0a 2f 2a 0a 2a 2a 20 54 68 65 20 61 54 6f  }../*.** The aTo
0ae0: 6b 65 6e 5b 30 2e 2e 6e 54 6f 6b 65 6e 2d 31 5d  ken[0..nToken-1]
0af0: 20 62 6c 6f 62 20 61 72 72 61 79 20 69 73 20 61   blob array is a
0b00: 20 70 61 72 73 65 20 6f 66 20 61 20 22 66 69 6c   parse of a "fil
0b10: 65 22 20 6c 69 6e 65 20 0a 2a 2a 20 6d 65 73 73  e" line .** mess
0b20: 61 67 65 2e 20 20 54 68 69 73 20 72 6f 75 74 69  age.  This routi
0b30: 6e 65 20 66 69 6e 69 73 68 65 73 20 70 61 72 73  ne finishes pars
0b40: 69 6e 67 20 74 68 61 74 20 6d 65 73 73 61 67 65  ing that message
0b50: 20 61 6e 64 20 64 6f 65 73 0a 2a 2a 20 61 20 72   and does.** a r
0b60: 65 63 6f 72 64 20 69 6e 73 65 72 74 20 6f 66 20  ecord insert of 
0b70: 74 68 65 20 66 69 6c 65 2e 0a 2a 2a 0a 2a 2a 20  the file..**.** 
0b80: 54 68 65 20 66 69 6c 65 20 6c 69 6e 65 20 69 73  The file line is
0b90: 20 69 6e 20 6f 6e 65 20 6f 66 20 74 68 65 20 66   in one of the f
0ba0: 6f 6c 6c 6f 77 69 6e 67 20 74 77 6f 20 66 6f 72  ollowing two for
0bb0: 6d 73 3a 0a 2a 2a 0a 2a 2a 20 20 20 20 20 20 66  ms:.**.**      f
0bc0: 69 6c 65 20 55 55 49 44 20 53 49 5a 45 20 5c 6e  ile UUID SIZE \n
0bd0: 20 43 4f 4e 54 45 4e 54 0a 2a 2a 20 20 20 20 20   CONTENT.**     
0be0: 20 66 69 6c 65 20 55 55 49 44 20 44 45 4c 54 41   file UUID DELTA
0bf0: 53 52 43 20 53 49 5a 45 20 5c 6e 20 43 4f 4e 54  SRC SIZE \n CONT
0c00: 45 4e 54 0a 2a 2a 0a 2a 2a 20 54 68 65 20 63 6f  ENT.**.** The co
0c10: 6e 74 65 6e 74 20 69 73 20 53 49 5a 45 20 62 79  ntent is SIZE by
0c20: 74 65 73 20 69 6d 6d 65 64 69 61 74 65 6c 79 20  tes immediately 
0c30: 66 6f 6c 6c 6f 77 69 6e 67 20 74 68 65 20 6e 65  following the ne
0c40: 77 6c 69 6e 65 2e 0a 2a 2a 20 49 66 20 44 45 4c  wline..** If DEL
0c50: 54 41 53 52 43 20 65 78 69 73 74 73 2c 20 74 68  TASRC exists, th
0c60: 65 6e 20 74 68 65 20 43 4f 4e 54 45 4e 54 20 69  en the CONTENT i
0c70: 73 20 61 20 64 65 6c 74 61 20 61 67 61 69 6e 73  s a delta agains
0c80: 74 20 74 68 65 0a 2a 2a 20 63 6f 6e 74 65 6e 74  t the.** content
0c90: 20 6f 66 20 44 45 4c 54 41 53 52 43 2e 0a 2a 2a   of DELTASRC..**
0ca0: 0a 2a 2a 20 49 66 20 61 6e 79 20 65 72 72 6f 72  .** If any error
0cb0: 20 6f 63 63 75 72 73 2c 20 77 72 69 74 65 20 61   occurs, write a
0cc0: 20 6d 65 73 73 61 67 65 20 69 6e 74 6f 20 70 45   message into pE
0cd0: 72 72 20 77 68 69 63 68 20 68 61 73 20 61 6c 72  rr which has alr
0ce0: 65 61 64 79 0a 2a 2a 20 62 65 20 69 6e 69 74 69  eady.** be initi
0cf0: 61 6c 69 7a 65 64 20 74 6f 20 61 6e 20 65 6d 70  alized to an emp
0d00: 74 79 20 73 74 72 69 6e 67 2e 0a 2a 2f 0a 73 74  ty string..*/.st
0d10: 61 74 69 63 20 76 6f 69 64 20 78 66 65 72 5f 61  atic void xfer_a
0d20: 63 63 65 70 74 5f 66 69 6c 65 28 58 66 65 72 20  ccept_file(Xfer 
0d30: 2a 70 58 66 65 72 29 7b 0a 20 20 69 6e 74 20 6e  *pXfer){.  int n
0d40: 3b 0a 20 20 69 6e 74 20 72 69 64 3b 0a 20 20 69  ;.  int rid;.  i
0d50: 6e 74 20 73 72 63 69 64 20 3d 20 30 3b 0a 20 20  nt srcid = 0;.  
0d60: 42 6c 6f 62 20 63 6f 6e 74 65 6e 74 2c 20 68 61  Blob content, ha
0d70: 73 68 3b 0a 20 20 0a 20 20 69 66 28 20 70 58 66  sh;.  .  if( pXf
0d80: 65 72 2d 3e 6e 54 6f 6b 65 6e 3c 33 20 0a 20 20  er->nToken<3 .  
0d90: 20 7c 7c 20 70 58 66 65 72 2d 3e 6e 54 6f 6b 65   || pXfer->nToke
0da0: 6e 3e 34 0a 20 20 20 7c 7c 20 21 62 6c 6f 62 5f  n>4.   || !blob_
0db0: 69 73 5f 75 75 69 64 28 26 70 58 66 65 72 2d 3e  is_uuid(&pXfer->
0dc0: 61 54 6f 6b 65 6e 5b 31 5d 29 0a 20 20 20 7c 7c  aToken[1]).   ||
0dd0: 20 21 62 6c 6f 62 5f 69 73 5f 69 6e 74 28 26 70   !blob_is_int(&p
0de0: 58 66 65 72 2d 3e 61 54 6f 6b 65 6e 5b 70 58 66  Xfer->aToken[pXf
0df0: 65 72 2d 3e 6e 54 6f 6b 65 6e 2d 31 5d 2c 20 26  er->nToken-1], &
0e00: 6e 29 0a 20 20 20 7c 7c 20 6e 3c 30 0a 20 20 20  n).   || n<0.   
0e10: 7c 7c 20 28 70 58 66 65 72 2d 3e 6e 54 6f 6b 65  || (pXfer->nToke
0e20: 6e 3d 3d 34 20 26 26 20 21 62 6c 6f 62 5f 69 73  n==4 && !blob_is
0e30: 5f 75 75 69 64 28 26 70 58 66 65 72 2d 3e 61 54  _uuid(&pXfer->aT
0e40: 6f 6b 65 6e 5b 32 5d 29 29 0a 20 20 29 7b 0a 20  oken[2])).  ){. 
0e50: 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66 28     blob_appendf(
0e60: 26 70 58 66 65 72 2d 3e 65 72 72 2c 20 22 6d 61  &pXfer->err, "ma
0e70: 6c 66 6f 72 6d 65 64 20 66 69 6c 65 20 6c 69 6e  lformed file lin
0e80: 65 22 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 3b  e");.    return;
0e90: 0a 20 20 7d 0a 20 20 62 6c 6f 62 5f 7a 65 72 6f  .  }.  blob_zero
0ea0: 28 26 63 6f 6e 74 65 6e 74 29 3b 0a 20 20 62 6c  (&content);.  bl
0eb0: 6f 62 5f 7a 65 72 6f 28 26 68 61 73 68 29 3b 0a  ob_zero(&hash);.
0ec0: 20 20 62 6c 6f 62 5f 65 78 74 72 61 63 74 28 70    blob_extract(p
0ed0: 58 66 65 72 2d 3e 70 49 6e 2c 20 6e 2c 20 26 63  Xfer->pIn, n, &c
0ee0: 6f 6e 74 65 6e 74 29 3b 0a 20 20 69 66 28 20 75  ontent);.  if( u
0ef0: 75 69 64 5f 69 73 5f 73 68 75 6e 6e 65 64 28 62  uid_is_shunned(b
0f00: 6c 6f 62 5f 73 74 72 28 26 70 58 66 65 72 2d 3e  lob_str(&pXfer->
0f10: 61 54 6f 6b 65 6e 5b 31 5d 29 29 20 29 7b 0a 20  aToken[1])) ){. 
0f20: 20 20 20 2f 2a 20 49 67 6e 6f 72 65 20 66 69 6c     /* Ignore fil
0f30: 65 73 20 74 68 61 74 20 68 61 76 65 20 62 65 65  es that have bee
0f40: 6e 20 73 68 75 6e 6e 65 64 20 2a 2f 0a 20 20 20  n shunned */.   
0f50: 20 72 65 74 75 72 6e 3b 0a 20 20 7d 0a 20 20 69   return;.  }.  i
0f60: 66 28 20 70 58 66 65 72 2d 3e 6e 54 6f 6b 65 6e  f( pXfer->nToken
0f70: 3d 3d 34 20 29 7b 0a 20 20 20 20 42 6c 6f 62 20  ==4 ){.    Blob 
0f80: 73 72 63 3b 0a 20 20 20 20 73 72 63 69 64 20 3d  src;.    srcid =
0f90: 20 72 69 64 5f 66 72 6f 6d 5f 75 75 69 64 28 26   rid_from_uuid(&
0fa0: 70 58 66 65 72 2d 3e 61 54 6f 6b 65 6e 5b 32 5d  pXfer->aToken[2]
0fb0: 2c 20 31 29 3b 0a 20 20 20 20 69 66 28 20 63 6f  , 1);.    if( co
0fc0: 6e 74 65 6e 74 5f 67 65 74 28 73 72 63 69 64 2c  ntent_get(srcid,
0fd0: 20 26 73 72 63 29 3d 3d 30 20 29 7b 0a 20 20 20   &src)==0 ){.   
0fe0: 20 20 20 63 6f 6e 74 65 6e 74 5f 70 75 74 28 26     content_put(&
0ff0: 63 6f 6e 74 65 6e 74 2c 20 62 6c 6f 62 5f 73 74  content, blob_st
1000: 72 28 26 70 58 66 65 72 2d 3e 61 54 6f 6b 65 6e  r(&pXfer->aToken
1010: 5b 31 5d 29 2c 20 73 72 63 69 64 29 3b 0a 20 20  [1]), srcid);.  
1020: 20 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66      blob_appendf
1030: 28 70 58 66 65 72 2d 3e 70 4f 75 74 2c 20 22 67  (pXfer->pOut, "g
1040: 69 6d 6d 65 20 25 62 5c 6e 22 2c 20 26 70 58 66  imme %b\n", &pXf
1050: 65 72 2d 3e 61 54 6f 6b 65 6e 5b 32 5d 29 3b 0a  er->aToken[2]);.
1060: 20 20 20 20 20 20 70 58 66 65 72 2d 3e 6e 47 69        pXfer->nGi
1070: 6d 6d 65 53 65 6e 74 2b 2b 3b 0a 20 20 20 20 20  mmeSent++;.     
1080: 20 70 58 66 65 72 2d 3e 6e 44 61 6e 67 6c 69 6e   pXfer->nDanglin
1090: 67 46 69 6c 65 2b 2b 3b 0a 20 20 20 20 20 20 72  gFile++;.      r
10a0: 65 74 75 72 6e 3b 0a 20 20 20 20 7d 0a 20 20 20  eturn;.    }.   
10b0: 20 70 58 66 65 72 2d 3e 6e 44 65 6c 74 61 52 63   pXfer->nDeltaRc
10c0: 76 64 2b 2b 3b 0a 20 20 20 20 62 6c 6f 62 5f 64  vd++;.    blob_d
10d0: 65 6c 74 61 5f 61 70 70 6c 79 28 26 73 72 63 2c  elta_apply(&src,
10e0: 20 26 63 6f 6e 74 65 6e 74 2c 20 26 63 6f 6e 74   &content, &cont
10f0: 65 6e 74 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 72  ent);.    blob_r
1100: 65 73 65 74 28 26 73 72 63 29 3b 0a 20 20 7d 65  eset(&src);.  }e
1110: 6c 73 65 7b 0a 20 20 20 20 70 58 66 65 72 2d 3e  lse{.    pXfer->
1120: 6e 46 69 6c 65 52 63 76 64 2b 2b 3b 0a 20 20 7d  nFileRcvd++;.  }
1130: 0a 20 20 73 68 61 31 73 75 6d 5f 62 6c 6f 62 28  .  sha1sum_blob(
1140: 26 63 6f 6e 74 65 6e 74 2c 20 26 68 61 73 68 29  &content, &hash)
1150: 3b 0a 20 20 69 66 28 20 21 62 6c 6f 62 5f 65 71  ;.  if( !blob_eq
1160: 5f 73 74 72 28 26 70 58 66 65 72 2d 3e 61 54 6f  _str(&pXfer->aTo
1170: 6b 65 6e 5b 31 5d 2c 20 62 6c 6f 62 5f 73 74 72  ken[1], blob_str
1180: 28 26 68 61 73 68 29 2c 20 2d 31 29 20 29 7b 0a  (&hash), -1) ){.
1190: 20 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66      blob_appendf
11a0: 28 26 70 58 66 65 72 2d 3e 65 72 72 2c 20 22 63  (&pXfer->err, "c
11b0: 6f 6e 74 65 6e 74 20 64 6f 65 73 20 6e 6f 74 20  ontent does not 
11c0: 6d 61 74 63 68 20 73 68 61 31 20 68 61 73 68 22  match sha1 hash"
11d0: 29 3b 0a 20 20 7d 0a 20 20 72 69 64 20 3d 20 63  );.  }.  rid = c
11e0: 6f 6e 74 65 6e 74 5f 70 75 74 28 26 63 6f 6e 74  ontent_put(&cont
11f0: 65 6e 74 2c 20 62 6c 6f 62 5f 73 74 72 28 26 68  ent, blob_str(&h
1200: 61 73 68 29 2c 20 30 29 3b 0a 20 20 62 6c 6f 62  ash), 0);.  blob
1210: 5f 72 65 73 65 74 28 26 68 61 73 68 29 3b 0a 20  _reset(&hash);. 
1220: 20 69 66 28 20 72 69 64 3d 3d 30 20 29 7b 0a 20   if( rid==0 ){. 
1230: 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66 28     blob_appendf(
1240: 26 70 58 66 65 72 2d 3e 65 72 72 2c 20 22 25 73  &pXfer->err, "%s
1250: 22 2c 20 67 2e 7a 45 72 72 4d 73 67 29 3b 0a 20  ", g.zErrMsg);. 
1260: 20 7d 65 6c 73 65 7b 0a 20 20 20 20 2f 2a 20 64   }else{.    /* d
1270: 62 5f 6d 75 6c 74 69 5f 65 78 65 63 28 22 44 45  b_multi_exec("DE
1280: 4c 45 54 45 20 46 52 4f 4d 20 70 68 61 6e 74 6f  LETE FROM phanto
1290: 6d 20 57 48 45 52 45 20 72 69 64 3d 25 64 22 2c  m WHERE rid=%d",
12a0: 20 72 69 64 29 3b 20 2a 2f 0a 20 20 20 20 6d 61   rid); */.    ma
12b0: 6e 69 66 65 73 74 5f 63 72 6f 73 73 6c 69 6e 6b  nifest_crosslink
12c0: 28 72 69 64 2c 20 26 63 6f 6e 74 65 6e 74 29 3b  (rid, &content);
12d0: 0a 20 20 7d 0a 20 20 72 65 6d 6f 74 65 5f 68 61  .  }.  remote_ha
12e0: 73 28 72 69 64 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a  s(rid);.}../*.**
12f0: 20 54 72 79 20 74 6f 20 73 65 6e 64 20 61 20 66   Try to send a f
1300: 69 6c 65 20 61 73 20 61 20 64 65 6c 74 61 20 61  ile as a delta a
1310: 67 61 69 6e 73 74 20 69 74 73 20 70 61 72 65 6e  gainst its paren
1320: 74 2e 0a 2a 2a 20 49 66 20 73 75 63 63 65 73 73  t..** If success
1330: 66 75 6c 2c 20 72 65 74 75 72 6e 20 74 68 65 20  ful, return the 
1340: 6e 75 6d 62 65 72 20 6f 66 20 62 79 74 65 73 20  number of bytes 
1350: 69 6e 20 74 68 65 20 64 65 6c 74 61 2e 0a 2a 2a  in the delta..**
1360: 20 49 66 20 77 65 20 63 61 6e 6e 6f 74 20 67 65   If we cannot ge
1370: 6e 65 72 61 74 65 20 61 6e 20 61 70 70 72 6f 70  nerate an approp
1380: 72 69 61 74 65 20 64 65 6c 74 61 2c 20 74 68 65  riate delta, the
1390: 6e 20 73 65 6e 64 0a 2a 2a 20 6e 6f 74 68 69 6e  n send.** nothin
13a0: 67 20 61 6e 64 20 72 65 74 75 72 6e 20 7a 65 72  g and return zer
13b0: 6f 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74  o..*/.static int
13c0: 20 73 65 6e 64 5f 64 65 6c 74 61 5f 70 61 72 65   send_delta_pare
13d0: 6e 74 28 0a 20 20 58 66 65 72 20 2a 70 58 66 65  nt(.  Xfer *pXfe
13e0: 72 2c 20 20 20 20 20 20 20 20 20 20 20 20 2f 2a  r,            /*
13f0: 20 54 68 65 20 74 72 61 6e 73 66 65 72 20 63 6f   The transfer co
1400: 6e 74 65 78 74 20 2a 2f 0a 20 20 69 6e 74 20 72  ntext */.  int r
1410: 69 64 2c 20 20 20 20 20 20 20 20 20 20 20 20 20  id,             
1420: 20 20 20 2f 2a 20 72 65 63 6f 72 64 20 69 64 20     /* record id 
1430: 6f 66 20 74 68 65 20 66 69 6c 65 20 74 6f 20 73  of the file to s
1440: 65 6e 64 20 2a 2f 0a 20 20 42 6c 6f 62 20 2a 70  end */.  Blob *p
1450: 43 6f 6e 74 65 6e 74 2c 20 20 20 20 20 20 20 20  Content,        
1460: 20 2f 2a 20 54 68 65 20 63 6f 6e 74 65 6e 74 20   /* The content 
1470: 6f 66 20 74 68 65 20 66 69 6c 65 20 74 6f 20 73  of the file to s
1480: 65 6e 64 20 2a 2f 0a 20 20 42 6c 6f 62 20 2a 70  end */.  Blob *p
1490: 55 75 69 64 20 20 20 20 20 20 20 20 20 20 20 20  Uuid            
14a0: 20 2f 2a 20 54 68 65 20 55 55 49 44 20 6f 66 20   /* The UUID of 
14b0: 74 68 65 20 66 69 6c 65 20 74 6f 20 73 65 6e 64  the file to send
14c0: 20 2a 2f 0a 29 7b 0a 20 20 73 74 61 74 69 63 20   */.){.  static 
14d0: 63 6f 6e 73 74 20 63 68 61 72 20 2a 61 7a 51 75  const char *azQu
14e0: 65 72 79 5b 5d 20 3d 20 7b 0a 20 20 20 20 22 53  ery[] = {.    "S
14f0: 45 4c 45 43 54 20 70 69 64 20 46 52 4f 4d 20 70  ELECT pid FROM p
1500: 6c 69 6e 6b 20 78 22 0a 20 20 20 20 22 20 57 48  link x".    " WH
1510: 45 52 45 20 63 69 64 3d 25 64 22 0a 20 20 20 20  ERE cid=%d".    
1520: 22 20 20 20 41 4e 44 20 4e 4f 54 20 45 58 49 53  "   AND NOT EXIS
1530: 54 53 28 53 45 4c 45 43 54 20 31 20 46 52 4f 4d  TS(SELECT 1 FROM
1540: 20 70 68 61 6e 74 6f 6d 20 57 48 45 52 45 20 72   phantom WHERE r
1550: 69 64 3d 70 69 64 29 22 0a 20 20 20 20 22 20 20  id=pid)".    "  
1560: 20 41 4e 44 20 4e 4f 54 20 45 58 49 53 54 53 28   AND NOT EXISTS(
1570: 53 45 4c 45 43 54 20 31 20 46 52 4f 4d 20 70 6c  SELECT 1 FROM pl
1580: 69 6e 6b 20 79 22 0a 20 20 20 20 20 20 20 20 20  ink y".         
1590: 20 20 20 20 20 20 20 20 20 20 20 20 20 22 20 57               " W
15a0: 48 45 52 45 20 79 2e 70 69 64 3d 78 2e 63 69 64  HERE y.pid=x.cid
15b0: 20 41 4e 44 20 79 2e 63 69 64 3d 78 2e 70 69 64   AND y.cid=x.pid
15c0: 29 22 2c 0a 0a 20 20 20 20 22 53 45 4c 45 43 54  )",..    "SELECT
15d0: 20 70 69 64 20 46 52 4f 4d 20 6d 6c 69 6e 6b 20   pid FROM mlink 
15e0: 78 22 0a 20 20 20 20 22 20 57 48 45 52 45 20 66  x".    " WHERE f
15f0: 69 64 3d 25 64 22 0a 20 20 20 20 22 20 20 20 41  id=%d".    "   A
1600: 4e 44 20 4e 4f 54 20 45 58 49 53 54 53 28 53 45  ND NOT EXISTS(SE
1610: 4c 45 43 54 20 31 20 46 52 4f 4d 20 70 68 61 6e  LECT 1 FROM phan
1620: 74 6f 6d 20 57 48 45 52 45 20 72 69 64 3d 70 69  tom WHERE rid=pi
1630: 64 29 22 0a 20 20 20 20 22 20 20 20 41 4e 44 20  d)".    "   AND 
1640: 4e 4f 54 20 45 58 49 53 54 53 28 53 45 4c 45 43  NOT EXISTS(SELEC
1650: 54 20 31 20 46 52 4f 4d 20 6d 6c 69 6e 6b 20 79  T 1 FROM mlink y
1660: 22 0a 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ".              
1670: 20 20 20 20 20 20 20 22 20 20 57 48 45 52 45 20         "  WHERE 
1680: 79 2e 70 69 64 3d 78 2e 66 69 64 20 41 4e 44 20  y.pid=x.fid AND 
1690: 79 2e 66 69 64 3d 78 2e 70 69 64 29 22 0a 20 20  y.fid=x.pid)".  
16a0: 7d 3b 0a 20 20 69 6e 74 20 69 3b 0a 20 20 42 6c  };.  int i;.  Bl
16b0: 6f 62 20 73 72 63 2c 20 64 65 6c 74 61 3b 0a 20  ob src, delta;. 
16c0: 20 69 6e 74 20 73 69 7a 65 20 3d 20 30 3b 0a 20   int size = 0;. 
16d0: 20 69 6e 74 20 73 72 63 49 64 20 3d 20 30 3b 0a   int srcId = 0;.
16e0: 0a 20 20 66 6f 72 28 69 3d 30 3b 20 73 72 63 49  .  for(i=0; srcI
16f0: 64 3d 3d 30 20 26 26 20 69 3c 63 6f 75 6e 74 28  d==0 && i<count(
1700: 61 7a 51 75 65 72 79 29 3b 20 69 2b 2b 29 7b 0a  azQuery); i++){.
1710: 20 20 20 20 73 72 63 49 64 20 3d 20 64 62 5f 69      srcId = db_i
1720: 6e 74 28 30 2c 20 61 7a 51 75 65 72 79 5b 69 5d  nt(0, azQuery[i]
1730: 2c 20 72 69 64 29 3b 0a 20 20 7d 0a 20 20 69 66  , rid);.  }.  if
1740: 28 20 73 72 63 49 64 3e 30 20 26 26 20 63 6f 6e  ( srcId>0 && con
1750: 74 65 6e 74 5f 67 65 74 28 73 72 63 49 64 2c 20  tent_get(srcId, 
1760: 26 73 72 63 29 20 29 7b 0a 20 20 20 20 63 68 61  &src) ){.    cha
1770: 72 20 2a 7a 55 75 69 64 20 3d 20 64 62 5f 74 65  r *zUuid = db_te
1780: 78 74 28 30 2c 20 22 53 45 4c 45 43 54 20 75 75  xt(0, "SELECT uu
1790: 69 64 20 46 52 4f 4d 20 62 6c 6f 62 20 57 48 45  id FROM blob WHE
17a0: 52 45 20 72 69 64 3d 25 64 22 2c 20 73 72 63 49  RE rid=%d", srcI
17b0: 64 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 64 65 6c  d);.    blob_del
17c0: 74 61 5f 63 72 65 61 74 65 28 26 73 72 63 2c 20  ta_create(&src, 
17d0: 70 43 6f 6e 74 65 6e 74 2c 20 26 64 65 6c 74 61  pContent, &delta
17e0: 29 3b 0a 20 20 20 20 73 69 7a 65 20 3d 20 62 6c  );.    size = bl
17f0: 6f 62 5f 73 69 7a 65 28 26 64 65 6c 74 61 29 3b  ob_size(&delta);
1800: 0a 20 20 20 20 69 66 28 20 73 69 7a 65 3e 3d 62  .    if( size>=b
1810: 6c 6f 62 5f 73 69 7a 65 28 70 43 6f 6e 74 65 6e  lob_size(pConten
1820: 74 29 2d 35 30 20 29 7b 0a 20 20 20 20 20 20 73  t)-50 ){.      s
1830: 69 7a 65 20 3d 20 30 3b 0a 20 20 20 20 7d 65 6c  ize = 0;.    }el
1840: 73 65 20 69 66 28 20 75 75 69 64 5f 69 73 5f 73  se if( uuid_is_s
1850: 68 75 6e 6e 65 64 28 7a 55 75 69 64 29 20 29 7b  hunned(zUuid) ){
1860: 0a 20 20 20 20 20 20 73 69 7a 65 20 3d 20 30 3b  .      size = 0;
1870: 0a 20 20 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20  .    }else{.    
1880: 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66 28 70    blob_appendf(p
1890: 58 66 65 72 2d 3e 70 4f 75 74 2c 20 22 66 69 6c  Xfer->pOut, "fil
18a0: 65 20 25 62 20 25 73 20 25 64 5c 6e 22 2c 20 70  e %b %s %d\n", p
18b0: 55 75 69 64 2c 20 7a 55 75 69 64 2c 20 73 69 7a  Uuid, zUuid, siz
18c0: 65 29 3b 0a 20 20 20 20 20 20 62 6c 6f 62 5f 61  e);.      blob_a
18d0: 70 70 65 6e 64 28 70 58 66 65 72 2d 3e 70 4f 75  ppend(pXfer->pOu
18e0: 74 2c 20 62 6c 6f 62 5f 62 75 66 66 65 72 28 26  t, blob_buffer(&
18f0: 64 65 6c 74 61 29 2c 20 73 69 7a 65 29 3b 0a 20  delta), size);. 
1900: 20 20 20 20 20 2f 2a 20 62 6c 6f 62 5f 61 70 70       /* blob_app
1910: 65 6e 64 66 28 70 58 66 65 72 2d 3e 70 4f 75 74  endf(pXfer->pOut
1920: 2c 20 22 5c 6e 22 2c 20 31 29 3b 20 2a 2f 0a 20  , "\n", 1); */. 
1930: 20 20 20 7d 0a 20 20 20 20 62 6c 6f 62 5f 72 65     }.    blob_re
1940: 73 65 74 28 26 64 65 6c 74 61 29 3b 0a 20 20 20  set(&delta);.   
1950: 20 66 72 65 65 28 7a 55 75 69 64 29 3b 0a 20 20   free(zUuid);.  
1960: 20 20 62 6c 6f 62 5f 72 65 73 65 74 28 26 73 72    blob_reset(&sr
1970: 63 29 3b 0a 20 20 7d 0a 20 20 72 65 74 75 72 6e  c);.  }.  return
1980: 20 73 69 7a 65 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20   size;.}../*.** 
1990: 54 72 79 20 74 6f 20 73 65 6e 64 20 61 20 66 69  Try to send a fi
19a0: 6c 65 20 61 73 20 61 20 6e 61 74 69 76 65 20 64  le as a native d
19b0: 65 6c 74 61 2e 20 20 0a 2a 2a 20 49 66 20 73 75  elta.  .** If su
19c0: 63 63 65 73 73 66 75 6c 2c 20 72 65 74 75 72 6e  ccessful, return
19d0: 20 74 68 65 20 6e 75 6d 62 65 72 20 6f 66 20 62   the number of b
19e0: 79 74 65 73 20 69 6e 20 74 68 65 20 64 65 6c 74  ytes in the delt
19f0: 61 2e 0a 2a 2a 20 49 66 20 77 65 20 63 61 6e 6e  a..** If we cann
1a00: 6f 74 20 67 65 6e 65 72 61 74 65 20 61 6e 20 61  ot generate an a
1a10: 70 70 72 6f 70 72 69 61 74 65 20 64 65 6c 74 61  ppropriate delta
1a20: 2c 20 74 68 65 6e 20 73 65 6e 64 0a 2a 2a 20 6e  , then send.** n
1a30: 6f 74 68 69 6e 67 20 61 6e 64 20 72 65 74 75 72  othing and retur
1a40: 6e 20 7a 65 72 6f 2e 0a 2a 2f 0a 73 74 61 74 69  n zero..*/.stati
1a50: 63 20 69 6e 74 20 73 65 6e 64 5f 64 65 6c 74 61  c int send_delta
1a60: 5f 6e 61 74 69 76 65 28 0a 20 20 58 66 65 72 20  _native(.  Xfer 
1a70: 2a 70 58 66 65 72 2c 20 20 20 20 20 20 20 20 20  *pXfer,         
1a80: 20 20 20 2f 2a 20 54 68 65 20 74 72 61 6e 73 66     /* The transf
1a90: 65 72 20 63 6f 6e 74 65 78 74 20 2a 2f 0a 20 20  er context */.  
1aa0: 69 6e 74 20 72 69 64 2c 20 20 20 20 20 20 20 20  int rid,        
1ab0: 20 20 20 20 20 20 20 20 2f 2a 20 72 65 63 6f 72          /* recor
1ac0: 64 20 69 64 20 6f 66 20 74 68 65 20 66 69 6c 65  d id of the file
1ad0: 20 74 6f 20 73 65 6e 64 20 2a 2f 0a 20 20 42 6c   to send */.  Bl
1ae0: 6f 62 20 2a 70 55 75 69 64 20 20 20 20 20 20 20  ob *pUuid       
1af0: 20 20 20 20 20 20 2f 2a 20 54 68 65 20 55 55 49        /* The UUI
1b00: 44 20 6f 66 20 74 68 65 20 66 69 6c 65 20 74 6f  D of the file to
1b10: 20 73 65 6e 64 20 2a 2f 0a 29 7b 0a 20 20 42 6c   send */.){.  Bl
1b20: 6f 62 20 73 72 63 2c 20 64 65 6c 74 61 3b 0a 20  ob src, delta;. 
1b30: 20 69 6e 74 20 73 69 7a 65 20 3d 20 30 3b 0a 20   int size = 0;. 
1b40: 20 69 6e 74 20 73 72 63 49 64 3b 0a 0a 20 20 73   int srcId;..  s
1b50: 72 63 49 64 20 3d 20 64 62 5f 69 6e 74 28 30 2c  rcId = db_int(0,
1b60: 20 22 53 45 4c 45 43 54 20 73 72 63 69 64 20 46   "SELECT srcid F
1b70: 52 4f 4d 20 64 65 6c 74 61 20 57 48 45 52 45 20  ROM delta WHERE 
1b80: 72 69 64 3d 25 64 22 2c 20 72 69 64 29 3b 0a 20  rid=%d", rid);. 
1b90: 20 69 66 28 20 73 72 63 49 64 3e 30 20 29 7b 0a   if( srcId>0 ){.
1ba0: 20 20 20 20 62 6c 6f 62 5f 7a 65 72 6f 28 26 73      blob_zero(&s
1bb0: 72 63 29 3b 0a 20 20 20 20 64 62 5f 62 6c 6f 62  rc);.    db_blob
1bc0: 28 26 73 72 63 2c 20 22 53 45 4c 45 43 54 20 75  (&src, "SELECT u
1bd0: 75 69 64 20 46 52 4f 4d 20 62 6c 6f 62 20 57 48  uid FROM blob WH
1be0: 45 52 45 20 72 69 64 3d 25 64 22 2c 20 73 72 63  ERE rid=%d", src
1bf0: 49 64 29 3b 0a 20 20 20 20 69 66 28 20 75 75 69  Id);.    if( uui
1c00: 64 5f 69 73 5f 73 68 75 6e 6e 65 64 28 62 6c 6f  d_is_shunned(blo
1c10: 62 5f 73 74 72 28 26 73 72 63 29 29 20 29 7b 0a  b_str(&src)) ){.
1c20: 20 20 20 20 20 20 62 6c 6f 62 5f 72 65 73 65 74        blob_reset
1c30: 28 26 73 72 63 29 3b 0a 20 20 20 20 20 20 72 65  (&src);.      re
1c40: 74 75 72 6e 20 30 3b 0a 20 20 20 20 7d 0a 20 20  turn 0;.    }.  
1c50: 20 20 62 6c 6f 62 5f 7a 65 72 6f 28 26 64 65 6c    blob_zero(&del
1c60: 74 61 29 3b 0a 20 20 20 20 64 62 5f 62 6c 6f 62  ta);.    db_blob
1c70: 28 26 64 65 6c 74 61 2c 20 22 53 45 4c 45 43 54  (&delta, "SELECT
1c80: 20 63 6f 6e 74 65 6e 74 20 46 52 4f 4d 20 62 6c   content FROM bl
1c90: 6f 62 20 57 48 45 52 45 20 72 69 64 3d 25 64 22  ob WHERE rid=%d"
1ca0: 2c 20 72 69 64 29 3b 0a 20 20 20 20 62 6c 6f 62  , rid);.    blob
1cb0: 5f 75 6e 63 6f 6d 70 72 65 73 73 28 26 64 65 6c  _uncompress(&del
1cc0: 74 61 2c 20 26 64 65 6c 74 61 29 3b 0a 20 20 20  ta, &delta);.   
1cd0: 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66 28 70 58   blob_appendf(pX
1ce0: 66 65 72 2d 3e 70 4f 75 74 2c 20 22 66 69 6c 65  fer->pOut, "file
1cf0: 20 25 62 20 25 62 20 25 64 5c 6e 22 2c 0a 20 20   %b %b %d\n",.  
1d00: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 70 55                pU
1d10: 75 69 64 2c 20 26 73 72 63 2c 20 62 6c 6f 62 5f  uid, &src, blob_
1d20: 73 69 7a 65 28 26 64 65 6c 74 61 29 29 3b 0a 20  size(&delta));. 
1d30: 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 28 70     blob_append(p
1d40: 58 66 65 72 2d 3e 70 4f 75 74 2c 20 62 6c 6f 62  Xfer->pOut, blob
1d50: 5f 62 75 66 66 65 72 28 26 64 65 6c 74 61 29 2c  _buffer(&delta),
1d60: 20 62 6c 6f 62 5f 73 69 7a 65 28 26 64 65 6c 74   blob_size(&delt
1d70: 61 29 29 3b 0a 20 20 20 20 73 69 7a 65 20 3d 20  a));.    size = 
1d80: 62 6c 6f 62 5f 73 69 7a 65 28 26 64 65 6c 74 61  blob_size(&delta
1d90: 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 72 65 73 65  );.    blob_rese
1da0: 74 28 26 64 65 6c 74 61 29 3b 0a 20 20 20 20 62  t(&delta);.    b
1db0: 6c 6f 62 5f 72 65 73 65 74 28 26 73 72 63 29 3b  lob_reset(&src);
1dc0: 0a 20 20 7d 65 6c 73 65 7b 0a 20 20 20 20 73 69  .  }else{.    si
1dd0: 7a 65 20 3d 20 30 3b 0a 20 20 7d 0a 20 20 72 65  ze = 0;.  }.  re
1de0: 74 75 72 6e 20 73 69 7a 65 3b 0a 7d 0a 0a 2f 2a  turn size;.}../*
1df0: 0a 2a 2a 20 53 65 6e 64 20 74 68 65 20 66 69 6c  .** Send the fil
1e00: 65 20 69 64 65 6e 74 69 66 69 65 64 20 62 79 20  e identified by 
1e10: 72 69 64 2e 0a 2a 2a 0a 2a 2a 20 54 68 65 20 70  rid..**.** The p
1e20: 55 75 69 64 20 63 61 6e 20 62 65 20 4e 55 4c 4c  Uuid can be NULL
1e30: 20 69 6e 20 77 68 69 63 68 20 63 61 73 65 20 74   in which case t
1e40: 68 65 20 63 6f 72 72 65 63 74 20 55 55 49 44 20  he correct UUID 
1e50: 69 73 20 63 6f 6d 70 75 74 65 64 0a 2a 2a 20 66  is computed.** f
1e60: 72 6f 6d 20 74 68 65 20 72 69 64 2e 0a 2a 2a 0a  rom the rid..**.
1e70: 2a 2a 20 54 72 79 20 74 6f 20 73 65 6e 64 20 74  ** Try to send t
1e80: 68 65 20 66 69 6c 65 20 61 73 20 61 20 6e 61 74  he file as a nat
1e90: 69 76 65 20 64 65 6c 74 61 20 69 66 20 6e 61 74  ive delta if nat
1ea0: 69 76 65 44 65 6c 74 61 20 69 73 20 74 72 75 65  iveDelta is true
1eb0: 2c 20 6f 72 0a 2a 2a 20 61 73 20 61 20 70 61 72  , or.** as a par
1ec0: 65 6e 74 20 64 65 6c 74 61 20 69 66 20 6e 61 74  ent delta if nat
1ed0: 69 76 65 44 65 6c 74 61 20 69 73 20 66 61 6c 73  iveDelta is fals
1ee0: 65 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69  e..*/.static voi
1ef0: 64 20 73 65 6e 64 5f 66 69 6c 65 28 58 66 65 72  d send_file(Xfer
1f00: 20 2a 70 58 66 65 72 2c 20 69 6e 74 20 72 69 64   *pXfer, int rid
1f10: 2c 20 42 6c 6f 62 20 2a 70 55 75 69 64 2c 20 69  , Blob *pUuid, i
1f20: 6e 74 20 6e 61 74 69 76 65 44 65 6c 74 61 29 7b  nt nativeDelta){
1f30: 0a 20 20 42 6c 6f 62 20 63 6f 6e 74 65 6e 74 2c  .  Blob content,
1f40: 20 75 75 69 64 3b 0a 20 20 69 6e 74 20 73 69 7a   uuid;.  int siz
1f50: 65 20 3d 20 30 3b 0a 0a 20 20 69 66 28 20 64 62  e = 0;..  if( db
1f60: 5f 65 78 69 73 74 73 28 22 53 45 4c 45 43 54 20  _exists("SELECT 
1f70: 31 20 46 52 4f 4d 20 6f 6e 72 65 6d 6f 74 65 20  1 FROM onremote 
1f80: 57 48 45 52 45 20 72 69 64 3d 25 64 22 2c 20 72  WHERE rid=%d", r
1f90: 69 64 29 20 29 7b 0a 20 20 20 20 20 72 65 74 75  id) ){.     retu
1fa0: 72 6e 3b 0a 20 20 7d 0a 20 20 62 6c 6f 62 5f 7a  rn;.  }.  blob_z
1fb0: 65 72 6f 28 26 75 75 69 64 29 3b 0a 20 20 64 62  ero(&uuid);.  db
1fc0: 5f 62 6c 6f 62 28 26 75 75 69 64 2c 20 22 53 45  _blob(&uuid, "SE
1fd0: 4c 45 43 54 20 75 75 69 64 20 46 52 4f 4d 20 62  LECT uuid FROM b
1fe0: 6c 6f 62 20 57 48 45 52 45 20 72 69 64 3d 25 64  lob WHERE rid=%d
1ff0: 20 41 4e 44 20 73 69 7a 65 3e 3d 30 22 2c 20 72   AND size>=0", r
2000: 69 64 29 3b 0a 20 20 69 66 28 20 62 6c 6f 62 5f  id);.  if( blob_
2010: 73 69 7a 65 28 26 75 75 69 64 29 3d 3d 30 20 29  size(&uuid)==0 )
2020: 7b 0a 20 20 20 20 72 65 74 75 72 6e 3b 0a 20 20  {.    return;.  
2030: 7d 0a 20 20 69 66 28 20 70 55 75 69 64 20 29 7b  }.  if( pUuid ){
2040: 0a 20 20 20 20 69 66 28 20 62 6c 6f 62 5f 63 6f  .    if( blob_co
2050: 6d 70 61 72 65 28 70 55 75 69 64 2c 20 26 75 75  mpare(pUuid, &uu
2060: 69 64 29 21 3d 30 20 29 7b 0a 20 20 20 20 20 20  id)!=0 ){.      
2070: 62 6c 6f 62 5f 72 65 73 65 74 28 26 75 75 69 64  blob_reset(&uuid
2080: 29 3b 0a 20 20 20 20 20 20 72 65 74 75 72 6e 3b  );.      return;
2090: 0a 20 20 20 20 7d 0a 20 20 7d 65 6c 73 65 7b 0a  .    }.  }else{.
20a0: 20 20 20 20 70 55 75 69 64 20 3d 20 26 75 75 69      pUuid = &uui
20b0: 64 3b 0a 20 20 7d 0a 20 20 69 66 28 20 75 75 69  d;.  }.  if( uui
20c0: 64 5f 69 73 5f 73 68 75 6e 6e 65 64 28 62 6c 6f  d_is_shunned(blo
20d0: 62 5f 73 74 72 28 70 55 75 69 64 29 29 20 29 7b  b_str(pUuid)) ){
20e0: 0a 20 20 20 20 62 6c 6f 62 5f 72 65 73 65 74 28  .    blob_reset(
20f0: 26 75 75 69 64 29 3b 0a 20 20 20 20 72 65 74 75  &uuid);.    retu
2100: 72 6e 3b 0a 20 20 7d 0a 20 20 69 66 28 20 70 58  rn;.  }.  if( pX
2110: 66 65 72 2d 3e 6d 78 53 65 6e 64 3c 3d 62 6c 6f  fer->mxSend<=blo
2120: 62 5f 73 69 7a 65 28 70 58 66 65 72 2d 3e 70 4f  b_size(pXfer->pO
2130: 75 74 29 20 29 7b 0a 20 20 20 20 62 6c 6f 62 5f  ut) ){.    blob_
2140: 61 70 70 65 6e 64 66 28 70 58 66 65 72 2d 3e 70  appendf(pXfer->p
2150: 4f 75 74 2c 20 22 69 67 6f 74 20 25 62 5c 6e 22  Out, "igot %b\n"
2160: 2c 20 70 55 75 69 64 29 3b 0a 20 20 20 20 70 58  , pUuid);.    pX
2170: 66 65 72 2d 3e 6e 49 47 6f 74 53 65 6e 74 2b 2b  fer->nIGotSent++
2180: 3b 0a 20 20 20 20 62 6c 6f 62 5f 72 65 73 65 74  ;.    blob_reset
2190: 28 26 75 75 69 64 29 3b 0a 20 20 20 20 72 65 74  (&uuid);.    ret
21a0: 75 72 6e 3b 0a 20 20 7d 0a 20 20 69 66 28 20 6e  urn;.  }.  if( n
21b0: 61 74 69 76 65 44 65 6c 74 61 20 29 7b 0a 20 20  ativeDelta ){.  
21c0: 20 20 73 69 7a 65 20 3d 20 73 65 6e 64 5f 64 65    size = send_de
21d0: 6c 74 61 5f 6e 61 74 69 76 65 28 70 58 66 65 72  lta_native(pXfer
21e0: 2c 20 72 69 64 2c 20 70 55 75 69 64 29 3b 0a 20  , rid, pUuid);. 
21f0: 20 20 20 69 66 28 20 73 69 7a 65 20 29 7b 0a 20     if( size ){. 
2200: 20 20 20 20 20 70 58 66 65 72 2d 3e 6e 44 65 6c       pXfer->nDel
2210: 74 61 53 65 6e 74 2b 2b 3b 0a 20 20 20 20 7d 0a  taSent++;.    }.
2220: 20 20 7d 0a 20 20 69 66 28 20 73 69 7a 65 3d 3d    }.  if( size==
2230: 30 20 29 7b 0a 20 20 20 20 63 6f 6e 74 65 6e 74  0 ){.    content
2240: 5f 67 65 74 28 72 69 64 2c 20 26 63 6f 6e 74 65  _get(rid, &conte
2250: 6e 74 29 3b 0a 0a 20 20 20 20 69 66 28 20 21 6e  nt);..    if( !n
2260: 61 74 69 76 65 44 65 6c 74 61 20 26 26 20 62 6c  ativeDelta && bl
2270: 6f 62 5f 73 69 7a 65 28 26 63 6f 6e 74 65 6e 74  ob_size(&content
2280: 29 3e 31 30 30 20 29 7b 0a 20 20 20 20 20 20 73  )>100 ){.      s
2290: 69 7a 65 20 3d 20 73 65 6e 64 5f 64 65 6c 74 61  ize = send_delta
22a0: 5f 70 61 72 65 6e 74 28 70 58 66 65 72 2c 20 72  _parent(pXfer, r
22b0: 69 64 2c 20 26 63 6f 6e 74 65 6e 74 2c 20 70 55  id, &content, pU
22c0: 75 69 64 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20  uid);.    }.    
22d0: 69 66 28 20 73 69 7a 65 3d 3d 30 20 29 7b 0a 20  if( size==0 ){. 
22e0: 20 20 20 20 20 69 6e 74 20 73 69 7a 65 20 3d 20       int size = 
22f0: 62 6c 6f 62 5f 73 69 7a 65 28 26 63 6f 6e 74 65  blob_size(&conte
2300: 6e 74 29 3b 0a 20 20 20 20 20 20 62 6c 6f 62 5f  nt);.      blob_
2310: 61 70 70 65 6e 64 66 28 70 58 66 65 72 2d 3e 70  appendf(pXfer->p
2320: 4f 75 74 2c 20 22 66 69 6c 65 20 25 62 20 25 64  Out, "file %b %d
2330: 5c 6e 22 2c 20 70 55 75 69 64 2c 20 73 69 7a 65  \n", pUuid, size
2340: 29 3b 0a 20 20 20 20 20 20 62 6c 6f 62 5f 61 70  );.      blob_ap
2350: 70 65 6e 64 28 70 58 66 65 72 2d 3e 70 4f 75 74  pend(pXfer->pOut
2360: 2c 20 62 6c 6f 62 5f 62 75 66 66 65 72 28 26 63  , blob_buffer(&c
2370: 6f 6e 74 65 6e 74 29 2c 20 73 69 7a 65 29 3b 0a  ontent), size);.
2380: 20 20 20 20 20 20 70 58 66 65 72 2d 3e 6e 46 69        pXfer->nFi
2390: 6c 65 53 65 6e 74 2b 2b 3b 0a 20 20 20 20 7d 65  leSent++;.    }e
23a0: 6c 73 65 7b 0a 20 20 20 20 20 20 70 58 66 65 72  lse{.      pXfer
23b0: 2d 3e 6e 44 65 6c 74 61 53 65 6e 74 2b 2b 3b 0a  ->nDeltaSent++;.
23c0: 20 20 20 20 7d 0a 20 20 7d 0a 20 20 72 65 6d 6f      }.  }.  remo
23d0: 74 65 5f 68 61 73 28 72 69 64 29 3b 0a 20 20 62  te_has(rid);.  b
23e0: 6c 6f 62 5f 72 65 73 65 74 28 26 75 75 69 64 29  lob_reset(&uuid)
23f0: 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 65 6e 64 20  ;.}../*.** Send 
2400: 61 20 67 69 6d 6d 65 20 6d 65 73 73 61 67 65 20  a gimme message 
2410: 66 6f 72 20 65 76 65 72 79 20 70 68 61 6e 74 6f  for every phanto
2420: 6d 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69  m..*/.static voi
2430: 64 20 72 65 71 75 65 73 74 5f 70 68 61 6e 74 6f  d request_phanto
2440: 6d 73 28 58 66 65 72 20 2a 70 58 66 65 72 2c 20  ms(Xfer *pXfer, 
2450: 69 6e 74 20 6d 61 78 52 65 71 29 7b 0a 20 20 53  int maxReq){.  S
2460: 74 6d 74 20 71 3b 0a 20 20 64 62 5f 70 72 65 70  tmt q;.  db_prep
2470: 61 72 65 28 26 71 2c 20 0a 20 20 20 20 22 53 45  are(&q, .    "SE
2480: 4c 45 43 54 20 75 75 69 64 20 46 52 4f 4d 20 70  LECT uuid FROM p
2490: 68 61 6e 74 6f 6d 20 4a 4f 49 4e 20 62 6c 6f 62  hantom JOIN blob
24a0: 20 55 53 49 4e 47 28 72 69 64 29 22 0a 20 20 20   USING(rid)".   
24b0: 20 22 20 57 48 45 52 45 20 4e 4f 54 20 45 58 49   " WHERE NOT EXI
24c0: 53 54 53 28 53 45 4c 45 43 54 20 31 20 46 52 4f  STS(SELECT 1 FRO
24d0: 4d 20 73 68 75 6e 20 57 48 45 52 45 20 75 75 69  M shun WHERE uui
24e0: 64 3d 62 6c 6f 62 2e 75 75 69 64 29 22 0a 20 20  d=blob.uuid)".  
24f0: 29 3b 0a 20 20 77 68 69 6c 65 28 20 64 62 5f 73  );.  while( db_s
2500: 74 65 70 28 26 71 29 3d 3d 53 51 4c 49 54 45 5f  tep(&q)==SQLITE_
2510: 52 4f 57 20 26 26 20 6d 61 78 52 65 71 2d 2d 20  ROW && maxReq-- 
2520: 3e 20 30 20 29 7b 0a 20 20 20 20 63 6f 6e 73 74  > 0 ){.    const
2530: 20 63 68 61 72 20 2a 7a 55 75 69 64 20 3d 20 64   char *zUuid = d
2540: 62 5f 63 6f 6c 75 6d 6e 5f 74 65 78 74 28 26 71  b_column_text(&q
2550: 2c 20 30 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 61  , 0);.    blob_a
2560: 70 70 65 6e 64 66 28 70 58 66 65 72 2d 3e 70 4f  ppendf(pXfer->pO
2570: 75 74 2c 20 22 67 69 6d 6d 65 20 25 73 5c 6e 22  ut, "gimme %s\n"
2580: 2c 20 7a 55 75 69 64 29 3b 0a 20 20 20 20 70 58  , zUuid);.    pX
2590: 66 65 72 2d 3e 6e 47 69 6d 6d 65 53 65 6e 74 2b  fer->nGimmeSent+
25a0: 2b 3b 0a 20 20 7d 0a 20 20 64 62 5f 66 69 6e 61  +;.  }.  db_fina
25b0: 6c 69 7a 65 28 26 71 29 3b 0a 7d 0a 0a 2f 2a 0a  lize(&q);.}../*.
25c0: 2a 2a 20 43 6f 6d 70 75 74 65 20 61 6e 20 53 48  ** Compute an SH
25d0: 41 31 20 68 61 73 68 20 6f 6e 20 74 68 65 20 74  A1 hash on the t
25e0: 61 69 6c 20 6f 66 20 70 4d 73 67 2e 20 20 56 65  ail of pMsg.  Ve
25f0: 72 69 66 79 20 74 68 61 74 20 69 74 20 6d 61 74  rify that it mat
2600: 63 68 65 73 20 74 68 65 0a 2a 2a 20 74 68 65 20  ches the.** the 
2610: 68 61 73 68 20 67 69 76 65 6e 20 69 6e 20 70 48  hash given in pH
2620: 61 73 68 2e 20 20 52 65 74 75 72 6e 20 31 20 6f  ash.  Return 1 o
2630: 6e 20 61 20 73 75 63 63 65 73 73 66 75 6c 20 6d  n a successful m
2640: 61 74 63 68 2e 20 20 52 65 74 75 72 6e 20 30 0a  atch.  Return 0.
2650: 2a 2a 20 69 66 20 74 68 65 72 65 20 69 73 20 61  ** if there is a
2660: 20 6d 69 73 6d 61 74 63 68 2e 0a 2a 2f 0a 73 74   mismatch..*/.st
2670: 61 74 69 63 20 69 6e 74 20 63 68 65 63 6b 5f 74  atic int check_t
2680: 61 69 6c 5f 68 61 73 68 28 42 6c 6f 62 20 2a 70  ail_hash(Blob *p
2690: 48 61 73 68 2c 20 42 6c 6f 62 20 2a 70 4d 73 67  Hash, Blob *pMsg
26a0: 29 7b 0a 20 20 42 6c 6f 62 20 74 61 69 6c 3b 0a  ){.  Blob tail;.
26b0: 20 20 42 6c 6f 62 20 68 32 3b 0a 20 20 69 6e 74    Blob h2;.  int
26c0: 20 72 63 3b 0a 20 20 62 6c 6f 62 5f 74 61 69 6c   rc;.  blob_tail
26d0: 28 70 4d 73 67 2c 20 26 74 61 69 6c 29 3b 0a 20  (pMsg, &tail);. 
26e0: 20 73 68 61 31 73 75 6d 5f 62 6c 6f 62 28 26 74   sha1sum_blob(&t
26f0: 61 69 6c 2c 20 26 68 32 29 3b 0a 20 20 72 63 20  ail, &h2);.  rc 
2700: 3d 20 62 6c 6f 62 5f 63 6f 6d 70 61 72 65 28 70  = blob_compare(p
2710: 48 61 73 68 2c 20 26 68 32 29 3b 0a 20 20 62 6c  Hash, &h2);.  bl
2720: 6f 62 5f 72 65 73 65 74 28 26 68 32 29 3b 0a 20  ob_reset(&h2);. 
2730: 20 62 6c 6f 62 5f 72 65 73 65 74 28 26 74 61 69   blob_reset(&tai
2740: 6c 29 3b 0a 20 20 72 65 74 75 72 6e 20 72 63 3d  l);.  return rc=
2750: 3d 30 3b 0a 7d 0a 0a 0a 2f 2a 0a 2a 2a 20 43 68  =0;.}.../*.** Ch
2760: 65 63 6b 20 74 68 65 20 73 69 67 6e 61 74 75 72  eck the signatur
2770: 65 20 6f 6e 20 61 6e 20 61 70 70 6c 69 63 61 74  e on an applicat
2780: 69 6f 6e 2f 78 2d 66 6f 73 73 69 6c 20 70 61 79  ion/x-fossil pay
2790: 6c 6f 61 64 20 72 65 63 65 69 76 65 64 20 62 79  load received by
27a0: 0a 2a 2a 20 74 68 65 20 48 54 54 50 20 73 65 72  .** the HTTP ser
27b0: 76 65 72 2e 20 20 54 68 65 20 73 69 67 6e 61 74  ver.  The signat
27c0: 75 72 65 20 69 73 20 61 20 6c 69 6e 65 20 6f 66  ure is a line of
27d0: 20 74 68 65 20 66 6f 6c 6c 6f 77 69 6e 67 20 66   the following f
27e0: 6f 72 6d 3a 0a 2a 2a 0a 2a 2a 20 20 20 20 20 20  orm:.**.**      
27f0: 20 20 6c 6f 67 69 6e 20 4c 4f 47 49 4e 20 4e 4f    login LOGIN NO
2800: 4e 43 45 20 53 49 47 4e 41 54 55 52 45 0a 2a 2a  NCE SIGNATURE.**
2810: 0a 2a 2a 20 54 68 65 20 4e 4f 4e 43 45 20 69 73  .** The NONCE is
2820: 20 74 68 65 20 53 48 41 31 20 68 61 73 68 20 6f   the SHA1 hash o
2830: 66 20 74 68 65 20 72 65 6d 61 69 6e 64 65 72 20  f the remainder 
2840: 6f 66 20 74 68 65 20 69 6e 70 75 74 2e 20 20 0a  of the input.  .
2850: 2a 2a 20 53 49 47 4e 41 54 55 52 45 20 69 73 20  ** SIGNATURE is 
2860: 74 68 65 20 53 48 41 31 20 63 68 65 63 6b 73 75  the SHA1 checksu
2870: 6d 20 6f 66 20 74 68 65 20 4e 4f 4e 43 45 20 63  m of the NONCE c
2880: 6f 6e 63 61 74 65 6e 61 74 65 64 20 0a 2a 2a 20  oncatenated .** 
2890: 77 69 74 68 20 74 68 65 20 75 73 65 72 73 20 70  with the users p
28a0: 61 73 73 77 6f 72 64 2e 0a 2a 2a 0a 2a 2a 20 54  assword..**.** T
28b0: 68 65 20 70 61 72 61 6d 65 74 65 72 73 20 74 6f  he parameters to
28c0: 20 74 68 69 73 20 72 6f 75 74 69 6e 65 20 61 72   this routine ar
28d0: 65 20 65 70 68 65 72 6d 65 72 61 6c 20 62 6c 6f  e ephermeral blo
28e0: 62 73 20 68 6f 6c 64 69 6e 67 20 74 68 65 0a 2a  bs holding the.*
28f0: 2a 20 4c 4f 47 49 4e 2c 20 4e 4f 4e 43 45 20 61  * LOGIN, NONCE a
2900: 6e 64 20 53 49 47 4e 41 54 55 52 45 2e 0a 2a 2a  nd SIGNATURE..**
2910: 0a 2a 2a 20 54 68 69 73 20 72 6f 75 74 69 6e 65  .** This routine
2920: 20 61 74 74 65 6d 70 74 73 20 74 6f 20 6c 6f 63   attempts to loc
2930: 61 74 65 20 74 68 65 20 75 73 65 72 20 61 6e 64  ate the user and
2940: 20 76 65 72 69 66 79 20 74 68 65 20 73 69 67 6e   verify the sign
2950: 61 74 75 72 65 2e 0a 2a 2a 20 49 66 20 65 76 65  ature..** If eve
2960: 72 79 74 68 69 6e 67 20 63 68 65 63 6b 73 20 6f  rything checks o
2970: 75 74 2c 20 74 68 65 20 55 53 45 52 2e 43 41 50  ut, the USER.CAP
2980: 20 63 6f 6c 75 6d 6e 20 66 6f 72 20 74 68 65 20   column for the 
2990: 55 53 45 52 20 74 61 62 6c 65 0a 2a 2a 20 69 73  USER table.** is
29a0: 20 63 6f 6e 73 75 6c 74 65 64 20 74 6f 20 73 65   consulted to se
29b0: 74 20 70 72 69 76 69 6c 65 67 65 73 20 69 6e 20  t privileges in 
29c0: 74 68 65 20 67 6c 6f 62 61 6c 20 67 20 76 61 72  the global g var
29d0: 69 61 62 6c 65 2e 0a 2a 2a 0a 2a 2a 20 49 66 20  iable..**.** If 
29e0: 61 6e 79 74 68 69 6e 67 20 66 61 69 6c 73 20 74  anything fails t
29f0: 6f 20 63 68 65 63 6b 20 6f 75 74 2c 20 6e 6f 20  o check out, no 
2a00: 63 68 61 6e 67 65 73 20 61 72 65 20 6d 61 64 65  changes are made
2a10: 20 74 6f 20 70 72 69 76 69 6c 65 67 65 73 2e 0a   to privileges..
2a20: 2a 2a 0a 2a 2a 20 53 69 67 6e 61 74 75 72 65 20  **.** Signature 
2a30: 67 65 6e 65 72 61 74 69 6f 6e 20 6f 6e 20 74 68  generation on th
2a40: 65 20 63 6c 69 65 6e 74 20 73 69 64 65 20 69 73  e client side is
2a50: 20 68 61 6e 64 6c 65 64 20 62 79 20 74 68 65 20   handled by the 
2a60: 0a 2a 2a 20 68 74 74 70 5f 65 78 63 68 61 6e 67  .** http_exchang
2a70: 65 28 29 20 72 6f 75 74 69 6e 65 2e 0a 2a 2f 0a  e() routine..*/.
2a80: 76 6f 69 64 20 63 68 65 63 6b 5f 6c 6f 67 69 6e  void check_login
2a90: 28 42 6c 6f 62 20 2a 70 4c 6f 67 69 6e 2c 20 42  (Blob *pLogin, B
2aa0: 6c 6f 62 20 2a 70 4e 6f 6e 63 65 2c 20 42 6c 6f  lob *pNonce, Blo
2ab0: 62 20 2a 70 53 69 67 29 7b 0a 20 20 53 74 6d 74  b *pSig){.  Stmt
2ac0: 20 71 3b 0a 20 20 69 6e 74 20 72 63 3b 0a 0a 20   q;.  int rc;.. 
2ad0: 20 64 62 5f 70 72 65 70 61 72 65 28 26 71 2c 20   db_prepare(&q, 
2ae0: 22 53 45 4c 45 43 54 20 70 77 2c 20 63 61 70 2c  "SELECT pw, cap,
2af0: 20 75 69 64 20 46 52 4f 4d 20 75 73 65 72 20 57   uid FROM user W
2b00: 48 45 52 45 20 6c 6f 67 69 6e 3d 25 42 22 2c 20  HERE login=%B", 
2b10: 70 4c 6f 67 69 6e 29 3b 0a 20 20 69 66 28 20 64  pLogin);.  if( d
2b20: 62 5f 73 74 65 70 28 26 71 29 3d 3d 53 51 4c 49  b_step(&q)==SQLI
2b30: 54 45 5f 52 4f 57 20 29 7b 0a 20 20 20 20 42 6c  TE_ROW ){.    Bl
2b40: 6f 62 20 70 77 2c 20 63 6f 6d 62 69 6e 65 64 2c  ob pw, combined,
2b50: 20 68 61 73 68 3b 0a 20 20 20 20 62 6c 6f 62 5f   hash;.    blob_
2b60: 7a 65 72 6f 28 26 70 77 29 3b 0a 20 20 20 20 64  zero(&pw);.    d
2b70: 62 5f 65 70 68 65 6d 65 72 61 6c 5f 62 6c 6f 62  b_ephemeral_blob
2b80: 28 26 71 2c 20 30 2c 20 26 70 77 29 3b 0a 20 20  (&q, 0, &pw);.  
2b90: 20 20 62 6c 6f 62 5f 7a 65 72 6f 28 26 63 6f 6d    blob_zero(&com
2ba0: 62 69 6e 65 64 29 3b 0a 20 20 20 20 62 6c 6f 62  bined);.    blob
2bb0: 5f 63 6f 70 79 28 26 63 6f 6d 62 69 6e 65 64 2c  _copy(&combined,
2bc0: 20 70 4e 6f 6e 63 65 29 3b 0a 20 20 20 20 62 6c   pNonce);.    bl
2bd0: 6f 62 5f 61 70 70 65 6e 64 28 26 63 6f 6d 62 69  ob_append(&combi
2be0: 6e 65 64 2c 20 62 6c 6f 62 5f 62 75 66 66 65 72  ned, blob_buffer
2bf0: 28 26 70 77 29 2c 20 62 6c 6f 62 5f 73 69 7a 65  (&pw), blob_size
2c00: 28 26 70 77 29 29 3b 0a 20 20 20 20 2f 2a 20 43  (&pw));.    /* C
2c10: 47 49 44 45 42 55 47 28 28 22 70 72 65 73 69 67  GIDEBUG(("presig
2c20: 3d 5b 25 73 5d 5c 6e 22 2c 20 62 6c 6f 62 5f 73  =[%s]\n", blob_s
2c30: 74 72 28 26 63 6f 6d 62 69 6e 65 64 29 29 29 3b  tr(&combined)));
2c40: 20 2a 2f 0a 20 20 20 20 73 68 61 31 73 75 6d 5f   */.    sha1sum_
2c50: 62 6c 6f 62 28 26 63 6f 6d 62 69 6e 65 64 2c 20  blob(&combined, 
2c60: 26 68 61 73 68 29 3b 0a 20 20 20 20 72 63 20 3d  &hash);.    rc =
2c70: 20 62 6c 6f 62 5f 63 6f 6d 70 61 72 65 28 26 68   blob_compare(&h
2c80: 61 73 68 2c 20 70 53 69 67 29 3b 0a 20 20 20 20  ash, pSig);.    
2c90: 62 6c 6f 62 5f 72 65 73 65 74 28 26 68 61 73 68  blob_reset(&hash
2ca0: 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 72 65 73 65  );.    blob_rese
2cb0: 74 28 26 63 6f 6d 62 69 6e 65 64 29 3b 0a 20 20  t(&combined);.  
2cc0: 20 20 69 66 28 20 72 63 3d 3d 30 20 29 7b 0a 20    if( rc==0 ){. 
2cd0: 20 20 20 20 20 63 6f 6e 73 74 20 63 68 61 72 20       const char 
2ce0: 2a 7a 43 61 70 3b 0a 20 20 20 20 20 20 7a 43 61  *zCap;.      zCa
2cf0: 70 20 3d 20 64 62 5f 63 6f 6c 75 6d 6e 5f 74 65  p = db_column_te
2d00: 78 74 28 26 71 2c 20 31 29 3b 0a 20 20 20 20 20  xt(&q, 1);.     
2d10: 20 6c 6f 67 69 6e 5f 73 65 74 5f 63 61 70 61 62   login_set_capab
2d20: 69 6c 69 74 69 65 73 28 7a 43 61 70 29 3b 0a 20  ilities(zCap);. 
2d30: 20 20 20 20 20 67 2e 75 73 65 72 55 69 64 20 3d       g.userUid =
2d40: 20 64 62 5f 63 6f 6c 75 6d 6e 5f 69 6e 74 28 26   db_column_int(&
2d50: 71 2c 20 32 29 3b 0a 20 20 20 20 20 20 67 2e 7a  q, 2);.      g.z
2d60: 4c 6f 67 69 6e 20 3d 20 6d 70 72 69 6e 74 66 28  Login = mprintf(
2d70: 22 25 62 22 2c 20 70 4c 6f 67 69 6e 29 3b 0a 20  "%b", pLogin);. 
2d80: 20 20 20 20 20 67 2e 7a 4e 6f 6e 63 65 20 3d 20       g.zNonce = 
2d90: 6d 70 72 69 6e 74 66 28 22 25 62 22 2c 20 70 4e  mprintf("%b", pN
2da0: 6f 6e 63 65 29 3b 0a 20 20 20 20 7d 0a 20 20 7d  once);.    }.  }
2db0: 0a 20 20 64 62 5f 72 65 73 65 74 28 26 71 29 3b  .  db_reset(&q);
2dc0: 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 65 6e 64 20 74  .}../*.** Send t
2dd0: 68 65 20 63 6f 6e 74 65 6e 74 20 6f 66 20 61 6c  he content of al
2de0: 6c 20 66 69 6c 65 73 20 69 6e 20 74 68 65 20 75  l files in the u
2df0: 6e 73 65 6e 74 20 74 61 62 6c 65 2e 0a 2a 2a 0a  nsent table..**.
2e00: 2a 2a 20 54 68 69 73 20 69 73 20 72 65 61 6c 6c  ** This is reall
2e10: 79 20 6a 75 73 74 20 61 6e 20 6f 70 74 69 6d 69  y just an optimi
2e20: 7a 61 74 69 6f 6e 2e 20 20 49 66 20 79 6f 75 20  zation.  If you 
2e30: 63 6c 65 61 72 20 74 68 65 0a 2a 2a 20 75 6e 73  clear the.** uns
2e40: 65 6e 74 20 74 61 62 6c 65 2c 20 61 6c 6c 20 74  ent table, all t
2e50: 68 65 20 72 69 67 68 74 20 66 69 6c 65 73 20 77  he right files w
2e60: 69 6c 6c 20 73 74 69 6c 6c 20 67 65 74 20 74 72  ill still get tr
2e70: 61 6e 73 66 65 72 72 65 64 2e 0a 2a 2a 20 49 74  ansferred..** It
2e80: 20 6a 75 73 74 20 6d 69 67 68 74 20 72 65 71 75   just might requ
2e90: 69 72 65 20 61 6e 20 65 78 74 72 61 20 72 6f 75  ire an extra rou
2ea0: 6e 64 20 74 72 69 70 20 6f 72 20 74 77 6f 2e 0a  nd trip or two..
2eb0: 2a 2f 0a 73 74 61 74 69 63 20 76 6f 69 64 20 73  */.static void s
2ec0: 65 6e 64 5f 75 6e 73 65 6e 74 28 58 66 65 72 20  end_unsent(Xfer 
2ed0: 2a 70 58 66 65 72 29 7b 0a 20 20 53 74 6d 74 20  *pXfer){.  Stmt 
2ee0: 71 3b 0a 20 20 64 62 5f 70 72 65 70 61 72 65 28  q;.  db_prepare(
2ef0: 26 71 2c 20 22 53 45 4c 45 43 54 20 72 69 64 20  &q, "SELECT rid 
2f00: 46 52 4f 4d 20 75 6e 73 65 6e 74 22 29 3b 0a 20  FROM unsent");. 
2f10: 20 77 68 69 6c 65 28 20 64 62 5f 73 74 65 70 28   while( db_step(
2f20: 26 71 29 3d 3d 53 51 4c 49 54 45 5f 52 4f 57 20  &q)==SQLITE_ROW 
2f30: 29 7b 0a 20 20 20 20 69 6e 74 20 72 69 64 20 3d  ){.    int rid =
2f40: 20 64 62 5f 63 6f 6c 75 6d 6e 5f 69 6e 74 28 26   db_column_int(&
2f50: 71 2c 20 30 29 3b 0a 20 20 20 20 73 65 6e 64 5f  q, 0);.    send_
2f60: 66 69 6c 65 28 70 58 66 65 72 2c 20 72 69 64 2c  file(pXfer, rid,
2f70: 20 30 2c 20 30 29 3b 0a 20 20 7d 0a 20 20 64 62   0, 0);.  }.  db
2f80: 5f 66 69 6e 61 6c 69 7a 65 28 26 71 29 3b 0a 20  _finalize(&q);. 
2f90: 20 64 62 5f 6d 75 6c 74 69 5f 65 78 65 63 28 22   db_multi_exec("
2fa0: 44 45 4c 45 54 45 20 46 52 4f 4d 20 75 6e 73 65  DELETE FROM unse
2fb0: 6e 74 22 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 43  nt");.}../*.** C
2fc0: 68 65 63 6b 20 74 6f 20 73 65 65 20 69 66 20 74  heck to see if t
2fd0: 68 65 20 6e 75 6d 62 65 72 20 6f 66 20 75 6e 63  he number of unc
2fe0: 6c 75 73 74 65 72 65 64 20 65 6e 74 72 69 65 73  lustered entries
2ff0: 20 69 73 20 67 72 65 61 74 65 72 20 74 68 61 6e   is greater than
3000: 0a 2a 2a 20 31 30 30 20 61 6e 64 20 69 66 20 69  .** 100 and if i
3010: 74 20 69 73 2c 20 66 6f 72 6d 20 61 20 6e 65 77  t is, form a new
3020: 20 63 6c 75 73 74 65 72 2e 20 20 55 6e 63 6c 75   cluster.  Unclu
3030: 73 74 65 72 65 64 20 70 68 61 6e 74 6f 6d 73 20  stered phantoms 
3040: 64 6f 20 6e 6f 74 0a 2a 2a 20 63 6f 75 6e 74 20  do not.** count 
3050: 74 6f 77 61 72 64 20 74 68 65 20 31 30 30 20 74  toward the 100 t
3060: 6f 74 61 6c 2e 20 20 41 6e 64 20 70 68 61 6e 74  otal.  And phant
3070: 6f 6d 73 20 61 72 65 20 6e 65 76 65 72 20 61 64  oms are never ad
3080: 64 65 64 20 74 6f 20 61 20 6e 65 77 0a 2a 2a 20  ded to a new.** 
3090: 63 6c 75 73 74 65 72 2e 0a 2a 2f 0a 73 74 61 74  cluster..*/.stat
30a0: 69 63 20 76 6f 69 64 20 63 72 65 61 74 65 5f 63  ic void create_c
30b0: 6c 75 73 74 65 72 28 76 6f 69 64 29 7b 0a 20 20  luster(void){.  
30c0: 42 6c 6f 62 20 63 6c 75 73 74 65 72 2c 20 63 6b  Blob cluster, ck
30d0: 73 75 6d 3b 0a 20 20 53 74 6d 74 20 71 3b 0a 20  sum;.  Stmt q;. 
30e0: 20 69 6e 74 20 6e 55 6e 63 6c 3b 0a 20 20 6e 55   int nUncl;.  nU
30f0: 6e 63 6c 20 3d 20 64 62 5f 69 6e 74 28 30 2c 20  ncl = db_int(0, 
3100: 22 53 45 4c 45 43 54 20 63 6f 75 6e 74 28 2a 29  "SELECT count(*)
3110: 20 46 52 4f 4d 20 75 6e 63 6c 75 73 74 65 72 65   FROM unclustere
3120: 64 22 0a 20 20 20 20 20 20 20 20 20 20 20 20 20  d".             
3130: 20 20 20 20 20 20 20 22 20 57 48 45 52 45 20 4e         " WHERE N
3140: 4f 54 20 45 58 49 53 54 53 28 53 45 4c 45 43 54  OT EXISTS(SELECT
3150: 20 31 20 46 52 4f 4d 20 70 68 61 6e 74 6f 6d 22   1 FROM phantom"
3160: 0a 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  .               
3170: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
3180: 20 20 20 20 20 20 20 22 20 57 48 45 52 45 20 72         " WHERE r
3190: 69 64 3d 75 6e 63 6c 75 73 74 65 72 65 64 2e 72  id=unclustered.r
31a0: 69 64 29 22 29 3b 0a 20 20 69 66 28 20 6e 55 6e  id)");.  if( nUn
31b0: 63 6c 3c 31 30 30 20 29 7b 0a 20 20 20 20 72 65  cl<100 ){.    re
31c0: 74 75 72 6e 3b 0a 20 20 7d 0a 20 20 62 6c 6f 62  turn;.  }.  blob
31d0: 5f 7a 65 72 6f 28 26 63 6c 75 73 74 65 72 29 3b  _zero(&cluster);
31e0: 0a 20 20 64 62 5f 70 72 65 70 61 72 65 28 26 71  .  db_prepare(&q
31f0: 2c 20 22 53 45 4c 45 43 54 20 75 75 69 64 20 46  , "SELECT uuid F
3200: 52 4f 4d 20 75 6e 63 6c 75 73 74 65 72 65 64 2c  ROM unclustered,
3210: 20 62 6c 6f 62 22 0a 20 20 20 20 20 20 20 20 20   blob".         
3220: 20 20 20 20 20 20 20 20 22 20 57 48 45 52 45 20          " WHERE 
3230: 4e 4f 54 20 45 58 49 53 54 53 28 53 45 4c 45 43  NOT EXISTS(SELEC
3240: 54 20 31 20 46 52 4f 4d 20 70 68 61 6e 74 6f 6d  T 1 FROM phantom
3250: 22 0a 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ".              
3260: 20 20 20 22 20 20 20 20 20 20 20 20 20 20 20 20     "            
3270: 20 20 20 20 20 20 20 57 48 45 52 45 20 72 69 64         WHERE rid
3280: 21 3d 75 6e 63 6c 75 73 74 65 72 65 64 2e 72 69  !=unclustered.ri
3290: 64 29 22 0a 20 20 20 20 20 20 20 20 20 20 20 20  d)".            
32a0: 20 20 20 20 20 22 20 20 20 41 4e 44 20 75 6e 63       "   AND unc
32b0: 6c 75 73 74 65 72 65 64 2e 72 69 64 3d 62 6c 6f  lustered.rid=blo
32c0: 62 2e 72 69 64 22 0a 20 20 20 20 20 20 20 20 20  b.rid".         
32d0: 20 20 20 20 20 20 20 20 22 20 20 20 41 4e 44 20          "   AND 
32e0: 4e 4f 54 20 45 58 49 53 54 53 28 53 45 4c 45 43  NOT EXISTS(SELEC
32f0: 54 20 31 20 46 52 4f 4d 20 73 68 75 6e 20 57 48  T 1 FROM shun WH
3300: 45 52 45 20 75 75 69 64 3d 62 6c 6f 62 2e 75 75  ERE uuid=blob.uu
3310: 69 64 29 22 0a 20 20 20 20 20 20 20 20 20 20 20  id)".           
3320: 20 20 20 20 20 20 22 20 4f 52 44 45 52 20 42 59        " ORDER BY
3330: 20 31 22 29 3b 0a 20 20 77 68 69 6c 65 28 20 64   1");.  while( d
3340: 62 5f 73 74 65 70 28 26 71 29 3d 3d 53 51 4c 49  b_step(&q)==SQLI
3350: 54 45 5f 52 4f 57 20 29 7b 0a 20 20 20 20 62 6c  TE_ROW ){.    bl
3360: 6f 62 5f 61 70 70 65 6e 64 66 28 26 63 6c 75 73  ob_appendf(&clus
3370: 74 65 72 2c 20 22 4d 20 25 73 5c 6e 22 2c 20 64  ter, "M %s\n", d
3380: 62 5f 63 6f 6c 75 6d 6e 5f 74 65 78 74 28 26 71  b_column_text(&q
3390: 2c 20 30 29 29 3b 0a 20 20 7d 0a 20 20 64 62 5f  , 0));.  }.  db_
33a0: 66 69 6e 61 6c 69 7a 65 28 26 71 29 3b 0a 20 20  finalize(&q);.  
33b0: 6d 64 35 73 75 6d 5f 62 6c 6f 62 28 26 63 6c 75  md5sum_blob(&clu
33c0: 73 74 65 72 2c 20 26 63 6b 73 75 6d 29 3b 0a 20  ster, &cksum);. 
33d0: 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66 28 26 63   blob_appendf(&c
33e0: 6c 75 73 74 65 72 2c 20 22 5a 20 25 62 5c 6e 22  luster, "Z %b\n"
33f0: 2c 20 26 63 6b 73 75 6d 29 3b 0a 20 20 62 6c 6f  , &cksum);.  blo
3400: 62 5f 72 65 73 65 74 28 26 63 6b 73 75 6d 29 3b  b_reset(&cksum);
3410: 0a 20 20 64 62 5f 6d 75 6c 74 69 5f 65 78 65 63  .  db_multi_exec
3420: 28 22 44 45 4c 45 54 45 20 46 52 4f 4d 20 75 6e  ("DELETE FROM un
3430: 63 6c 75 73 74 65 72 65 64 22 29 3b 0a 20 20 63  clustered");.  c
3440: 6f 6e 74 65 6e 74 5f 70 75 74 28 26 63 6c 75 73  ontent_put(&clus
3450: 74 65 72 2c 20 30 2c 20 30 29 3b 0a 20 20 62 6c  ter, 0, 0);.  bl
3460: 6f 62 5f 72 65 73 65 74 28 26 63 6c 75 73 74 65  ob_reset(&cluste
3470: 72 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 53 65 6e  r);.}../*.** Sen
3480: 64 20 61 6e 20 69 67 6f 74 20 6d 65 73 73 61 67  d an igot messag
3490: 65 20 66 6f 72 20 65 76 65 72 79 20 65 6e 74 72  e for every entr
34a0: 79 20 69 6e 20 75 6e 63 6c 75 73 74 65 72 65 64  y in unclustered
34b0: 20 74 61 62 6c 65 2e 0a 2a 2a 20 52 65 74 75 72   table..** Retur
34c0: 6e 20 74 68 65 20 6e 75 6d 62 65 72 20 6f 66 20  n the number of 
34d0: 63 61 72 64 73 20 73 65 6e 74 2e 0a 2a 2f 0a 73  cards sent..*/.s
34e0: 74 61 74 69 63 20 69 6e 74 20 73 65 6e 64 5f 75  tatic int send_u
34f0: 6e 63 6c 75 73 74 65 72 65 64 28 58 66 65 72 20  nclustered(Xfer 
3500: 2a 70 58 66 65 72 29 7b 0a 20 20 53 74 6d 74 20  *pXfer){.  Stmt 
3510: 71 3b 0a 20 20 69 6e 74 20 63 6e 74 20 3d 20 30  q;.  int cnt = 0
3520: 3b 0a 20 20 64 62 5f 70 72 65 70 61 72 65 28 26  ;.  db_prepare(&
3530: 71 2c 20 0a 20 20 20 20 22 53 45 4c 45 43 54 20  q, .    "SELECT 
3540: 75 75 69 64 20 46 52 4f 4d 20 75 6e 63 6c 75 73  uuid FROM unclus
3550: 74 65 72 65 64 20 4a 4f 49 4e 20 62 6c 6f 62 20  tered JOIN blob 
3560: 55 53 49 4e 47 28 72 69 64 29 22 0a 20 20 20 20  USING(rid)".    
3570: 22 20 57 48 45 52 45 20 4e 4f 54 20 45 58 49 53  " WHERE NOT EXIS
3580: 54 53 28 53 45 4c 45 43 54 20 31 20 46 52 4f 4d  TS(SELECT 1 FROM
3590: 20 73 68 75 6e 20 57 48 45 52 45 20 75 75 69 64   shun WHERE uuid
35a0: 3d 62 6c 6f 62 2e 75 75 69 64 29 22 0a 20 20 29  =blob.uuid)".  )
35b0: 3b 0a 20 20 77 68 69 6c 65 28 20 64 62 5f 73 74  ;.  while( db_st
35c0: 65 70 28 26 71 29 3d 3d 53 51 4c 49 54 45 5f 52  ep(&q)==SQLITE_R
35d0: 4f 57 20 29 7b 0a 20 20 20 20 62 6c 6f 62 5f 61  OW ){.    blob_a
35e0: 70 70 65 6e 64 66 28 70 58 66 65 72 2d 3e 70 4f  ppendf(pXfer->pO
35f0: 75 74 2c 20 22 69 67 6f 74 20 25 73 5c 6e 22 2c  ut, "igot %s\n",
3600: 20 64 62 5f 63 6f 6c 75 6d 6e 5f 74 65 78 74 28   db_column_text(
3610: 26 71 2c 20 30 29 29 3b 0a 20 20 20 20 63 6e 74  &q, 0));.    cnt
3620: 2b 2b 3b 0a 20 20 7d 0a 20 20 64 62 5f 66 69 6e  ++;.  }.  db_fin
3630: 61 6c 69 7a 65 28 26 71 29 3b 0a 20 20 72 65 74  alize(&q);.  ret
3640: 75 72 6e 20 63 6e 74 3b 0a 7d 0a 0a 2f 2a 0a 2a  urn cnt;.}../*.*
3650: 2a 20 53 65 6e 64 20 61 6e 20 69 67 6f 74 20 6d  * Send an igot m
3660: 65 73 73 61 67 65 20 66 6f 72 20 65 76 65 72 79  essage for every
3670: 20 61 72 74 69 66 61 63 74 2e 0a 2a 2f 0a 73 74   artifact..*/.st
3680: 61 74 69 63 20 76 6f 69 64 20 73 65 6e 64 5f 61  atic void send_a
3690: 6c 6c 28 58 66 65 72 20 2a 70 58 66 65 72 29 7b  ll(Xfer *pXfer){
36a0: 0a 20 20 53 74 6d 74 20 71 3b 0a 20 20 64 62 5f  .  Stmt q;.  db_
36b0: 70 72 65 70 61 72 65 28 26 71 2c 20 0a 20 20 20  prepare(&q, .   
36c0: 20 22 53 45 4c 45 43 54 20 75 75 69 64 20 46 52   "SELECT uuid FR
36d0: 4f 4d 20 62 6c 6f 62 20 22 0a 20 20 20 20 22 20  OM blob ".    " 
36e0: 57 48 45 52 45 20 4e 4f 54 20 45 58 49 53 54 53  WHERE NOT EXISTS
36f0: 28 53 45 4c 45 43 54 20 31 20 46 52 4f 4d 20 73  (SELECT 1 FROM s
3700: 68 75 6e 20 57 48 45 52 45 20 75 75 69 64 3d 62  hun WHERE uuid=b
3710: 6c 6f 62 2e 75 75 69 64 29 22 0a 20 20 29 3b 0a  lob.uuid)".  );.
3720: 20 20 77 68 69 6c 65 28 20 64 62 5f 73 74 65 70    while( db_step
3730: 28 26 71 29 3d 3d 53 51 4c 49 54 45 5f 52 4f 57  (&q)==SQLITE_ROW
3740: 20 29 7b 0a 20 20 20 20 62 6c 6f 62 5f 61 70 70   ){.    blob_app
3750: 65 6e 64 66 28 70 58 66 65 72 2d 3e 70 4f 75 74  endf(pXfer->pOut
3760: 2c 20 22 69 67 6f 74 20 25 73 5c 6e 22 2c 20 64  , "igot %s\n", d
3770: 62 5f 63 6f 6c 75 6d 6e 5f 74 65 78 74 28 26 71  b_column_text(&q
3780: 2c 20 30 29 29 3b 0a 20 20 7d 0a 20 20 64 62 5f  , 0));.  }.  db_
3790: 66 69 6e 61 6c 69 7a 65 28 26 71 29 3b 0a 7d 0a  finalize(&q);.}.
37a0: 0a 2f 2a 0a 2a 2a 20 49 66 20 74 68 69 73 20 76  ./*.** If this v
37b0: 61 72 69 61 62 6c 65 20 69 73 20 73 65 74 2c 20  ariable is set, 
37c0: 64 69 73 61 62 6c 65 20 6c 6f 67 69 6e 20 63 68  disable login ch
37d0: 65 63 6b 73 2e 20 20 55 73 65 64 20 66 6f 72 20  ecks.  Used for 
37e0: 64 65 62 75 67 67 69 6e 67 0a 2a 2a 20 6f 6e 6c  debugging.** onl
37f0: 79 2e 0a 2a 2f 0a 73 74 61 74 69 63 20 69 6e 74  y..*/.static int
3800: 20 64 69 73 61 62 6c 65 4c 6f 67 69 6e 20 3d 20   disableLogin = 
3810: 30 3b 0a 0a 2f 2a 0a 2a 2a 20 54 68 65 20 43 47  0;../*.** The CG
3820: 49 2f 48 54 54 50 20 70 72 65 70 72 6f 63 65 73  I/HTTP preproces
3830: 73 6f 72 20 61 6c 77 61 79 73 20 72 65 64 69 72  sor always redir
3840: 65 63 74 73 20 72 65 71 75 65 73 74 73 20 77 69  ects requests wi
3850: 74 68 20 61 20 63 6f 6e 74 65 6e 74 2d 74 79 70  th a content-typ
3860: 65 0a 2a 2a 20 6f 66 20 61 70 70 6c 69 63 61 74  e.** of applicat
3870: 69 6f 6e 2f 78 2d 66 6f 73 73 69 6c 20 6f 72 20  ion/x-fossil or 
3880: 61 70 70 6c 69 63 61 74 69 6f 6e 2f 78 2d 66 6f  application/x-fo
3890: 73 73 69 6c 2d 64 65 62 75 67 20 74 6f 20 74 68  ssil-debug to th
38a0: 69 73 20 70 61 67 65 2c 0a 2a 2a 20 72 65 67 61  is page,.** rega
38b0: 72 64 6c 65 73 73 20 6f 66 20 77 68 61 74 20 70  rdless of what p
38c0: 61 74 68 20 77 61 73 20 73 70 65 63 69 66 69 65  ath was specifie
38d0: 64 20 69 6e 20 74 68 65 20 48 54 54 50 20 68 65  d in the HTTP he
38e0: 61 64 65 72 2e 20 20 54 68 69 73 20 61 6c 6c 6f  ader.  This allo
38f0: 77 73 0a 2a 2a 20 63 6c 6f 6e 65 20 63 6c 69 65  ws.** clone clie
3900: 6e 74 73 20 74 6f 20 73 70 65 63 69 66 79 20 61  nts to specify a
3910: 20 55 52 4c 20 74 68 61 74 20 6f 6d 69 74 73 20   URL that omits 
3920: 64 65 66 61 75 6c 74 20 70 61 74 68 6e 61 6d 65  default pathname
3930: 73 2c 20 73 75 63 68 0a 2a 2a 20 61 73 20 22 68  s, such.** as "h
3940: 74 74 70 3a 2f 2f 66 6f 73 73 69 6c 2d 73 63 6d  ttp://fossil-scm
3950: 2e 6d 6f 72 67 2f 22 20 69 6e 73 74 65 61 64 20  .morg/" instead 
3960: 6f 66 20 22 68 74 74 70 3a 2f 2f 66 6f 73 73 69  of "http://fossi
3970: 6c 2d 73 63 6d 2e 6f 72 67 2f 69 6e 64 65 78 2e  l-scm.org/index.
3980: 63 67 69 22 2e 0a 2a 2a 0a 2a 2a 20 57 45 42 50  cgi"..**.** WEBP
3990: 41 47 45 3a 20 78 66 65 72 0a 2a 2a 0a 2a 2a 20  AGE: xfer.**.** 
39a0: 54 68 69 73 20 69 73 20 74 68 65 20 74 72 61 6e  This is the tran
39b0: 73 66 65 72 20 68 61 6e 64 6c 65 72 20 6f 6e 20  sfer handler on 
39c0: 74 68 65 20 73 65 72 76 65 72 20 73 69 64 65 2e  the server side.
39d0: 20 20 54 68 65 20 74 72 61 6e 73 66 65 72 0a 2a    The transfer.*
39e0: 2a 20 6d 65 73 73 61 67 65 20 68 61 73 20 62 65  * message has be
39f0: 65 6e 20 75 6e 63 6f 6d 70 72 65 73 73 65 64 20  en uncompressed 
3a00: 61 6e 64 20 70 6c 61 63 65 64 20 69 6e 20 74 68  and placed in th
3a10: 65 20 67 2e 63 67 69 49 6e 20 62 6c 6f 62 2e 0a  e g.cgiIn blob..
3a20: 2a 2a 20 50 72 6f 63 65 73 73 20 74 68 69 73 20  ** Process this 
3a30: 6d 65 73 73 61 67 65 20 61 6e 64 20 66 6f 72 6d  message and form
3a40: 20 61 6e 20 61 70 70 72 6f 70 72 69 61 74 65 20   an appropriate 
3a50: 72 65 70 6c 79 2e 0a 2a 2f 0a 76 6f 69 64 20 70  reply..*/.void p
3a60: 61 67 65 5f 78 66 65 72 28 76 6f 69 64 29 7b 0a  age_xfer(void){.
3a70: 20 20 69 6e 74 20 69 73 50 75 6c 6c 20 3d 20 30    int isPull = 0
3a80: 3b 0a 20 20 69 6e 74 20 69 73 50 75 73 68 20 3d  ;.  int isPush =
3a90: 20 30 3b 0a 20 20 69 6e 74 20 6e 45 72 72 20 3d   0;.  int nErr =
3aa0: 20 30 3b 0a 20 20 58 66 65 72 20 78 66 65 72 3b   0;.  Xfer xfer;
3ab0: 0a 20 20 69 6e 74 20 64 65 6c 74 61 46 6c 61 67  .  int deltaFlag
3ac0: 20 3d 20 30 3b 0a 20 20 69 6e 74 20 69 73 43 6c   = 0;.  int isCl
3ad0: 6f 6e 65 20 3d 20 30 3b 0a 20 20 69 6e 74 20 6e  one = 0;.  int n
3ae0: 47 69 6d 6d 65 20 3d 20 30 3b 0a 0a 20 20 6d 65  Gimme = 0;..  me
3af0: 6d 73 65 74 28 26 78 66 65 72 2c 20 30 2c 20 73  mset(&xfer, 0, s
3b00: 69 7a 65 6f 66 28 78 66 65 72 29 29 3b 0a 20 20  izeof(xfer));.  
3b10: 62 6c 6f 62 61 72 72 61 79 5f 7a 65 72 6f 28 78  blobarray_zero(x
3b20: 66 65 72 2e 61 54 6f 6b 65 6e 2c 20 63 6f 75 6e  fer.aToken, coun
3b30: 74 28 78 66 65 72 2e 61 54 6f 6b 65 6e 29 29 3b  t(xfer.aToken));
3b40: 0a 20 20 63 67 69 5f 73 65 74 5f 63 6f 6e 74 65  .  cgi_set_conte
3b50: 6e 74 5f 74 79 70 65 28 67 2e 7a 43 6f 6e 74 65  nt_type(g.zConte
3b60: 6e 74 54 79 70 65 29 3b 0a 20 20 62 6c 6f 62 5f  ntType);.  blob_
3b70: 7a 65 72 6f 28 26 78 66 65 72 2e 65 72 72 29 3b  zero(&xfer.err);
3b80: 0a 20 20 78 66 65 72 2e 70 49 6e 20 3d 20 26 67  .  xfer.pIn = &g
3b90: 2e 63 67 69 49 6e 3b 0a 20 20 78 66 65 72 2e 70  .cgiIn;.  xfer.p
3ba0: 4f 75 74 20 3d 20 63 67 69 5f 6f 75 74 70 75 74  Out = cgi_output
3bb0: 5f 62 6c 6f 62 28 29 3b 0a 20 20 78 66 65 72 2e  _blob();.  xfer.
3bc0: 6d 78 53 65 6e 64 20 3d 20 64 62 5f 67 65 74 5f  mxSend = db_get_
3bd0: 69 6e 74 28 22 6d 61 78 2d 64 6f 77 6e 6c 6f 61  int("max-downloa
3be0: 64 22 2c 20 35 30 30 30 30 30 30 29 3b 0a 0a 20  d", 5000000);.. 
3bf0: 20 64 62 5f 62 65 67 69 6e 5f 74 72 61 6e 73 61   db_begin_transa
3c00: 63 74 69 6f 6e 28 29 3b 0a 20 20 64 62 5f 6d 75  ction();.  db_mu
3c10: 6c 74 69 5f 65 78 65 63 28 0a 20 20 20 20 20 22  lti_exec(.     "
3c20: 43 52 45 41 54 45 20 54 45 4d 50 20 54 41 42 4c  CREATE TEMP TABL
3c30: 45 20 6f 6e 72 65 6d 6f 74 65 28 72 69 64 20 49  E onremote(rid I
3c40: 4e 54 45 47 45 52 20 50 52 49 4d 41 52 59 20 4b  NTEGER PRIMARY K
3c50: 45 59 29 3b 22 0a 20 20 29 3b 0a 20 20 77 68 69  EY);".  );.  whi
3c60: 6c 65 28 20 62 6c 6f 62 5f 6c 69 6e 65 28 78 66  le( blob_line(xf
3c70: 65 72 2e 70 49 6e 2c 20 26 78 66 65 72 2e 6c 69  er.pIn, &xfer.li
3c80: 6e 65 29 20 29 7b 0a 20 20 20 20 69 66 28 20 62  ne) ){.    if( b
3c90: 6c 6f 62 5f 62 75 66 66 65 72 28 26 78 66 65 72  lob_buffer(&xfer
3ca0: 2e 6c 69 6e 65 29 5b 30 5d 3d 3d 27 23 27 20 29  .line)[0]=='#' )
3cb0: 20 63 6f 6e 74 69 6e 75 65 3b 0a 20 20 20 20 78   continue;.    x
3cc0: 66 65 72 2e 6e 54 6f 6b 65 6e 20 3d 20 62 6c 6f  fer.nToken = blo
3cd0: 62 5f 74 6f 6b 65 6e 69 7a 65 28 26 78 66 65 72  b_tokenize(&xfer
3ce0: 2e 6c 69 6e 65 2c 20 78 66 65 72 2e 61 54 6f 6b  .line, xfer.aTok
3cf0: 65 6e 2c 20 63 6f 75 6e 74 28 78 66 65 72 2e 61  en, count(xfer.a
3d00: 54 6f 6b 65 6e 29 29 3b 0a 0a 20 20 20 20 2f 2a  Token));..    /*
3d10: 20 20 20 66 69 6c 65 20 55 55 49 44 20 53 49 5a     file UUID SIZ
3d20: 45 20 5c 6e 20 43 4f 4e 54 45 4e 54 0a 20 20 20  E \n CONTENT.   
3d30: 20 2a 2a 20 20 20 66 69 6c 65 20 55 55 49 44 20   **   file UUID 
3d40: 44 45 4c 54 41 53 52 43 20 53 49 5a 45 20 5c 6e  DELTASRC SIZE \n
3d50: 20 43 4f 4e 54 45 4e 54 0a 20 20 20 20 2a 2a 0a   CONTENT.    **.
3d60: 20 20 20 20 2a 2a 20 41 63 63 65 70 74 20 61 20      ** Accept a 
3d70: 66 69 6c 65 20 66 72 6f 6d 20 74 68 65 20 63 6c  file from the cl
3d80: 69 65 6e 74 2e 0a 20 20 20 20 2a 2f 0a 20 20 20  ient..    */.   
3d90: 20 69 66 28 20 62 6c 6f 62 5f 65 71 28 26 78 66   if( blob_eq(&xf
3da0: 65 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 20 22 66  er.aToken[0], "f
3db0: 69 6c 65 22 29 20 29 7b 0a 20 20 20 20 20 20 69  ile") ){.      i
3dc0: 66 28 20 21 69 73 50 75 73 68 20 29 7b 0a 20 20  f( !isPush ){.  
3dd0: 20 20 20 20 20 20 63 67 69 5f 72 65 73 65 74 5f        cgi_reset_
3de0: 63 6f 6e 74 65 6e 74 28 29 3b 0a 20 20 20 20 20  content();.     
3df0: 20 20 20 40 20 65 72 72 6f 72 20 6e 6f 74 5c 73     @ error not\s
3e00: 61 75 74 68 6f 72 69 7a 65 64 5c 73 74 6f 5c 73  authorized\sto\s
3e10: 77 72 69 74 65 0a 20 20 20 20 20 20 20 20 6e 45  write.        nE
3e20: 72 72 2b 2b 3b 0a 20 20 20 20 20 20 20 20 62 72  rr++;.        br
3e30: 65 61 6b 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20  eak;.      }.   
3e40: 20 20 20 78 66 65 72 5f 61 63 63 65 70 74 5f 66     xfer_accept_f
3e50: 69 6c 65 28 26 78 66 65 72 29 3b 0a 20 20 20 20  ile(&xfer);.    
3e60: 20 20 69 66 28 20 62 6c 6f 62 5f 73 69 7a 65 28    if( blob_size(
3e70: 26 78 66 65 72 2e 65 72 72 29 20 29 7b 0a 20 20  &xfer.err) ){.  
3e80: 20 20 20 20 20 20 63 67 69 5f 72 65 73 65 74 5f        cgi_reset_
3e90: 63 6f 6e 74 65 6e 74 28 29 3b 0a 20 20 20 20 20  content();.     
3ea0: 20 20 20 40 20 65 72 72 6f 72 20 25 54 28 62 6c     @ error %T(bl
3eb0: 6f 62 5f 73 74 72 28 26 78 66 65 72 2e 65 72 72  ob_str(&xfer.err
3ec0: 29 29 0a 20 20 20 20 20 20 20 20 6e 45 72 72 2b  )).        nErr+
3ed0: 2b 3b 0a 20 20 20 20 20 20 20 20 62 72 65 61 6b  +;.        break
3ee0: 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20 7d 65  ;.      }.    }e
3ef0: 6c 73 65 0a 0a 20 20 20 20 2f 2a 20 20 20 67 69  lse..    /*   gi
3f00: 6d 6d 65 20 55 55 49 44 0a 20 20 20 20 2a 2a 0a  mme UUID.    **.
3f10: 20 20 20 20 2a 2a 20 43 6c 69 65 6e 74 20 69 73      ** Client is
3f20: 20 72 65 71 75 65 73 74 69 6e 67 20 61 20 66 69   requesting a fi
3f30: 6c 65 2e 20 20 53 65 6e 64 20 69 74 2e 0a 20 20  le.  Send it..  
3f40: 20 20 2a 2f 0a 20 20 20 20 69 66 28 20 62 6c 6f    */.    if( blo
3f50: 62 5f 65 71 28 26 78 66 65 72 2e 61 54 6f 6b 65  b_eq(&xfer.aToke
3f60: 6e 5b 30 5d 2c 20 22 67 69 6d 6d 65 22 29 0a 20  n[0], "gimme"). 
3f70: 20 20 20 20 26 26 20 78 66 65 72 2e 6e 54 6f 6b      && xfer.nTok
3f80: 65 6e 3d 3d 32 0a 20 20 20 20 20 26 26 20 62 6c  en==2.     && bl
3f90: 6f 62 5f 69 73 5f 75 75 69 64 28 26 78 66 65 72  ob_is_uuid(&xfer
3fa0: 2e 61 54 6f 6b 65 6e 5b 31 5d 29 0a 20 20 20 20  .aToken[1]).    
3fb0: 29 7b 0a 20 20 20 20 20 20 6e 47 69 6d 6d 65 2b  ){.      nGimme+
3fc0: 2b 3b 0a 20 20 20 20 20 20 69 66 28 20 69 73 50  +;.      if( isP
3fd0: 75 6c 6c 20 29 7b 0a 20 20 20 20 20 20 20 20 69  ull ){.        i
3fe0: 6e 74 20 72 69 64 20 3d 20 72 69 64 5f 66 72 6f  nt rid = rid_fro
3ff0: 6d 5f 75 75 69 64 28 26 78 66 65 72 2e 61 54 6f  m_uuid(&xfer.aTo
4000: 6b 65 6e 5b 31 5d 2c 20 30 29 3b 0a 20 20 20 20  ken[1], 0);.    
4010: 20 20 20 20 69 66 28 20 72 69 64 20 29 7b 0a 20      if( rid ){. 
4020: 20 20 20 20 20 20 20 20 20 73 65 6e 64 5f 66 69           send_fi
4030: 6c 65 28 26 78 66 65 72 2c 20 72 69 64 2c 20 26  le(&xfer, rid, &
4040: 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 31 5d 2c 20  xfer.aToken[1], 
4050: 64 65 6c 74 61 46 6c 61 67 29 3b 0a 20 20 20 20  deltaFlag);.    
4060: 20 20 20 20 7d 0a 20 20 20 20 20 20 7d 0a 20 20      }.      }.  
4070: 20 20 7d 65 6c 73 65 0a 0a 20 20 20 20 2f 2a 20    }else..    /* 
4080: 20 20 69 67 6f 74 20 55 55 49 44 0a 20 20 20 20    igot UUID.    
4090: 2a 2a 0a 20 20 20 20 2a 2a 20 43 6c 69 65 6e 74  **.    ** Client
40a0: 20 61 6e 6e 6f 75 6e 63 65 73 20 74 68 61 74 20   announces that 
40b0: 69 74 20 68 61 73 20 61 20 70 61 72 74 69 63 75  it has a particu
40c0: 6c 61 72 20 66 69 6c 65 2e 0a 20 20 20 20 2a 2f  lar file..    */
40d0: 0a 20 20 20 20 69 66 28 20 78 66 65 72 2e 6e 54  .    if( xfer.nT
40e0: 6f 6b 65 6e 3d 3d 32 0a 20 20 20 20 20 26 26 20  oken==2.     && 
40f0: 62 6c 6f 62 5f 65 71 28 26 78 66 65 72 2e 61 54  blob_eq(&xfer.aT
4100: 6f 6b 65 6e 5b 30 5d 2c 20 22 69 67 6f 74 22 29  oken[0], "igot")
4110: 0a 20 20 20 20 20 26 26 20 62 6c 6f 62 5f 69 73  .     && blob_is
4120: 5f 75 75 69 64 28 26 78 66 65 72 2e 61 54 6f 6b  _uuid(&xfer.aTok
4130: 65 6e 5b 31 5d 29 0a 20 20 20 20 29 7b 0a 20 20  en[1]).    ){.  
4140: 20 20 20 20 69 66 28 20 69 73 50 75 73 68 20 29      if( isPush )
4150: 7b 0a 20 20 20 20 20 20 20 20 72 69 64 5f 66 72  {.        rid_fr
4160: 6f 6d 5f 75 75 69 64 28 26 78 66 65 72 2e 61 54  om_uuid(&xfer.aT
4170: 6f 6b 65 6e 5b 31 5d 2c 20 31 29 3b 0a 20 20 20  oken[1], 1);.   
4180: 20 20 20 7d 0a 20 20 20 20 7d 65 6c 73 65 0a 20     }.    }else. 
4190: 20 0a 20 20 20 20 0a 20 20 20 20 2f 2a 20 20 20   .    .    /*   
41a0: 20 70 75 6c 6c 20 20 53 45 52 56 45 52 43 4f 44   pull  SERVERCOD
41b0: 45 20 20 50 52 4f 4a 45 43 54 43 4f 44 45 0a 20  E  PROJECTCODE. 
41c0: 20 20 20 2a 2a 20 20 20 20 70 75 73 68 20 20 53     **    push  S
41d0: 45 52 56 45 52 43 4f 44 45 20 20 50 52 4f 4a 45  ERVERCODE  PROJE
41e0: 43 54 43 4f 44 45 0a 20 20 20 20 2a 2a 0a 20 20  CTCODE.    **.  
41f0: 20 20 2a 2a 20 54 68 65 20 63 6c 69 65 6e 74 20    ** The client 
4200: 77 61 6e 74 73 20 65 69 74 68 65 72 20 73 65 6e  wants either sen
4210: 64 20 6f 72 20 72 65 63 65 69 76 65 2e 20 20 54  d or receive.  T
4220: 68 65 20 73 65 72 76 65 72 20 73 68 6f 75 6c 64  he server should
4230: 0a 20 20 20 20 2a 2a 20 76 65 72 69 66 79 20 74  .    ** verify t
4240: 68 61 74 20 74 68 65 20 70 72 6f 6a 65 63 74 20  hat the project 
4250: 63 6f 64 65 20 6d 61 74 63 68 65 73 20 61 6e 64  code matches and
4260: 20 74 68 61 74 20 74 68 65 20 73 65 72 76 65 72   that the server
4270: 20 63 6f 64 65 0a 20 20 20 20 2a 2a 20 64 6f 65   code.    ** doe
4280: 73 20 6e 6f 74 20 6d 61 74 63 68 2e 0a 20 20 20  s not match..   
4290: 20 2a 2f 0a 20 20 20 20 69 66 28 20 78 66 65 72   */.    if( xfer
42a0: 2e 6e 54 6f 6b 65 6e 3d 3d 33 0a 20 20 20 20 20  .nToken==3.     
42b0: 26 26 20 28 62 6c 6f 62 5f 65 71 28 26 78 66 65  && (blob_eq(&xfe
42c0: 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 20 22 70 75  r.aToken[0], "pu
42d0: 6c 6c 22 29 20 7c 7c 20 62 6c 6f 62 5f 65 71 28  ll") || blob_eq(
42e0: 26 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c  &xfer.aToken[0],
42f0: 20 22 70 75 73 68 22 29 29 0a 20 20 20 20 20 26   "push")).     &
4300: 26 20 62 6c 6f 62 5f 69 73 5f 75 75 69 64 28 26  & blob_is_uuid(&
4310: 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 31 5d 29 0a  xfer.aToken[1]).
4320: 20 20 20 20 20 26 26 20 62 6c 6f 62 5f 69 73 5f       && blob_is_
4330: 75 75 69 64 28 26 78 66 65 72 2e 61 54 6f 6b 65  uuid(&xfer.aToke
4340: 6e 5b 32 5d 29 0a 20 20 20 20 29 7b 0a 20 20 20  n[2]).    ){.   
4350: 20 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a     const char *z
4360: 53 43 6f 64 65 3b 0a 20 20 20 20 20 20 63 6f 6e  SCode;.      con
4370: 73 74 20 63 68 61 72 20 2a 7a 50 43 6f 64 65 3b  st char *zPCode;
4380: 0a 0a 20 20 20 20 20 20 7a 53 43 6f 64 65 20 3d  ..      zSCode =
4390: 20 64 62 5f 67 65 74 28 22 73 65 72 76 65 72 2d   db_get("server-
43a0: 63 6f 64 65 22 2c 20 30 29 3b 0a 20 20 20 20 20  code", 0);.     
43b0: 20 69 66 28 20 7a 53 43 6f 64 65 3d 3d 30 20 29   if( zSCode==0 )
43c0: 7b 0a 20 20 20 20 20 20 20 20 66 6f 73 73 69 6c  {.        fossil
43d0: 5f 70 61 6e 69 63 28 22 6d 69 73 73 69 6e 67 20  _panic("missing 
43e0: 73 65 72 76 65 72 20 63 6f 64 65 22 29 3b 0a 20  server code");. 
43f0: 20 20 20 20 20 7d 0a 20 20 20 20 20 20 69 66 28       }.      if(
4400: 20 62 6c 6f 62 5f 65 71 5f 73 74 72 28 26 78 66   blob_eq_str(&xf
4410: 65 72 2e 61 54 6f 6b 65 6e 5b 31 5d 2c 20 7a 53  er.aToken[1], zS
4420: 43 6f 64 65 2c 20 2d 31 29 20 29 7b 0a 20 20 20  Code, -1) ){.   
4430: 20 20 20 20 20 63 67 69 5f 72 65 73 65 74 5f 63       cgi_reset_c
4440: 6f 6e 74 65 6e 74 28 29 3b 0a 20 20 20 20 20 20  ontent();.      
4450: 20 20 40 20 65 72 72 6f 72 20 73 65 72 76 65 72    @ error server
4460: 5c 73 6c 6f 6f 70 0a 20 20 20 20 20 20 20 20 6e  \sloop.        n
4470: 45 72 72 2b 2b 3b 0a 20 20 20 20 20 20 20 20 62  Err++;.        b
4480: 72 65 61 6b 3b 0a 20 20 20 20 20 20 7d 0a 20 20  reak;.      }.  
4490: 20 20 20 20 7a 50 43 6f 64 65 20 3d 20 64 62 5f      zPCode = db_
44a0: 67 65 74 28 22 70 72 6f 6a 65 63 74 2d 63 6f 64  get("project-cod
44b0: 65 22 2c 20 30 29 3b 0a 20 20 20 20 20 20 69 66  e", 0);.      if
44c0: 28 20 7a 50 43 6f 64 65 3d 3d 30 20 29 7b 0a 20  ( zPCode==0 ){. 
44d0: 20 20 20 20 20 20 20 66 6f 73 73 69 6c 5f 70 61         fossil_pa
44e0: 6e 69 63 28 22 6d 69 73 73 69 6e 67 20 70 72 6f  nic("missing pro
44f0: 6a 65 63 74 20 63 6f 64 65 22 29 3b 0a 20 20 20  ject code");.   
4500: 20 20 20 7d 0a 20 20 20 20 20 20 69 66 28 20 21     }.      if( !
4510: 62 6c 6f 62 5f 65 71 5f 73 74 72 28 26 78 66 65  blob_eq_str(&xfe
4520: 72 2e 61 54 6f 6b 65 6e 5b 32 5d 2c 20 7a 50 43  r.aToken[2], zPC
4530: 6f 64 65 2c 20 2d 31 29 20 29 7b 0a 20 20 20 20  ode, -1) ){.    
4540: 20 20 20 20 63 67 69 5f 72 65 73 65 74 5f 63 6f      cgi_reset_co
4550: 6e 74 65 6e 74 28 29 3b 0a 20 20 20 20 20 20 20  ntent();.       
4560: 20 40 20 65 72 72 6f 72 20 77 72 6f 6e 67 5c 73   @ error wrong\s
4570: 70 72 6f 6a 65 63 74 0a 20 20 20 20 20 20 20 20  project.        
4580: 6e 45 72 72 2b 2b 3b 0a 20 20 20 20 20 20 20 20  nErr++;.        
4590: 62 72 65 61 6b 3b 0a 20 20 20 20 20 20 7d 0a 20  break;.      }. 
45a0: 20 20 20 20 20 6c 6f 67 69 6e 5f 63 68 65 63 6b       login_check
45b0: 5f 63 72 65 64 65 6e 74 69 61 6c 73 28 29 3b 0a  _credentials();.
45c0: 20 20 20 20 20 20 69 66 28 20 62 6c 6f 62 5f 65        if( blob_e
45d0: 71 28 26 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 30  q(&xfer.aToken[0
45e0: 5d 2c 20 22 70 75 6c 6c 22 29 20 29 7b 0a 20 20  ], "pull") ){.  
45f0: 20 20 20 20 20 20 69 66 28 20 21 67 2e 6f 6b 52        if( !g.okR
4600: 65 61 64 20 29 7b 0a 20 20 20 20 20 20 20 20 20  ead ){.         
4610: 20 63 67 69 5f 72 65 73 65 74 5f 63 6f 6e 74 65   cgi_reset_conte
4620: 6e 74 28 29 3b 0a 20 20 20 20 20 20 20 20 20 20  nt();.          
4630: 40 20 65 72 72 6f 72 20 6e 6f 74 5c 73 61 75 74  @ error not\saut
4640: 68 6f 72 69 7a 65 64 5c 73 74 6f 5c 73 72 65 61  horized\sto\srea
4650: 64 0a 20 20 20 20 20 20 20 20 20 20 6e 45 72 72  d.          nErr
4660: 2b 2b 3b 0a 20 20 20 20 20 20 20 20 20 20 62 72  ++;.          br
4670: 65 61 6b 3b 0a 20 20 20 20 20 20 20 20 7d 0a 20  eak;.        }. 
4680: 20 20 20 20 20 20 20 69 73 50 75 6c 6c 20 3d 20         isPull = 
4690: 31 3b 0a 20 20 20 20 20 20 7d 65 6c 73 65 7b 0a  1;.      }else{.
46a0: 20 20 20 20 20 20 20 20 69 66 28 20 21 67 2e 6f          if( !g.o
46b0: 6b 57 72 69 74 65 20 29 7b 0a 20 20 20 20 20 20  kWrite ){.      
46c0: 20 20 20 20 69 66 28 20 21 69 73 50 75 6c 6c 20      if( !isPull 
46d0: 29 7b 0a 20 20 20 20 20 20 20 20 20 20 20 20 63  ){.            c
46e0: 67 69 5f 72 65 73 65 74 5f 63 6f 6e 74 65 6e 74  gi_reset_content
46f0: 28 29 3b 0a 20 20 20 20 20 20 20 20 20 20 20 20  ();.            
4700: 40 20 65 72 72 6f 72 20 6e 6f 74 5c 73 61 75 74  @ error not\saut
4710: 68 6f 72 69 7a 65 64 5c 73 74 6f 5c 73 77 72 69  horized\sto\swri
4720: 74 65 0a 20 20 20 20 20 20 20 20 20 20 20 20 6e  te.            n
4730: 45 72 72 2b 2b 3b 0a 20 20 20 20 20 20 20 20 20  Err++;.         
4740: 20 7d 65 6c 73 65 7b 0a 20 20 20 20 20 20 20 20   }else{.        
4750: 20 20 20 20 40 20 6d 65 73 73 61 67 65 20 70 75      @ message pu
4760: 6c 6c 5c 73 6f 6e 6c 79 5c 73 2d 5c 73 6e 6f 74  ll\sonly\s-\snot
4770: 5c 73 61 75 74 68 6f 72 69 7a 65 64 5c 73 74 6f  \sauthorized\sto
4780: 5c 73 70 75 73 68 0a 20 20 20 20 20 20 20 20 20  \spush.         
4790: 20 7d 0a 20 20 20 20 20 20 20 20 7d 65 6c 73 65   }.        }else
47a0: 7b 0a 20 20 20 20 20 20 20 20 20 20 69 73 50 75  {.          isPu
47b0: 73 68 20 3d 20 31 3b 0a 20 20 20 20 20 20 20 20  sh = 1;.        
47c0: 7d 0a 20 20 20 20 20 20 7d 0a 20 20 20 20 7d 65  }.      }.    }e
47d0: 6c 73 65 0a 0a 20 20 20 20 2f 2a 20 20 20 20 63  lse..    /*    c
47e0: 6c 6f 6e 65 0a 20 20 20 20 2a 2a 0a 20 20 20 20  lone.    **.    
47f0: 2a 2a 20 54 68 65 20 63 6c 69 65 6e 74 20 6b 6e  ** The client kn
4800: 6f 77 73 20 6e 6f 74 68 69 6e 67 2e 20 20 54 65  ows nothing.  Te
4810: 6c 6c 20 61 6c 6c 2e 0a 20 20 20 20 2a 2f 0a 20  ll all..    */. 
4820: 20 20 20 69 66 28 20 62 6c 6f 62 5f 65 71 28 26     if( blob_eq(&
4830: 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 20  xfer.aToken[0], 
4840: 22 63 6c 6f 6e 65 22 29 20 29 7b 0a 20 20 20 20  "clone") ){.    
4850: 20 20 6c 6f 67 69 6e 5f 63 68 65 63 6b 5f 63 72    login_check_cr
4860: 65 64 65 6e 74 69 61 6c 73 28 29 3b 0a 20 20 20  edentials();.   
4870: 20 20 20 69 66 28 20 21 67 2e 6f 6b 43 6c 6f 6e     if( !g.okClon
4880: 65 20 29 7b 0a 20 20 20 20 20 20 20 20 63 67 69  e ){.        cgi
4890: 5f 72 65 73 65 74 5f 63 6f 6e 74 65 6e 74 28 29  _reset_content()
48a0: 3b 0a 20 20 20 20 20 20 20 20 40 20 65 72 72 6f  ;.        @ erro
48b0: 72 20 6e 6f 74 5c 73 61 75 74 68 6f 72 69 7a 65  r not\sauthorize
48c0: 64 5c 73 74 6f 5c 73 63 6c 6f 6e 65 0a 20 20 20  d\sto\sclone.   
48d0: 20 20 20 20 20 6e 45 72 72 2b 2b 3b 0a 20 20 20       nErr++;.   
48e0: 20 20 20 20 20 62 72 65 61 6b 3b 0a 20 20 20 20       break;.    
48f0: 20 20 7d 0a 20 20 20 20 20 20 69 73 43 6c 6f 6e    }.      isClon
4900: 65 20 3d 20 31 3b 0a 20 20 20 20 20 20 69 73 50  e = 1;.      isP
4910: 75 6c 6c 20 3d 20 31 3b 0a 20 20 20 20 20 20 64  ull = 1;.      d
4920: 65 6c 74 61 46 6c 61 67 20 3d 20 31 3b 0a 20 20  eltaFlag = 1;.  
4930: 20 20 20 20 40 20 70 75 73 68 20 25 73 28 64 62      @ push %s(db
4940: 5f 67 65 74 28 22 73 65 72 76 65 72 2d 63 6f 64  _get("server-cod
4950: 65 22 2c 20 22 78 22 29 29 20 25 73 28 64 62 5f  e", "x")) %s(db_
4960: 67 65 74 28 22 70 72 6f 6a 65 63 74 2d 63 6f 64  get("project-cod
4970: 65 22 2c 20 22 78 22 29 29 0a 20 20 20 20 7d 65  e", "x")).    }e
4980: 6c 73 65 0a 0a 20 20 20 20 2f 2a 20 20 20 20 6c  lse..    /*    l
4990: 6f 67 69 6e 20 20 55 53 45 52 20 20 4e 4f 4e 43  ogin  USER  NONC
49a0: 45 20 20 53 49 47 4e 41 54 55 52 45 0a 20 20 20  E  SIGNATURE.   
49b0: 20 2a 2a 0a 20 20 20 20 2a 2a 20 43 68 65 63 6b   **.    ** Check
49c0: 20 66 6f 72 20 61 20 76 61 6c 69 64 20 6c 6f 67   for a valid log
49d0: 69 6e 2e 20 20 54 68 69 73 20 68 61 73 20 74 6f  in.  This has to
49e0: 20 68 61 70 70 65 6e 20 62 65 66 6f 72 65 20 61   happen before a
49f0: 6e 79 74 68 69 6e 67 20 65 6c 73 65 2e 0a 20 20  nything else..  
4a00: 20 20 2a 2a 20 54 68 65 20 63 6c 69 65 6e 74 20    ** The client 
4a10: 63 61 6e 20 73 65 6e 64 20 6d 75 6c 74 69 70 6c  can send multipl
4a20: 65 20 6c 6f 67 69 6e 73 2e 20 20 50 65 72 6d 69  e logins.  Permi
4a30: 73 73 69 6f 6e 73 20 61 72 65 20 63 75 6d 75 6c  ssions are cumul
4a40: 61 74 69 76 65 2e 0a 20 20 20 20 2a 2f 0a 20 20  ative..    */.  
4a50: 20 20 69 66 28 20 62 6c 6f 62 5f 65 71 28 26 78    if( blob_eq(&x
4a60: 66 65 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 20 22  fer.aToken[0], "
4a70: 6c 6f 67 69 6e 22 29 0a 20 20 20 20 20 26 26 20  login").     && 
4a80: 78 66 65 72 2e 6e 54 6f 6b 65 6e 3d 3d 34 0a 20  xfer.nToken==4. 
4a90: 20 20 20 29 7b 0a 20 20 20 20 20 20 69 66 28 20     ){.      if( 
4aa0: 64 69 73 61 62 6c 65 4c 6f 67 69 6e 20 29 7b 0a  disableLogin ){.
4ab0: 20 20 20 20 20 20 20 20 67 2e 6f 6b 52 65 61 64          g.okRead
4ac0: 20 3d 20 67 2e 6f 6b 57 72 69 74 65 20 3d 20 31   = g.okWrite = 1
4ad0: 3b 0a 20 20 20 20 20 20 7d 65 6c 73 65 20 69 66  ;.      }else if
4ae0: 28 20 63 68 65 63 6b 5f 74 61 69 6c 5f 68 61 73  ( check_tail_has
4af0: 68 28 26 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 32  h(&xfer.aToken[2
4b00: 5d 2c 20 78 66 65 72 2e 70 49 6e 29 20 29 7b 0a  ], xfer.pIn) ){.
4b10: 20 20 20 20 20 20 20 20 63 68 65 63 6b 5f 6c 6f          check_lo
4b20: 67 69 6e 28 26 78 66 65 72 2e 61 54 6f 6b 65 6e  gin(&xfer.aToken
4b30: 5b 31 5d 2c 20 26 78 66 65 72 2e 61 54 6f 6b 65  [1], &xfer.aToke
4b40: 6e 5b 32 5d 2c 20 26 78 66 65 72 2e 61 54 6f 6b  n[2], &xfer.aTok
4b50: 65 6e 5b 33 5d 29 3b 0a 20 20 20 20 20 20 7d 0a  en[3]);.      }.
4b60: 20 20 20 20 7d 65 6c 73 65 0a 20 20 20 20 0a 20      }else.    . 
4b70: 20 20 20 2f 2a 20 20 20 20 72 65 71 63 6f 6e 66     /*    reqconf
4b80: 69 67 20 20 4e 41 4d 45 0a 20 20 20 20 2a 2a 0a  ig  NAME.    **.
4b90: 20 20 20 20 2a 2a 20 52 65 71 75 65 73 74 20 61      ** Request a
4ba0: 20 63 6f 6e 66 69 67 75 72 61 74 69 6f 6e 20 76   configuration v
4bb0: 61 6c 75 65 0a 20 20 20 20 2a 2f 0a 20 20 20 20  alue.    */.    
4bc0: 69 66 28 20 62 6c 6f 62 5f 65 71 28 26 78 66 65  if( blob_eq(&xfe
4bd0: 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 20 22 72 65  r.aToken[0], "re
4be0: 71 63 6f 6e 66 69 67 22 29 0a 20 20 20 20 20 26  qconfig").     &
4bf0: 26 20 78 66 65 72 2e 6e 54 6f 6b 65 6e 3d 3d 32  & xfer.nToken==2
4c00: 0a 20 20 20 20 29 7b 0a 20 20 20 20 20 20 69 66  .    ){.      if
4c10: 28 20 67 2e 6f 6b 52 65 61 64 20 29 7b 0a 20 20  ( g.okRead ){.  
4c20: 20 20 20 20 20 20 63 68 61 72 20 2a 7a 4e 61 6d        char *zNam
4c30: 65 20 3d 20 62 6c 6f 62 5f 73 74 72 28 26 78 66  e = blob_str(&xf
4c40: 65 72 2e 61 54 6f 6b 65 6e 5b 31 5d 29 3b 0a 20  er.aToken[1]);. 
4c50: 20 20 20 20 20 20 20 69 66 28 20 63 6f 6e 66 69         if( confi
4c60: 67 75 72 65 5f 69 73 5f 65 78 70 6f 72 74 61 62  gure_is_exportab
4c70: 6c 65 28 7a 4e 61 6d 65 29 20 29 7b 0a 20 20 20  le(zName) ){.   
4c80: 20 20 20 20 20 20 20 63 68 61 72 20 2a 7a 56 61         char *zVa
4c90: 6c 75 65 20 3d 20 64 62 5f 67 65 74 28 7a 4e 61  lue = db_get(zNa
4ca0: 6d 65 2c 20 30 29 3b 0a 20 20 20 20 20 20 20 20  me, 0);.        
4cb0: 20 20 69 66 28 20 7a 56 61 6c 75 65 20 29 7b 0a    if( zValue ){.
4cc0: 20 20 20 20 20 20 20 20 20 20 20 20 62 6c 6f 62              blob
4cd0: 5f 61 70 70 65 6e 64 66 28 78 66 65 72 2e 70 4f  _appendf(xfer.pO
4ce0: 75 74 2c 20 22 63 6f 6e 66 69 67 20 25 73 20 25  ut, "config %s %
4cf0: 64 5c 6e 25 73 5c 6e 22 2c 20 7a 4e 61 6d 65 2c  d\n%s\n", zName,
4d00: 20 0a 20 20 20 20 20 20 20 20 20 20 20 20 20 20   .              
4d10: 20 20 20 20 20 20 20 20 20 20 20 73 74 72 6c 65             strle
4d20: 6e 28 7a 56 61 6c 75 65 29 2c 20 7a 56 61 6c 75  n(zValue), zValu
4d30: 65 29 3b 0a 20 20 20 20 20 20 20 20 20 20 20 20  e);.            
4d40: 66 72 65 65 28 7a 56 61 6c 75 65 29 3b 0a 20 20  free(zValue);.  
4d50: 20 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20          }.      
4d60: 20 20 7d 0a 20 20 20 20 20 20 7d 0a 20 20 20 20    }.      }.    
4d70: 7d 65 6c 73 65 0a 20 20 20 20 0a 20 20 20 20 2f  }else.    .    /
4d80: 2a 20 20 20 20 63 6f 6f 6b 69 65 20 54 45 58 54  *    cookie TEXT
4d90: 0a 20 20 20 20 2a 2a 0a 20 20 20 20 2a 2a 20 41  .    **.    ** A
4da0: 20 63 6f 6f 6b 69 65 20 63 6f 6e 74 61 69 6e 73   cookie contains
4db0: 20 61 20 61 72 62 69 74 72 61 72 79 2d 6c 65 6e   a arbitrary-len
4dc0: 67 74 68 20 61 72 67 75 6d 65 6e 74 20 74 68 61  gth argument tha
4dd0: 74 20 69 73 20 73 65 72 76 65 72 2d 64 65 66 69  t is server-defi
4de0: 6e 65 64 2e 0a 20 20 20 20 2a 2a 20 54 68 65 20  ned..    ** The 
4df0: 61 72 67 75 6d 65 6e 74 20 6d 75 73 74 20 62 65  argument must be
4e00: 20 65 6e 63 6f 64 65 64 20 73 6f 20 61 73 20 6e   encoded so as n
4e10: 6f 74 20 74 6f 20 63 6f 6e 74 61 69 6e 20 61 6e  ot to contain an
4e20: 79 20 77 68 69 74 65 73 70 61 63 65 2e 0a 20 20  y whitespace..  
4e30: 20 20 2a 2a 20 54 68 65 20 73 65 72 76 65 72 20    ** The server 
4e40: 63 61 6e 20 6f 70 74 69 6f 6e 61 6c 6c 79 20 73  can optionally s
4e50: 65 6e 64 20 61 20 63 6f 6f 6b 69 65 20 74 6f 20  end a cookie to 
4e60: 74 68 65 20 63 6c 69 65 6e 74 2e 20 20 54 68 65  the client.  The
4e70: 20 63 6c 69 65 6e 74 0a 20 20 20 20 2a 2a 20 6d   client.    ** m
4e80: 69 67 68 74 20 74 68 65 6e 20 72 65 74 75 72 6e  ight then return
4e90: 20 74 68 65 20 73 61 6d 65 20 63 6f 6f 6b 69 65   the same cookie
4ea0: 20 62 61 63 6b 20 74 6f 20 74 68 65 20 73 65 72   back to the ser
4eb0: 76 65 72 20 6f 6e 20 69 74 73 20 6e 65 78 74 0a  ver on its next.
4ec0: 20 20 20 20 2a 2a 20 63 6f 6d 6d 75 6e 69 63 61      ** communica
4ed0: 74 69 6f 6e 2e 20 20 54 68 65 20 63 6f 6f 6b 69  tion.  The cooki
4ee0: 65 20 6d 69 67 68 74 20 72 65 63 6f 72 64 20 69  e might record i
4ef0: 6e 66 6f 72 6d 61 74 69 6f 6e 20 74 68 61 74 20  nformation that 
4f00: 68 65 6c 70 73 0a 20 20 20 20 2a 2a 20 74 68 65  helps.    ** the
4f10: 20 73 65 72 76 65 72 20 6f 70 74 69 6d 69 7a 65   server optimize
4f20: 20 61 20 70 75 73 68 20 6f 72 20 70 75 6c 6c 2e   a push or pull.
4f30: 0a 20 20 20 20 2a 2a 0a 20 20 20 20 2a 2a 20 54  .    **.    ** T
4f40: 68 65 20 63 6c 69 65 6e 74 20 69 73 20 6e 6f 74  he client is not
4f50: 20 72 65 71 75 69 72 65 64 20 74 6f 20 72 65 74   required to ret
4f60: 75 72 6e 20 61 20 63 6f 6f 6b 69 65 2e 20 20 53  urn a cookie.  S
4f70: 6f 20 74 68 65 20 73 65 72 76 65 72 0a 20 20 20  o the server.   
4f80: 20 2a 2a 20 6d 75 73 74 20 6e 6f 74 20 64 65 70   ** must not dep
4f90: 65 6e 64 20 6f 6e 20 74 68 65 20 63 6f 6f 6b 69  end on the cooki
4fa0: 65 2e 20 20 54 68 65 20 63 6f 6f 6b 69 65 20 73  e.  The cookie s
4fb0: 68 6f 75 6c 64 20 62 65 20 61 6e 20 6f 70 74 69  hould be an opti
4fc0: 6d 69 7a 61 74 69 6f 6e 0a 20 20 20 20 2a 2a 20  mization.    ** 
4fd0: 6f 6e 6c 79 2e 20 20 54 68 65 20 63 6c 69 65 6e  only.  The clien
4fe0: 74 20 6d 69 67 68 74 20 61 6c 73 6f 20 73 65 6e  t might also sen
4ff0: 64 20 61 20 63 6f 6f 6b 69 65 20 74 68 61 74 20  d a cookie that 
5000: 63 61 6d 65 20 66 72 6f 6d 20 61 20 64 69 66 66  came from a diff
5010: 65 72 65 6e 74 0a 20 20 20 20 2a 2a 20 73 65 72  erent.    ** ser
5020: 76 65 72 2e 20 20 53 6f 20 74 68 65 20 73 65 72  ver.  So the ser
5030: 76 65 72 20 6d 75 73 74 20 62 65 20 70 72 65 70  ver must be prep
5040: 61 72 65 64 20 74 6f 20 64 69 73 74 69 6e 67 75  ared to distingu
5050: 69 73 68 20 69 74 73 20 6f 77 6e 20 63 6f 6f 6b  ish its own cook
5060: 69 65 0a 20 20 20 20 2a 2a 20 66 72 6f 6d 20 63  ie.    ** from c
5070: 6f 6f 6b 69 65 73 20 6f 72 69 67 69 6e 61 74 69  ookies originati
5080: 6e 67 20 66 72 6f 6d 20 6f 74 68 65 72 20 73 65  ng from other se
5090: 72 76 65 72 73 2e 20 20 54 68 65 20 63 6c 69 65  rvers.  The clie
50a0: 6e 74 20 6d 69 67 68 74 20 73 65 6e 64 0a 20 20  nt might send.  
50b0: 20 20 2a 2a 20 62 61 63 6b 20 73 65 76 65 72 61    ** back severa
50c0: 6c 20 64 69 66 66 65 72 65 6e 74 20 63 6f 6f 6b  l different cook
50d0: 69 65 73 20 74 6f 20 74 68 65 20 73 65 72 76 65  ies to the serve
50e0: 72 2e 20 20 54 68 65 20 73 65 72 76 65 72 20 73  r.  The server s
50f0: 68 6f 75 6c 64 20 62 65 0a 20 20 20 20 2a 2a 20  hould be.    ** 
5100: 70 72 65 70 61 72 65 64 20 74 6f 20 73 69 66 74  prepared to sift
5110: 20 74 68 72 6f 75 67 68 20 74 68 65 20 63 6f 6f   through the coo
5120: 6b 69 65 73 20 61 6e 64 20 70 69 63 6b 20 74 68  kies and pick th
5130: 65 20 6f 6e 65 20 74 68 61 74 20 69 74 20 77 61  e one that it wa
5140: 6e 74 73 2e 0a 20 20 20 20 2a 2f 0a 20 20 20 20  nts..    */.    
5150: 69 66 28 20 62 6c 6f 62 5f 65 71 28 26 78 66 65  if( blob_eq(&xfe
5160: 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 20 22 63 6f  r.aToken[0], "co
5170: 6f 6b 69 65 22 29 20 26 26 20 78 66 65 72 2e 6e  okie") && xfer.n
5180: 54 6f 6b 65 6e 3d 3d 32 20 29 7b 0a 20 20 20 20  Token==2 ){.    
5190: 20 20 2f 2a 20 50 72 6f 63 65 73 73 20 74 68 65    /* Process the
51a0: 20 63 6f 6f 6b 69 65 20 2a 2f 0a 20 20 20 20 7d   cookie */.    }
51b0: 65 6c 73 65 0a 0a 20 20 20 20 2f 2a 20 55 6e 6b  else..    /* Unk
51c0: 6e 6f 77 6e 20 6d 65 73 73 61 67 65 0a 20 20 20  nown message.   
51d0: 20 2a 2f 0a 20 20 20 20 7b 0a 20 20 20 20 20 20   */.    {.      
51e0: 63 67 69 5f 72 65 73 65 74 5f 63 6f 6e 74 65 6e  cgi_reset_conten
51f0: 74 28 29 3b 0a 20 20 20 20 20 20 40 20 65 72 72  t();.      @ err
5200: 6f 72 20 62 61 64 5c 73 63 6f 6d 6d 61 6e 64 3a  or bad\scommand:
5210: 5c 73 25 46 28 62 6c 6f 62 5f 73 74 72 28 26 78  \s%F(blob_str(&x
5220: 66 65 72 2e 6c 69 6e 65 29 29 0a 20 20 20 20 7d  fer.line)).    }
5230: 0a 20 20 20 20 62 6c 6f 62 61 72 72 61 79 5f 72  .    blobarray_r
5240: 65 73 65 74 28 78 66 65 72 2e 61 54 6f 6b 65 6e  eset(xfer.aToken
5250: 2c 20 78 66 65 72 2e 6e 54 6f 6b 65 6e 29 3b 0a  , xfer.nToken);.
5260: 20 20 7d 0a 20 20 69 66 28 20 69 73 50 75 73 68    }.  if( isPush
5270: 20 29 7b 0a 20 20 20 20 72 65 71 75 65 73 74 5f   ){.    request_
5280: 70 68 61 6e 74 6f 6d 73 28 26 78 66 65 72 2c 20  phantoms(&xfer, 
5290: 35 30 30 29 3b 0a 20 20 7d 0a 20 20 69 66 28 20  500);.  }.  if( 
52a0: 69 73 43 6c 6f 6e 65 20 26 26 20 6e 47 69 6d 6d  isClone && nGimm
52b0: 65 3d 3d 30 20 29 7b 0a 20 20 20 20 2f 2a 20 54  e==0 ){.    /* T
52c0: 68 65 20 69 6e 69 74 69 61 6c 20 22 63 6c 6f 6e  he initial "clon
52d0: 65 22 20 6d 65 73 73 61 67 65 20 66 72 6f 6d 20  e" message from 
52e0: 63 6c 69 65 6e 74 20 74 6f 20 73 65 72 76 65 72  client to server
52f0: 20 63 6f 6e 74 61 69 6e 73 20 6e 6f 0a 20 20 20   contains no.   
5300: 20 2a 2a 20 22 67 69 6d 6d 65 22 20 63 61 72 64   ** "gimme" card
5310: 73 2e 20 4f 6e 20 74 68 61 74 20 69 6e 69 74 69  s. On that initi
5320: 61 6c 20 6d 65 73 73 61 67 65 2c 20 73 65 6e 64  al message, send
5330: 20 74 68 65 20 63 6c 69 65 6e 74 20 61 6e 20 22   the client an "
5340: 69 67 6f 74 22 0a 20 20 20 20 2a 2a 20 63 61 72  igot".    ** car
5350: 64 20 66 6f 72 20 65 76 65 72 79 20 61 72 74 69  d for every arti
5360: 66 61 63 74 20 63 75 72 72 65 6e 74 6c 79 20 69  fact currently i
5370: 6e 20 74 68 65 20 72 65 73 70 6f 73 69 74 6f 72  n the respositor
5380: 79 2e 20 20 54 68 69 73 20 77 69 6c 6c 0a 20 20  y.  This will.  
5390: 20 20 2a 2a 20 63 61 75 73 65 20 74 68 65 20 63    ** cause the c
53a0: 6c 69 65 6e 74 20 74 6f 20 63 72 65 61 74 65 20  lient to create 
53b0: 70 68 61 6e 74 6f 6d 73 20 66 6f 72 20 61 6c 6c  phantoms for all
53c0: 20 61 72 74 69 66 61 63 74 73 2c 20 77 68 69 63   artifacts, whic
53d0: 68 20 77 69 6c 6c 0a 20 20 20 20 2a 2a 20 69 6e  h will.    ** in
53e0: 20 74 75 72 6e 20 6d 61 6b 65 20 73 75 72 65 20   turn make sure 
53f0: 74 68 61 74 20 74 68 65 20 65 6e 74 69 72 65 20  that the entire 
5400: 72 65 70 6f 73 69 74 6f 72 79 20 69 73 20 73 65  repository is se
5410: 6e 74 20 65 66 66 69 63 69 65 6e 74 6c 79 0a 20  nt efficiently. 
5420: 20 20 20 2a 2a 20 61 6e 64 20 65 78 70 65 64 69     ** and expedi
5430: 74 69 6f 75 73 6c 79 2e 0a 20 20 20 20 2a 2f 0a  tiously..    */.
5440: 20 20 20 20 73 65 6e 64 5f 61 6c 6c 28 26 78 66      send_all(&xf
5450: 65 72 29 3b 0a 20 20 7d 65 6c 73 65 20 69 66 28  er);.  }else if(
5460: 20 69 73 50 75 6c 6c 20 29 7b 0a 20 20 20 20 63   isPull ){.    c
5470: 72 65 61 74 65 5f 63 6c 75 73 74 65 72 28 29 3b  reate_cluster();
5480: 0a 20 20 20 20 73 65 6e 64 5f 75 6e 63 6c 75 73  .    send_unclus
5490: 74 65 72 65 64 28 26 78 66 65 72 29 3b 0a 20 20  tered(&xfer);.  
54a0: 7d 0a 20 20 64 62 5f 65 6e 64 5f 74 72 61 6e 73  }.  db_end_trans
54b0: 61 63 74 69 6f 6e 28 30 29 3b 0a 7d 0a 0a 2f 2a  action(0);.}../*
54c0: 0a 2a 2a 20 43 4f 4d 4d 41 4e 44 3a 20 74 65 73  .** COMMAND: tes
54d0: 74 2d 78 66 65 72 0a 2a 2a 0a 2a 2a 20 54 68 69  t-xfer.**.** Thi
54e0: 73 20 63 6f 6d 6d 61 6e 64 20 69 73 20 75 73 65  s command is use
54f0: 64 20 66 6f 72 20 64 65 62 75 67 67 69 6e 67 20  d for debugging 
5500: 74 68 65 20 73 65 72 76 65 72 2e 20 20 54 68 65  the server.  The
5510: 72 65 20 69 73 20 61 20 73 69 6e 67 6c 65 0a 2a  re is a single.*
5520: 2a 20 61 72 67 75 6d 65 6e 74 20 77 68 69 63 68  * argument which
5530: 20 69 73 20 74 68 65 20 75 6e 63 6f 6d 70 72 65   is the uncompre
5540: 73 73 65 64 20 63 6f 6e 74 65 6e 74 20 6f 66 20  ssed content of 
5550: 61 6e 20 22 78 66 65 72 22 20 6d 65 73 73 61 67  an "xfer" messag
5560: 65 0a 2a 2a 20 66 72 6f 6d 20 63 6c 69 65 6e 74  e.** from client
5570: 20 74 6f 20 73 65 72 76 65 72 2e 20 20 54 68 69   to server.  Thi
5580: 73 20 63 6f 6d 6d 61 6e 64 20 69 6e 74 65 72 70  s command interp
5590: 72 65 74 73 20 74 68 61 74 20 6d 65 73 73 61 67  rets that messag
55a0: 65 20 61 73 0a 2a 2a 20 69 66 20 68 61 64 20 62  e as.** if had b
55b0: 65 65 6e 20 72 65 63 65 69 76 65 64 20 62 79 20  een received by 
55c0: 74 68 65 20 73 65 72 76 65 72 2e 0a 2a 2a 0a 2a  the server..**.*
55d0: 2a 20 4f 6e 20 74 68 65 20 63 6c 69 65 6e 74 20  * On the client 
55e0: 73 69 64 65 2c 20 72 75 6e 3a 0a 2a 2a 0a 2a 2a  side, run:.**.**
55f0: 20 20 20 20 20 20 66 6f 73 73 69 6c 20 70 75 73        fossil pus
5600: 68 20 68 74 74 70 3a 2f 2f 62 6f 67 75 73 2f 20  h http://bogus/ 
5610: 2d 2d 68 74 74 70 74 72 61 63 65 0a 2a 2a 0a 2a  --httptrace.**.*
5620: 2a 20 4f 72 20 61 20 73 69 6d 69 6c 61 72 20 63  * Or a similar c
5630: 6f 6d 6d 61 6e 64 20 74 6f 20 70 72 6f 76 69 64  ommand to provid
5640: 65 20 74 68 65 20 6f 75 74 70 75 74 2e 20 20 54  e the output.  T
5650: 68 65 20 63 6f 6e 74 65 6e 74 20 6f 66 20 74 68  he content of th
5660: 65 0a 2a 2a 20 6d 65 73 73 61 67 65 20 77 69 6c  e.** message wil
5670: 6c 20 61 70 70 65 61 72 20 6f 6e 20 73 74 61 6e  l appear on stan
5680: 64 61 72 64 20 6f 75 74 70 75 74 2e 20 20 43 61  dard output.  Ca
5690: 70 74 75 72 65 20 74 68 69 73 20 6d 65 73 73 61  pture this messa
56a0: 67 65 0a 2a 2a 20 69 6e 74 6f 20 61 20 66 69 6c  ge.** into a fil
56b0: 65 20 6e 61 6d 65 64 20 28 66 6f 72 20 65 78 61  e named (for exa
56c0: 6d 70 6c 65 29 20 6f 75 74 2e 74 78 74 2e 20 20  mple) out.txt.  
56d0: 54 68 65 6e 20 72 75 6e 20 74 68 65 0a 2a 2a 20  Then run the.** 
56e0: 73 65 72 76 65 72 20 69 6e 20 67 64 62 3a 0a 2a  server in gdb:.*
56f0: 2a 0a 2a 2a 20 20 20 20 20 67 64 62 20 66 6f 73  *.**     gdb fos
5700: 73 69 6c 0a 2a 2a 20 20 20 20 20 72 20 74 65 73  sil.**     r tes
5710: 74 2d 78 66 65 72 20 6f 75 74 2e 74 78 74 0a 2a  t-xfer out.txt.*
5720: 2f 0a 76 6f 69 64 20 63 6d 64 5f 74 65 73 74 5f  /.void cmd_test_
5730: 78 66 65 72 28 76 6f 69 64 29 7b 0a 20 20 69 6e  xfer(void){.  in
5740: 74 20 6e 6f 74 55 73 65 64 3b 0a 20 20 69 66 28  t notUsed;.  if(
5750: 20 67 2e 61 72 67 63 21 3d 32 20 26 26 20 67 2e   g.argc!=2 && g.
5760: 61 72 67 63 21 3d 33 20 29 7b 0a 20 20 20 20 75  argc!=3 ){.    u
5770: 73 61 67 65 28 22 3f 4d 45 53 53 41 47 45 46 49  sage("?MESSAGEFI
5780: 4c 45 3f 22 29 3b 0a 20 20 7d 0a 20 20 64 62 5f  LE?");.  }.  db_
5790: 6d 75 73 74 5f 62 65 5f 77 69 74 68 69 6e 5f 74  must_be_within_t
57a0: 72 65 65 28 29 3b 0a 20 20 62 6c 6f 62 5f 7a 65  ree();.  blob_ze
57b0: 72 6f 28 26 67 2e 63 67 69 49 6e 29 3b 0a 20 20  ro(&g.cgiIn);.  
57c0: 62 6c 6f 62 5f 72 65 61 64 5f 66 72 6f 6d 5f 66  blob_read_from_f
57d0: 69 6c 65 28 26 67 2e 63 67 69 49 6e 2c 20 67 2e  ile(&g.cgiIn, g.
57e0: 61 72 67 63 3d 3d 32 20 3f 20 22 2d 22 20 3a 20  argc==2 ? "-" : 
57f0: 67 2e 61 72 67 76 5b 32 5d 29 3b 0a 20 20 64 69  g.argv[2]);.  di
5800: 73 61 62 6c 65 4c 6f 67 69 6e 20 3d 20 31 3b 0a  sableLogin = 1;.
5810: 20 20 70 61 67 65 5f 78 66 65 72 28 29 3b 0a 20    page_xfer();. 
5820: 20 70 72 69 6e 74 66 28 22 25 73 5c 6e 22 2c 20   printf("%s\n", 
5830: 63 67 69 5f 65 78 74 72 61 63 74 5f 63 6f 6e 74  cgi_extract_cont
5840: 65 6e 74 28 26 6e 6f 74 55 73 65 64 29 29 3b 0a  ent(&notUsed));.
5850: 7d 0a 0a 2f 2a 0a 2a 2a 20 46 6f 72 6d 61 74 20  }../*.** Format 
5860: 73 74 72 69 6e 67 73 20 66 6f 72 20 70 72 6f 67  strings for prog
5870: 72 65 73 73 20 72 65 70 6f 72 74 69 6e 67 2e 0a  ress reporting..
5880: 2a 2f 0a 73 74 61 74 69 63 20 63 6f 6e 73 74 20  */.static const 
5890: 63 68 61 72 20 7a 4c 61 62 65 6c 46 6f 72 6d 61  char zLabelForma
58a0: 74 5b 5d 20 3d 20 22 25 2d 31 30 73 20 25 31 30  t[] = "%-10s %10
58b0: 73 20 25 31 30 73 20 25 31 30 73 20 25 31 30 73  s %10s %10s %10s
58c0: 5c 6e 22 3b 0a 73 74 61 74 69 63 20 63 6f 6e 73  \n";.static cons
58d0: 74 20 63 68 61 72 20 7a 56 61 6c 75 65 46 6f 72  t char zValueFor
58e0: 6d 61 74 5b 5d 20 3d 20 22 5c 72 25 2d 31 30 73  mat[] = "\r%-10s
58f0: 20 25 31 30 64 20 25 31 30 64 20 25 31 30 64 20   %10d %10d %10d 
5900: 25 31 30 64 5c 6e 22 3b 0a 0a 0a 2f 2a 0a 2a 2a  %10d\n";.../*.**
5910: 20 53 79 6e 63 20 74 6f 20 74 68 65 20 68 6f 73   Sync to the hos
5920: 74 20 69 64 65 6e 74 69 66 69 65 64 20 69 6e 20  t identified in 
5930: 67 2e 75 72 6c 4e 61 6d 65 20 61 6e 64 20 67 2e  g.urlName and g.
5940: 75 72 6c 50 61 74 68 2e 20 20 54 68 69 73 0a 2a  urlPath.  This.*
5950: 2a 20 72 6f 75 74 69 6e 65 20 69 73 20 63 61 6c  * routine is cal
5960: 6c 65 64 20 62 79 20 74 68 65 20 63 6c 69 65 6e  led by the clien
5970: 74 2e 0a 2a 2a 0a 2a 2a 20 52 65 63 6f 72 64 73  t..**.** Records
5980: 20 61 72 65 20 70 75 73 68 65 64 20 74 6f 20 74   are pushed to t
5990: 68 65 20 73 65 72 76 65 72 20 69 66 20 70 75 73  he server if pus
59a0: 68 46 6c 61 67 20 69 73 20 74 72 75 65 2e 20 20  hFlag is true.  
59b0: 52 65 63 6f 72 64 73 0a 2a 2a 20 61 72 65 20 70  Records.** are p
59c0: 75 6c 6c 65 64 20 69 66 20 70 75 6c 6c 46 6c 61  ulled if pullFla
59d0: 67 20 69 73 20 74 72 75 65 2e 20 20 41 20 66 75  g is true.  A fu
59e0: 6c 6c 20 73 79 6e 63 20 6f 63 63 75 72 73 20 69  ll sync occurs i
59f0: 66 20 62 6f 74 68 20 61 72 65 0a 2a 2a 20 74 72  f both are.** tr
5a00: 75 65 2e 0a 2a 2f 0a 76 6f 69 64 20 63 6c 69 65  ue..*/.void clie
5a10: 6e 74 5f 73 79 6e 63 28 69 6e 74 20 70 75 73 68  nt_sync(int push
5a20: 46 6c 61 67 2c 20 69 6e 74 20 70 75 6c 6c 46 6c  Flag, int pullFl
5a30: 61 67 2c 20 69 6e 74 20 63 6c 6f 6e 65 46 6c 61  ag, int cloneFla
5a40: 67 2c 20 69 6e 74 20 63 6f 6e 66 69 67 4d 61 73  g, int configMas
5a50: 6b 29 7b 0a 20 20 69 6e 74 20 67 6f 20 3d 20 31  k){.  int go = 1
5a60: 3b 20 20 20 20 20 20 20 20 2f 2a 20 4c 6f 6f 70  ;        /* Loop
5a70: 20 75 6e 74 69 6c 20 7a 65 72 6f 20 2a 2f 0a 20   until zero */. 
5a80: 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 53 43   const char *zSC
5a90: 6f 64 65 20 3d 20 64 62 5f 67 65 74 28 22 73 65  ode = db_get("se
5aa0: 72 76 65 72 2d 63 6f 64 65 22 2c 20 22 78 22 29  rver-code", "x")
5ab0: 3b 0a 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a  ;.  const char *
5ac0: 7a 50 43 6f 64 65 20 3d 20 64 62 5f 67 65 74 28  zPCode = db_get(
5ad0: 22 70 72 6f 6a 65 63 74 2d 63 6f 64 65 22 2c 20  "project-code", 
5ae0: 30 29 3b 0a 20 20 69 6e 74 20 6e 43 61 72 64 20  0);.  int nCard 
5af0: 3d 20 30 3b 20 20 20 20 20 20 20 20 20 2f 2a 20  = 0;         /* 
5b00: 4e 75 6d 62 65 72 20 6f 66 20 63 61 72 64 73 20  Number of cards 
5b10: 73 65 6e 74 20 6f 72 20 72 65 63 65 69 76 65 64  sent or received
5b20: 20 2a 2f 0a 20 20 69 6e 74 20 6e 43 79 63 6c 65   */.  int nCycle
5b30: 20 3d 20 30 3b 20 20 20 20 20 20 20 20 2f 2a 20   = 0;        /* 
5b40: 4e 75 6d 62 65 72 20 6f 66 20 72 6f 75 6e 64 20  Number of round 
5b50: 74 72 69 70 73 20 74 6f 20 74 68 65 20 73 65 72  trips to the ser
5b60: 76 65 72 20 2a 2f 0a 20 20 69 6e 74 20 73 69 7a  ver */.  int siz
5b70: 65 3b 20 20 20 20 20 20 20 20 20 20 20 20 20 20  e;              
5b80: 20 2f 2a 20 53 69 7a 65 20 6f 66 20 61 20 63 6f   /* Size of a co
5b90: 6e 66 69 67 20 76 61 6c 75 65 20 2a 2f 0a 20 20  nfig value */.  
5ba0: 69 6e 74 20 6e 46 69 6c 65 53 65 6e 64 20 3d 20  int nFileSend = 
5bb0: 30 3b 0a 20 20 69 6e 74 20 6f 72 69 67 43 6f 6e  0;.  int origCon
5bc0: 66 69 67 4d 61 73 6b 3b 20 20 20 20 20 2f 2a 20  figMask;     /* 
5bd0: 4f 72 69 67 69 6e 61 6c 20 76 61 6c 75 65 20 6f  Original value o
5be0: 66 20 63 6f 6e 66 69 67 4d 61 73 6b 20 2a 2f 0a  f configMask */.
5bf0: 20 20 69 6e 74 20 6e 46 69 6c 65 52 65 63 76 3b    int nFileRecv;
5c00: 20 20 20 20 20 20 20 20 20 20 2f 2a 20 4e 75 6d            /* Num
5c10: 62 65 72 20 6f 66 20 66 69 6c 65 73 20 72 65 63  ber of files rec
5c20: 65 69 76 65 64 20 2a 2f 0a 20 20 69 6e 74 20 6d  eived */.  int m
5c30: 78 50 68 61 6e 74 6f 6d 52 65 71 20 3d 20 32 30  xPhantomReq = 20
5c40: 30 3b 20 2f 2a 20 4d 61 78 20 6e 75 6d 62 65 72  0; /* Max number
5c50: 20 6f 66 20 70 68 61 6e 74 6f 6d 73 20 74 6f 20   of phantoms to 
5c60: 72 65 71 75 65 73 74 20 70 65 72 20 63 6f 6d 6d  request per comm
5c70: 20 2a 2f 0a 20 20 63 6f 6e 73 74 20 63 68 61 72   */.  const char
5c80: 20 2a 7a 43 6f 6f 6b 69 65 3b 20 20 20 20 2f 2a   *zCookie;    /*
5c90: 20 53 65 72 76 65 72 20 63 6f 6f 6b 69 65 20 2a   Server cookie *
5ca0: 2f 0a 20 20 42 6c 6f 62 20 73 65 6e 64 3b 20 20  /.  Blob send;  
5cb0: 20 20 20 20 20 20 2f 2a 20 54 65 78 74 20 77 65        /* Text we
5cc0: 20 61 72 65 20 73 65 6e 64 69 6e 67 20 74 6f 20   are sending to 
5cd0: 74 68 65 20 73 65 72 76 65 72 20 2a 2f 0a 20 20  the server */.  
5ce0: 42 6c 6f 62 20 72 65 63 76 3b 20 20 20 20 20 20  Blob recv;      
5cf0: 20 20 2f 2a 20 52 65 70 6c 79 20 77 65 20 67 6f    /* Reply we go
5d00: 74 20 62 61 63 6b 20 66 72 6f 6d 20 74 68 65 20  t back from the 
5d10: 73 65 72 76 65 72 20 2a 2f 0a 20 20 58 66 65 72  server */.  Xfer
5d20: 20 78 66 65 72 3b 20 20 20 20 20 20 20 20 2f 2a   xfer;        /*
5d30: 20 54 72 61 6e 73 66 65 72 20 64 61 74 61 20 2a   Transfer data *
5d40: 2f 0a 0a 20 20 6d 65 6d 73 65 74 28 26 78 66 65  /..  memset(&xfe
5d50: 72 2c 20 30 2c 20 73 69 7a 65 6f 66 28 78 66 65  r, 0, sizeof(xfe
5d60: 72 29 29 3b 0a 20 20 78 66 65 72 2e 70 49 6e 20  r));.  xfer.pIn 
5d70: 3d 20 26 72 65 63 76 3b 0a 20 20 78 66 65 72 2e  = &recv;.  xfer.
5d80: 70 4f 75 74 20 3d 20 26 73 65 6e 64 3b 0a 20 20  pOut = &send;.  
5d90: 78 66 65 72 2e 6d 78 53 65 6e 64 20 3d 20 64 62  xfer.mxSend = db
5da0: 5f 67 65 74 5f 69 6e 74 28 22 6d 61 78 2d 75 70  _get_int("max-up
5db0: 6c 6f 61 64 22 2c 20 32 35 30 30 30 30 29 3b 0a  load", 250000);.
5dc0: 0a 20 20 61 73 73 65 72 74 28 20 70 75 73 68 46  .  assert( pushF
5dd0: 6c 61 67 20 7c 7c 20 70 75 6c 6c 46 6c 61 67 20  lag || pullFlag 
5de0: 7c 7c 20 63 6c 6f 6e 65 46 6c 61 67 20 7c 7c 20  || cloneFlag || 
5df0: 63 6f 6e 66 69 67 4d 61 73 6b 20 29 3b 0a 20 20  configMask );.  
5e00: 61 73 73 65 72 74 28 20 21 67 2e 75 72 6c 49 73  assert( !g.urlIs
5e10: 46 69 6c 65 20 29 3b 20 20 20 20 20 20 20 20 20  File );         
5e20: 20 2f 2a 20 54 68 69 73 20 6f 6e 6c 79 20 77 6f   /* This only wo
5e30: 72 6b 73 20 66 6f 72 20 6e 65 74 77 6f 72 6b 69  rks for networki
5e40: 6e 67 20 2a 2f 0a 0a 20 20 64 62 5f 62 65 67 69  ng */..  db_begi
5e50: 6e 5f 74 72 61 6e 73 61 63 74 69 6f 6e 28 29 3b  n_transaction();
5e60: 0a 20 20 64 62 5f 6d 75 6c 74 69 5f 65 78 65 63  .  db_multi_exec
5e70: 28 0a 20 20 20 20 22 43 52 45 41 54 45 20 54 45  (.    "CREATE TE
5e80: 4d 50 20 54 41 42 4c 45 20 6f 6e 72 65 6d 6f 74  MP TABLE onremot
5e90: 65 28 72 69 64 20 49 4e 54 45 47 45 52 20 50 52  e(rid INTEGER PR
5ea0: 49 4d 41 52 59 20 4b 45 59 29 3b 22 0a 20 20 29  IMARY KEY);".  )
5eb0: 3b 0a 20 20 62 6c 6f 62 61 72 72 61 79 5f 7a 65  ;.  blobarray_ze
5ec0: 72 6f 28 78 66 65 72 2e 61 54 6f 6b 65 6e 2c 20  ro(xfer.aToken, 
5ed0: 63 6f 75 6e 74 28 78 66 65 72 2e 61 54 6f 6b 65  count(xfer.aToke
5ee0: 6e 29 29 3b 0a 20 20 62 6c 6f 62 5f 7a 65 72 6f  n));.  blob_zero
5ef0: 28 26 73 65 6e 64 29 3b 0a 20 20 62 6c 6f 62 5f  (&send);.  blob_
5f00: 7a 65 72 6f 28 26 72 65 63 76 29 3b 0a 20 20 62  zero(&recv);.  b
5f10: 6c 6f 62 5f 7a 65 72 6f 28 26 78 66 65 72 2e 65  lob_zero(&xfer.e
5f20: 72 72 29 3b 0a 20 20 62 6c 6f 62 5f 7a 65 72 6f  rr);.  blob_zero
5f30: 28 26 78 66 65 72 2e 6c 69 6e 65 29 3b 0a 20 20  (&xfer.line);.  
5f40: 6f 72 69 67 43 6f 6e 66 69 67 4d 61 73 6b 20 3d  origConfigMask =
5f50: 20 63 6f 6e 66 69 67 4d 61 73 6b 3b 0a 0a 20 20   configMask;..  
5f60: 2f 2a 0a 20 20 2a 2a 20 41 6c 77 61 79 73 20 62  /*.  ** Always b
5f70: 65 67 69 6e 20 77 69 74 68 20 61 20 63 6c 6f 6e  egin with a clon
5f80: 65 2c 20 70 75 6c 6c 2c 20 6f 72 20 70 75 73 68  e, pull, or push
5f90: 20 6d 65 73 73 61 67 65 0a 20 20 2a 2f 0a 20 20   message.  */.  
5fa0: 69 66 28 20 63 6c 6f 6e 65 46 6c 61 67 20 29 7b  if( cloneFlag ){
5fb0: 0a 20 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64  .    blob_append
5fc0: 66 28 26 73 65 6e 64 2c 20 22 63 6c 6f 6e 65 5c  f(&send, "clone\
5fd0: 6e 22 29 3b 0a 20 20 20 20 70 75 73 68 46 6c 61  n");.    pushFla
5fe0: 67 20 3d 20 30 3b 0a 20 20 20 20 70 75 6c 6c 46  g = 0;.    pullF
5ff0: 6c 61 67 20 3d 20 30 3b 0a 20 20 20 20 6e 43 61  lag = 0;.    nCa
6000: 72 64 2b 2b 3b 0a 20 20 20 20 2f 2a 20 54 42 44  rd++;.    /* TBD
6010: 3a 20 52 65 71 75 65 73 74 20 61 6c 6c 20 74 72  : Request all tr
6020: 61 6e 73 66 65 72 61 62 6c 65 20 63 6f 6e 66 69  ansferable confi
6030: 67 75 72 61 74 69 6f 6e 20 76 61 6c 75 65 73 20  guration values 
6040: 2a 2f 0a 20 20 7d 65 6c 73 65 20 69 66 28 20 70  */.  }else if( p
6050: 75 6c 6c 46 6c 61 67 20 29 7b 0a 20 20 20 20 62  ullFlag ){.    b
6060: 6c 6f 62 5f 61 70 70 65 6e 64 66 28 26 73 65 6e  lob_appendf(&sen
6070: 64 2c 20 22 70 75 6c 6c 20 25 73 20 25 73 5c 6e  d, "pull %s %s\n
6080: 22 2c 20 7a 53 43 6f 64 65 2c 20 7a 50 43 6f 64  ", zSCode, zPCod
6090: 65 29 3b 0a 20 20 20 20 6e 43 61 72 64 2b 2b 3b  e);.    nCard++;
60a0: 0a 20 20 7d 0a 20 20 69 66 28 20 70 75 73 68 46  .  }.  if( pushF
60b0: 6c 61 67 20 29 7b 0a 20 20 20 20 62 6c 6f 62 5f  lag ){.    blob_
60c0: 61 70 70 65 6e 64 66 28 26 73 65 6e 64 2c 20 22  appendf(&send, "
60d0: 70 75 73 68 20 25 73 20 25 73 5c 6e 22 2c 20 7a  push %s %s\n", z
60e0: 53 43 6f 64 65 2c 20 7a 50 43 6f 64 65 29 3b 0a  SCode, zPCode);.
60f0: 20 20 20 20 6e 43 61 72 64 2b 2b 3b 0a 20 20 7d      nCard++;.  }
6100: 0a 20 20 70 72 69 6e 74 66 28 7a 4c 61 62 65 6c  .  printf(zLabel
6110: 46 6f 72 6d 61 74 2c 20 22 22 2c 20 22 42 79 74  Format, "", "Byt
6120: 65 73 22 2c 20 22 43 61 72 64 73 22 2c 20 22 41  es", "Cards", "A
6130: 72 74 69 66 61 63 74 73 22 2c 20 22 44 65 6c 74  rtifacts", "Delt
6140: 61 73 22 29 3b 0a 0a 20 20 77 68 69 6c 65 28 20  as");..  while( 
6150: 67 6f 20 29 7b 0a 20 20 20 20 69 6e 74 20 6e 65  go ){.    int ne
6160: 77 50 68 61 6e 74 6f 6d 20 3d 20 30 3b 0a 20 20  wPhantom = 0;.  
6170: 20 20 63 68 61 72 20 2a 7a 52 61 6e 64 6f 6d 6e    char *zRandomn
6180: 65 73 73 3b 0a 0a 20 20 20 20 2f 2a 20 53 65 6e  ess;..    /* Sen
6190: 64 20 6d 61 6b 65 20 74 68 65 20 6d 6f 73 74 20  d make the most 
61a0: 72 65 63 65 6e 74 6c 79 20 72 65 63 65 69 76 65  recently receive
61b0: 64 20 63 6f 6f 6b 69 65 2e 20 20 4c 65 74 20 74  d cookie.  Let t
61c0: 68 65 20 73 65 72 76 65 72 0a 20 20 20 20 2a 2a  he server.    **
61d0: 20 66 69 67 75 72 65 20 6f 75 74 20 69 66 20 74   figure out if t
61e0: 68 69 73 20 69 73 20 61 20 63 6f 6f 6b 69 65 20  his is a cookie 
61f0: 74 68 61 74 20 69 74 20 63 61 72 65 73 20 61 62  that it cares ab
6200: 6f 75 74 2e 0a 20 20 20 20 2a 2f 0a 20 20 20 20  out..    */.    
6210: 7a 43 6f 6f 6b 69 65 20 3d 20 64 62 5f 67 65 74  zCookie = db_get
6220: 28 22 63 6f 6f 6b 69 65 22 2c 20 30 29 3b 0a 20  ("cookie", 0);. 
6230: 20 20 20 69 66 28 20 7a 43 6f 6f 6b 69 65 20 29     if( zCookie )
6240: 7b 0a 20 20 20 20 20 20 62 6c 6f 62 5f 61 70 70  {.      blob_app
6250: 65 6e 64 66 28 26 73 65 6e 64 2c 20 22 63 6f 6f  endf(&send, "coo
6260: 6b 69 65 20 25 73 5c 6e 22 2c 20 7a 43 6f 6f 6b  kie %s\n", zCook
6270: 69 65 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20 0a  ie);.    }.    .
6280: 20 20 20 20 2f 2a 20 47 65 6e 65 72 61 74 65 20      /* Generate 
6290: 67 69 6d 6d 65 20 63 61 72 64 73 20 66 6f 72 20  gimme cards for 
62a0: 70 68 61 6e 74 6f 6d 73 20 61 6e 64 20 6c 65 61  phantoms and lea
62b0: 66 20 63 61 72 64 73 0a 20 20 20 20 2a 2a 20 66  f cards.    ** f
62c0: 6f 72 20 61 6c 6c 20 6c 65 61 76 65 73 2e 0a 20  or all leaves.. 
62d0: 20 20 20 2a 2f 0a 20 20 20 20 69 66 28 20 70 75     */.    if( pu
62e0: 6c 6c 46 6c 61 67 20 7c 7c 20 63 6c 6f 6e 65 46  llFlag || cloneF
62f0: 6c 61 67 20 29 7b 0a 20 20 20 20 20 20 72 65 71  lag ){.      req
6300: 75 65 73 74 5f 70 68 61 6e 74 6f 6d 73 28 26 78  uest_phantoms(&x
6310: 66 65 72 2c 20 6d 78 50 68 61 6e 74 6f 6d 52 65  fer, mxPhantomRe
6320: 71 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20 69 66  q);.    }.    if
6330: 28 20 70 75 73 68 46 6c 61 67 20 29 7b 0a 20 20  ( pushFlag ){.  
6340: 20 20 20 20 73 65 6e 64 5f 75 6e 73 65 6e 74 28      send_unsent(
6350: 26 78 66 65 72 29 3b 0a 20 20 20 20 20 20 6e 43  &xfer);.      nC
6360: 61 72 64 20 2b 3d 20 73 65 6e 64 5f 75 6e 63 6c  ard += send_uncl
6370: 75 73 74 65 72 65 64 28 26 78 66 65 72 29 3b 0a  ustered(&xfer);.
6380: 20 20 20 20 7d 0a 0a 20 20 20 20 2f 2a 20 53 65      }..    /* Se
6390: 6e 64 20 63 6f 6e 66 69 67 75 72 61 74 69 6f 6e  nd configuration
63a0: 20 70 61 72 61 6d 65 74 65 72 20 72 65 71 75 65   parameter reque
63b0: 73 74 73 20 2a 2f 0a 20 20 20 20 69 66 28 20 63  sts */.    if( c
63c0: 6f 6e 66 69 67 4d 61 73 6b 20 29 7b 0a 20 20 20  onfigMask ){.   
63d0: 20 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a     const char *z
63e0: 4e 61 6d 65 3b 0a 20 20 20 20 20 20 7a 4e 61 6d  Name;.      zNam
63f0: 65 20 3d 20 63 6f 6e 66 69 67 75 72 65 5f 66 69  e = configure_fi
6400: 72 73 74 5f 6e 61 6d 65 28 63 6f 6e 66 69 67 4d  rst_name(configM
6410: 61 73 6b 29 3b 0a 20 20 20 20 20 20 77 68 69 6c  ask);.      whil
6420: 65 28 20 7a 4e 61 6d 65 20 29 7b 0a 20 20 20 20  e( zName ){.    
6430: 20 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66      blob_appendf
6440: 28 26 73 65 6e 64 2c 20 22 72 65 71 63 6f 6e 66  (&send, "reqconf
6450: 69 67 20 25 73 5c 6e 22 2c 20 7a 4e 61 6d 65 29  ig %s\n", zName)
6460: 3b 0a 20 20 20 20 20 20 20 20 7a 4e 61 6d 65 20  ;.        zName 
6470: 3d 20 63 6f 6e 66 69 67 75 72 65 5f 6e 65 78 74  = configure_next
6480: 5f 6e 61 6d 65 28 63 6f 6e 66 69 67 4d 61 73 6b  _name(configMask
6490: 29 3b 0a 20 20 20 20 20 20 20 20 6e 43 61 72 64  );.        nCard
64a0: 2b 2b 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20  ++;.      }.    
64b0: 20 20 63 6f 6e 66 69 67 4d 61 73 6b 20 3d 20 30    configMask = 0
64c0: 3b 0a 20 20 20 20 7d 0a 0a 20 20 20 20 2f 2a 20  ;.    }..    /* 
64d0: 41 70 70 65 6e 64 20 72 61 6e 64 6f 6d 6e 65 73  Append randomnes
64e0: 73 20 74 6f 20 74 68 65 20 65 6e 64 20 6f 66 20  s to the end of 
64f0: 74 68 65 20 6d 65 73 73 61 67 65 20 2a 2f 0a 23  the message */.#
6500: 69 66 20 30 20 20 20 2f 2a 20 45 6e 61 62 6c 65  if 0   /* Enable
6510: 20 74 68 69 73 20 61 66 74 65 72 20 61 6c 6c 20   this after all 
6520: 73 65 72 76 65 72 73 20 68 61 76 65 20 75 70 67  servers have upg
6530: 72 61 64 65 64 20 2a 2f 0a 20 20 20 20 7a 52 61  raded */.    zRa
6540: 6e 64 6f 6d 6e 65 73 73 20 3d 20 64 62 5f 74 65  ndomness = db_te
6550: 78 74 28 30 2c 20 22 53 45 4c 45 43 54 20 68 65  xt(0, "SELECT he
6560: 78 28 72 61 6e 64 6f 6d 62 6c 6f 62 28 32 30 29  x(randomblob(20)
6570: 29 22 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 61 70  )");.    blob_ap
6580: 70 65 6e 64 66 28 26 73 65 6e 64 2c 20 22 23 20  pendf(&send, "# 
6590: 25 73 5c 6e 22 2c 20 7a 52 61 6e 64 6f 6d 6e 65  %s\n", zRandomne
65a0: 73 73 29 3b 0a 20 20 20 20 66 72 65 65 28 7a 52  ss);.    free(zR
65b0: 61 6e 64 6f 6d 6e 65 73 73 29 3b 0a 23 65 6e 64  andomness);.#end
65c0: 69 66 0a 0a 20 20 20 20 2f 2a 20 45 78 63 68 61  if..    /* Excha
65d0: 6e 67 65 20 6d 65 73 73 61 67 65 73 20 77 69 74  nge messages wit
65e0: 68 20 74 68 65 20 73 65 72 76 65 72 20 2a 2f 0a  h the server */.
65f0: 20 20 20 20 6e 46 69 6c 65 53 65 6e 64 20 3d 20      nFileSend = 
6600: 78 66 65 72 2e 6e 46 69 6c 65 53 65 6e 74 20 2b  xfer.nFileSent +
6610: 20 78 66 65 72 2e 6e 44 65 6c 74 61 53 65 6e 74   xfer.nDeltaSent
6620: 3b 0a 20 20 20 20 70 72 69 6e 74 66 28 7a 56 61  ;.    printf(zVa
6630: 6c 75 65 46 6f 72 6d 61 74 2c 20 22 53 65 6e 64  lueFormat, "Send
6640: 3a 22 2c 0a 20 20 20 20 20 20 20 20 20 20 20 20  :",.            
6650: 62 6c 6f 62 5f 73 69 7a 65 28 26 73 65 6e 64 29  blob_size(&send)
6660: 2c 20 6e 43 61 72 64 2b 78 66 65 72 2e 6e 47 69  , nCard+xfer.nGi
6670: 6d 6d 65 53 65 6e 74 2b 78 66 65 72 2e 6e 49 47  mmeSent+xfer.nIG
6680: 6f 74 53 65 6e 74 2c 0a 20 20 20 20 20 20 20 20  otSent,.        
6690: 20 20 20 20 78 66 65 72 2e 6e 46 69 6c 65 53 65      xfer.nFileSe
66a0: 6e 74 2c 20 78 66 65 72 2e 6e 44 65 6c 74 61 53  nt, xfer.nDeltaS
66b0: 65 6e 74 29 3b 0a 20 20 20 20 6e 43 61 72 64 20  ent);.    nCard 
66c0: 3d 20 30 3b 0a 20 20 20 20 78 66 65 72 2e 6e 46  = 0;.    xfer.nF
66d0: 69 6c 65 53 65 6e 74 20 3d 20 30 3b 0a 20 20 20  ileSent = 0;.   
66e0: 20 78 66 65 72 2e 6e 44 65 6c 74 61 53 65 6e 74   xfer.nDeltaSent
66f0: 20 3d 20 30 3b 0a 20 20 20 20 78 66 65 72 2e 6e   = 0;.    xfer.n
6700: 47 69 6d 6d 65 53 65 6e 74 20 3d 20 30 3b 0a 20  GimmeSent = 0;. 
6710: 20 20 20 66 66 6c 75 73 68 28 73 74 64 6f 75 74     fflush(stdout
6720: 29 3b 0a 20 20 20 20 68 74 74 70 5f 65 78 63 68  );.    http_exch
6730: 61 6e 67 65 28 26 73 65 6e 64 2c 20 26 72 65 63  ange(&send, &rec
6740: 76 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 72 65 73  v);.    blob_res
6750: 65 74 28 26 73 65 6e 64 29 3b 0a 0a 20 20 20 20  et(&send);..    
6760: 2f 2a 20 42 65 67 69 6e 20 63 6f 6e 73 74 72 75  /* Begin constru
6770: 63 74 69 6e 67 20 74 68 65 20 6e 65 78 74 20 6d  cting the next m
6780: 65 73 73 61 67 65 20 28 77 68 69 63 68 20 6d 69  essage (which mi
6790: 67 68 74 20 6e 65 76 65 72 20 62 65 0a 20 20 20  ght never be.   
67a0: 20 2a 2a 20 73 65 6e 74 29 20 62 79 20 62 65 67   ** sent) by beg
67b0: 69 6e 6e 69 6e 67 20 77 69 74 68 20 74 68 65 20  inning with the 
67c0: 70 75 6c 6c 20 6f 72 20 70 75 73 68 20 63 61 72  pull or push car
67d0: 64 73 0a 20 20 20 20 2a 2f 0a 20 20 20 20 69 66  ds.    */.    if
67e0: 28 20 70 75 6c 6c 46 6c 61 67 20 29 7b 0a 20 20  ( pullFlag ){.  
67f0: 20 20 20 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66      blob_appendf
6800: 28 26 73 65 6e 64 2c 20 22 70 75 6c 6c 20 25 73  (&send, "pull %s
6810: 20 25 73 5c 6e 22 2c 20 7a 53 43 6f 64 65 2c 20   %s\n", zSCode, 
6820: 7a 50 43 6f 64 65 29 3b 0a 20 20 20 20 20 20 6e  zPCode);.      n
6830: 43 61 72 64 2b 2b 3b 0a 20 20 20 20 7d 0a 20 20  Card++;.    }.  
6840: 20 20 69 66 28 20 70 75 73 68 46 6c 61 67 20 29    if( pushFlag )
6850: 7b 0a 20 20 20 20 20 20 62 6c 6f 62 5f 61 70 70  {.      blob_app
6860: 65 6e 64 66 28 26 73 65 6e 64 2c 20 22 70 75 73  endf(&send, "pus
6870: 68 20 25 73 20 25 73 5c 6e 22 2c 20 7a 53 43 6f  h %s %s\n", zSCo
6880: 64 65 2c 20 7a 50 43 6f 64 65 29 3b 0a 20 20 20  de, zPCode);.   
6890: 20 20 20 6e 43 61 72 64 2b 2b 3b 0a 20 20 20 20     nCard++;.    
68a0: 7d 0a 0a 20 20 20 20 2f 2a 20 50 72 6f 63 65 73  }..    /* Proces
68b0: 73 20 74 68 65 20 72 65 70 6c 79 20 74 68 61 74  s the reply that
68c0: 20 63 61 6d 65 20 62 61 63 6b 20 66 72 6f 6d 20   came back from 
68d0: 74 68 65 20 73 65 72 76 65 72 20 2a 2f 0a 20 20  the server */.  
68e0: 20 20 77 68 69 6c 65 28 20 62 6c 6f 62 5f 6c 69    while( blob_li
68f0: 6e 65 28 26 72 65 63 76 2c 20 26 78 66 65 72 2e  ne(&recv, &xfer.
6900: 6c 69 6e 65 29 20 29 7b 0a 20 20 20 20 20 20 69  line) ){.      i
6910: 66 28 20 62 6c 6f 62 5f 62 75 66 66 65 72 28 26  f( blob_buffer(&
6920: 78 66 65 72 2e 6c 69 6e 65 29 5b 30 5d 3d 3d 27  xfer.line)[0]=='
6930: 23 27 20 29 7b 0a 20 20 20 20 20 20 20 20 63 6f  #' ){.        co
6940: 6e 74 69 6e 75 65 3b 0a 20 20 20 20 20 20 7d 0a  ntinue;.      }.
6950: 20 20 20 20 20 20 78 66 65 72 2e 6e 54 6f 6b 65        xfer.nToke
6960: 6e 20 3d 20 62 6c 6f 62 5f 74 6f 6b 65 6e 69 7a  n = blob_tokeniz
6970: 65 28 26 78 66 65 72 2e 6c 69 6e 65 2c 20 78 66  e(&xfer.line, xf
6980: 65 72 2e 61 54 6f 6b 65 6e 2c 20 63 6f 75 6e 74  er.aToken, count
6990: 28 78 66 65 72 2e 61 54 6f 6b 65 6e 29 29 3b 0a  (xfer.aToken));.
69a0: 20 20 20 20 20 20 6e 43 61 72 64 2b 2b 3b 0a 20        nCard++;. 
69b0: 20 20 20 20 20 70 72 69 6e 74 66 28 22 5c 72 25       printf("\r%
69c0: 64 22 2c 20 6e 43 61 72 64 29 3b 0a 20 20 20 20  d", nCard);.    
69d0: 20 20 66 66 6c 75 73 68 28 73 74 64 6f 75 74 29    fflush(stdout)
69e0: 3b 0a 0a 20 20 20 20 20 20 2f 2a 20 20 20 66 69  ;..      /*   fi
69f0: 6c 65 20 55 55 49 44 20 53 49 5a 45 20 5c 6e 20  le UUID SIZE \n 
6a00: 43 4f 4e 54 45 4e 54 0a 20 20 20 20 20 20 2a 2a  CONTENT.      **
6a10: 20 20 20 66 69 6c 65 20 55 55 49 44 20 44 45 4c     file UUID DEL
6a20: 54 41 53 52 43 20 53 49 5a 45 20 5c 6e 20 43 4f  TASRC SIZE \n CO
6a30: 4e 54 45 4e 54 0a 20 20 20 20 20 20 2a 2a 0a 20  NTENT.      **. 
6a40: 20 20 20 20 20 2a 2a 20 52 65 63 65 69 76 65 20       ** Receive 
6a50: 61 20 66 69 6c 65 20 74 72 61 6e 73 6d 69 74 74  a file transmitt
6a60: 65 64 20 66 72 6f 6d 20 74 68 65 20 73 65 72 76  ed from the serv
6a70: 65 72 2e 0a 20 20 20 20 20 20 2a 2f 0a 20 20 20  er..      */.   
6a80: 20 20 20 69 66 28 20 62 6c 6f 62 5f 65 71 28 26     if( blob_eq(&
6a90: 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 22  xfer.aToken[0],"
6aa0: 66 69 6c 65 22 29 20 29 7b 0a 20 20 20 20 20 20  file") ){.      
6ab0: 20 20 78 66 65 72 5f 61 63 63 65 70 74 5f 66 69    xfer_accept_fi
6ac0: 6c 65 28 26 78 66 65 72 29 3b 0a 20 20 20 20 20  le(&xfer);.     
6ad0: 20 7d 65 6c 73 65 0a 0a 20 20 20 20 20 20 2f 2a   }else..      /*
6ae0: 20 20 20 67 69 6d 6d 65 20 55 55 49 44 0a 20 20     gimme UUID.  
6af0: 20 20 20 20 2a 2a 0a 20 20 20 20 20 20 2a 2a 20      **.      ** 
6b00: 53 65 72 76 65 72 20 69 73 20 72 65 71 75 65 73  Server is reques
6b10: 74 69 6e 67 20 61 20 66 69 6c 65 2e 20 20 49 66  ting a file.  If
6b20: 20 74 68 65 20 66 69 6c 65 20 69 73 20 61 20 6d   the file is a m
6b30: 61 6e 69 66 65 73 74 2c 20 61 73 73 75 6d 65 0a  anifest, assume.
6b40: 20 20 20 20 20 20 2a 2a 20 74 68 61 74 20 74 68        ** that th
6b50: 65 20 73 65 72 76 65 72 20 77 69 6c 6c 20 61 6c  e server will al
6b60: 73 6f 20 77 61 6e 74 20 74 6f 20 6b 6e 6f 77 20  so want to know 
6b70: 61 6c 6c 20 6f 66 20 74 68 65 20 63 6f 6e 74 65  all of the conte
6b80: 6e 74 20 66 69 6c 65 73 0a 20 20 20 20 20 20 2a  nt files.      *
6b90: 2a 20 61 73 73 6f 63 69 61 74 65 64 20 77 69 74  * associated wit
6ba0: 68 20 74 68 65 20 6d 61 6e 69 66 65 73 74 20 61  h the manifest a
6bb0: 6e 64 20 73 65 6e 64 20 74 68 6f 73 65 20 74 6f  nd send those to
6bc0: 6f 2e 0a 20 20 20 20 20 20 2a 2f 0a 20 20 20 20  o..      */.    
6bd0: 20 20 69 66 28 20 62 6c 6f 62 5f 65 71 28 26 78    if( blob_eq(&x
6be0: 66 65 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 20 22  fer.aToken[0], "
6bf0: 67 69 6d 6d 65 22 29 0a 20 20 20 20 20 20 20 26  gimme").       &
6c00: 26 20 78 66 65 72 2e 6e 54 6f 6b 65 6e 3d 3d 32  & xfer.nToken==2
6c10: 0a 20 20 20 20 20 20 20 26 26 20 62 6c 6f 62 5f  .       && blob_
6c20: 69 73 5f 75 75 69 64 28 26 78 66 65 72 2e 61 54  is_uuid(&xfer.aT
6c30: 6f 6b 65 6e 5b 31 5d 29 0a 20 20 20 20 20 20 29  oken[1]).      )
6c40: 7b 0a 20 20 20 20 20 20 20 20 69 66 28 20 70 75  {.        if( pu
6c50: 73 68 46 6c 61 67 20 29 7b 0a 20 20 20 20 20 20  shFlag ){.      
6c60: 20 20 20 20 69 6e 74 20 72 69 64 20 3d 20 72 69      int rid = ri
6c70: 64 5f 66 72 6f 6d 5f 75 75 69 64 28 26 78 66 65  d_from_uuid(&xfe
6c80: 72 2e 61 54 6f 6b 65 6e 5b 31 5d 2c 20 30 29 3b  r.aToken[1], 0);
6c90: 0a 20 20 20 20 20 20 20 20 20 20 69 66 28 20 72  .          if( r
6ca0: 69 64 20 29 20 73 65 6e 64 5f 66 69 6c 65 28 26  id ) send_file(&
6cb0: 78 66 65 72 2c 20 72 69 64 2c 20 26 78 66 65 72  xfer, rid, &xfer
6cc0: 2e 61 54 6f 6b 65 6e 5b 31 5d 2c 20 30 29 3b 0a  .aToken[1], 0);.
6cd0: 20 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20          }.      
6ce0: 7d 65 6c 73 65 0a 20 20 0a 20 20 20 20 20 20 2f  }else.  .      /
6cf0: 2a 20 20 20 69 67 6f 74 20 55 55 49 44 0a 20 20  *   igot UUID.  
6d00: 20 20 20 20 2a 2a 0a 20 20 20 20 20 20 2a 2a 20      **.      ** 
6d10: 53 65 72 76 65 72 20 61 6e 6e 6f 75 6e 63 65 73  Server announces
6d20: 20 74 68 61 74 20 69 74 20 68 61 73 20 61 20 70   that it has a p
6d30: 61 72 74 69 63 75 6c 61 72 20 66 69 6c 65 2e 20  articular file. 
6d40: 20 49 66 20 74 68 69 73 20 69 73 0a 20 20 20 20   If this is.    
6d50: 20 20 2a 2a 20 6e 6f 74 20 61 20 66 69 6c 65 20    ** not a file 
6d60: 74 68 61 74 20 77 65 20 68 61 76 65 20 61 6e 64  that we have and
6d70: 20 77 65 20 61 72 65 20 70 75 6c 6c 69 6e 67 2c   we are pulling,
6d80: 20 74 68 65 6e 20 63 72 65 61 74 65 20 61 0a 20   then create a. 
6d90: 20 20 20 20 20 2a 2a 20 70 68 61 6e 74 6f 6d 20       ** phantom 
6da0: 74 6f 20 63 61 75 73 65 20 74 68 69 73 20 66 69  to cause this fi
6db0: 6c 65 20 74 6f 20 62 65 20 72 65 71 75 65 73 74  le to be request
6dc0: 65 64 20 6f 6e 20 74 68 65 20 6e 65 78 74 20 63  ed on the next c
6dd0: 79 63 6c 65 2e 0a 20 20 20 20 20 20 2a 2a 20 41  ycle..      ** A
6de0: 6c 77 61 79 73 20 72 65 6d 65 6d 62 65 72 20 74  lways remember t
6df0: 68 61 74 20 74 68 65 20 73 65 72 76 65 72 20 68  hat the server h
6e00: 61 73 20 74 68 69 73 20 66 69 6c 65 20 73 6f 20  as this file so 
6e10: 74 68 61 74 20 77 65 20 64 6f 0a 20 20 20 20 20  that we do.     
6e20: 20 2a 2a 20 6e 6f 74 20 74 72 61 6e 73 6d 69 74   ** not transmit
6e30: 20 69 74 20 62 79 20 61 63 63 69 64 65 6e 74 2e   it by accident.
6e40: 0a 20 20 20 20 20 20 2a 2f 0a 20 20 20 20 20 20  .      */.      
6e50: 69 66 28 20 78 66 65 72 2e 6e 54 6f 6b 65 6e 3d  if( xfer.nToken=
6e60: 3d 32 0a 20 20 20 20 20 20 20 26 26 20 62 6c 6f  =2.       && blo
6e70: 62 5f 65 71 28 26 78 66 65 72 2e 61 54 6f 6b 65  b_eq(&xfer.aToke
6e80: 6e 5b 30 5d 2c 20 22 69 67 6f 74 22 29 0a 20 20  n[0], "igot").  
6e90: 20 20 20 20 20 26 26 20 62 6c 6f 62 5f 69 73 5f       && blob_is_
6ea0: 75 75 69 64 28 26 78 66 65 72 2e 61 54 6f 6b 65  uuid(&xfer.aToke
6eb0: 6e 5b 31 5d 29 0a 20 20 20 20 20 20 29 7b 0a 20  n[1]).      ){. 
6ec0: 20 20 20 20 20 20 20 69 6e 74 20 72 69 64 20 3d         int rid =
6ed0: 20 72 69 64 5f 66 72 6f 6d 5f 75 75 69 64 28 26   rid_from_uuid(&
6ee0: 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 31 5d 2c 20  xfer.aToken[1], 
6ef0: 30 29 3b 0a 20 20 20 20 20 20 20 20 69 66 28 20  0);.        if( 
6f00: 72 69 64 3d 3d 30 20 26 26 20 28 70 75 6c 6c 46  rid==0 && (pullF
6f10: 6c 61 67 20 7c 7c 20 63 6c 6f 6e 65 46 6c 61 67  lag || cloneFlag
6f20: 29 20 29 7b 0a 20 20 20 20 20 20 20 20 20 20 72  ) ){.          r
6f30: 69 64 20 3d 20 63 6f 6e 74 65 6e 74 5f 6e 65 77  id = content_new
6f40: 28 62 6c 6f 62 5f 73 74 72 28 26 78 66 65 72 2e  (blob_str(&xfer.
6f50: 61 54 6f 6b 65 6e 5b 31 5d 29 29 3b 0a 20 20 20  aToken[1]));.   
6f60: 20 20 20 20 20 20 20 69 66 28 20 72 69 64 20 29         if( rid )
6f70: 20 6e 65 77 50 68 61 6e 74 6f 6d 20 3d 20 31 3b   newPhantom = 1;
6f80: 0a 20 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20  .        }.     
6f90: 20 20 20 72 65 6d 6f 74 65 5f 68 61 73 28 72 69     remote_has(ri
6fa0: 64 29 3b 0a 20 20 20 20 20 20 7d 65 6c 73 65 0a  d);.      }else.
6fb0: 20 20 20 20 0a 20 20 20 20 20 20 0a 20 20 20 20      .      .    
6fc0: 20 20 2f 2a 20 20 20 70 75 73 68 20 20 53 45 52    /*   push  SER
6fd0: 56 45 52 43 4f 44 45 20 20 50 52 4f 44 55 43 54  VERCODE  PRODUCT
6fe0: 43 4f 44 45 0a 20 20 20 20 20 20 2a 2a 0a 20 20  CODE.      **.  
6ff0: 20 20 20 20 2a 2a 20 53 68 6f 75 6c 64 20 6f 6e      ** Should on
7000: 6c 79 20 68 61 70 70 65 6e 20 69 6e 20 72 65 73  ly happen in res
7010: 70 6f 6e 73 65 20 74 6f 20 61 20 63 6c 6f 6e 65  ponse to a clone
7020: 2e 20 20 54 68 69 73 20 6d 65 73 73 61 67 65 20  .  This message 
7030: 74 65 6c 6c 73 0a 20 20 20 20 20 20 2a 2a 20 74  tells.      ** t
7040: 68 65 20 63 6c 69 65 6e 74 20 77 68 61 74 20 70  he client what p
7050: 72 6f 64 75 63 74 20 74 6f 20 75 73 65 20 66 6f  roduct to use fo
7060: 72 20 74 68 65 20 6e 65 77 20 64 61 74 61 62 61  r the new databa
7070: 73 65 2e 0a 20 20 20 20 20 20 2a 2f 0a 20 20 20  se..      */.   
7080: 20 20 20 69 66 28 20 62 6c 6f 62 5f 65 71 28 26     if( blob_eq(&
7090: 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 22  xfer.aToken[0],"
70a0: 70 75 73 68 22 29 0a 20 20 20 20 20 20 20 26 26  push").       &&
70b0: 20 78 66 65 72 2e 6e 54 6f 6b 65 6e 3d 3d 33 0a   xfer.nToken==3.
70c0: 20 20 20 20 20 20 20 26 26 20 63 6c 6f 6e 65 46         && cloneF
70d0: 6c 61 67 0a 20 20 20 20 20 20 20 26 26 20 62 6c  lag.       && bl
70e0: 6f 62 5f 69 73 5f 75 75 69 64 28 26 78 66 65 72  ob_is_uuid(&xfer
70f0: 2e 61 54 6f 6b 65 6e 5b 31 5d 29 0a 20 20 20 20  .aToken[1]).    
7100: 20 20 20 26 26 20 62 6c 6f 62 5f 69 73 5f 75 75     && blob_is_uu
7110: 69 64 28 26 78 66 65 72 2e 61 54 6f 6b 65 6e 5b  id(&xfer.aToken[
7120: 32 5d 29 0a 20 20 20 20 20 20 29 7b 0a 20 20 20  2]).      ){.   
7130: 20 20 20 20 20 69 66 28 20 62 6c 6f 62 5f 65 71       if( blob_eq
7140: 5f 73 74 72 28 26 78 66 65 72 2e 61 54 6f 6b 65  _str(&xfer.aToke
7150: 6e 5b 31 5d 2c 20 7a 53 43 6f 64 65 2c 20 2d 31  n[1], zSCode, -1
7160: 29 20 29 7b 0a 20 20 20 20 20 20 20 20 20 20 66  ) ){.          f
7170: 6f 73 73 69 6c 5f 66 61 74 61 6c 28 22 73 65 72  ossil_fatal("ser
7180: 76 65 72 20 6c 6f 6f 70 22 29 3b 0a 20 20 20 20  ver loop");.    
7190: 20 20 20 20 7d 0a 20 20 20 20 20 20 20 20 69 66      }.        if
71a0: 28 20 7a 50 43 6f 64 65 3d 3d 30 20 29 7b 0a 20  ( zPCode==0 ){. 
71b0: 20 20 20 20 20 20 20 20 20 7a 50 43 6f 64 65 20           zPCode 
71c0: 3d 20 6d 70 72 69 6e 74 66 28 22 25 62 22 2c 20  = mprintf("%b", 
71d0: 26 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 32 5d 29  &xfer.aToken[2])
71e0: 3b 0a 20 20 20 20 20 20 20 20 20 20 64 62 5f 73  ;.          db_s
71f0: 65 74 28 22 70 72 6f 6a 65 63 74 2d 63 6f 64 65  et("project-code
7200: 22 2c 20 7a 50 43 6f 64 65 2c 20 30 29 3b 0a 20  ", zPCode, 0);. 
7210: 20 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 20         }.       
7220: 20 62 6c 6f 62 5f 61 70 70 65 6e 64 66 28 26 73   blob_appendf(&s
7230: 65 6e 64 2c 20 22 63 6c 6f 6e 65 5c 6e 22 29 3b  end, "clone\n");
7240: 0a 20 20 20 20 20 20 20 20 6e 43 61 72 64 2b 2b  .        nCard++
7250: 3b 0a 20 20 20 20 20 20 7d 65 6c 73 65 0a 20 20  ;.      }else.  
7260: 20 20 20 20 0a 20 20 20 20 20 20 2f 2a 20 20 20      .      /*   
7270: 63 6f 6e 66 69 67 20 4e 41 4d 45 20 53 49 5a 45  config NAME SIZE
7280: 20 5c 6e 20 43 4f 4e 54 45 4e 54 0a 20 20 20 20   \n CONTENT.    
7290: 20 20 2a 2a 0a 20 20 20 20 20 20 2a 2a 20 52 65    **.      ** Re
72a0: 63 65 69 76 65 20 61 20 63 6f 6e 66 69 67 75 72  ceive a configur
72b0: 61 74 69 6f 6e 20 76 61 6c 75 65 20 66 72 6f 6d  ation value from
72c0: 20 74 68 65 20 73 65 72 76 65 72 2e 0a 20 20 20   the server..   
72d0: 20 20 20 2a 2f 0a 20 20 20 20 20 20 69 66 28 20     */.      if( 
72e0: 62 6c 6f 62 5f 65 71 28 26 78 66 65 72 2e 61 54  blob_eq(&xfer.aT
72f0: 6f 6b 65 6e 5b 30 5d 2c 22 63 6f 6e 66 69 67 22  oken[0],"config"
7300: 29 20 26 26 20 78 66 65 72 2e 6e 54 6f 6b 65 6e  ) && xfer.nToken
7310: 3d 3d 33 0a 20 20 20 20 20 20 20 20 20 20 26 26  ==3.          &&
7320: 20 62 6c 6f 62 5f 69 73 5f 69 6e 74 28 26 78 66   blob_is_int(&xf
7330: 65 72 2e 61 54 6f 6b 65 6e 5b 32 5d 2c 20 26 73  er.aToken[2], &s
7340: 69 7a 65 29 20 29 7b 0a 20 20 20 20 20 20 20 20  ize) ){.        
7350: 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d  const char *zNam
7360: 65 20 3d 20 62 6c 6f 62 5f 73 74 72 28 26 78 66  e = blob_str(&xf
7370: 65 72 2e 61 54 6f 6b 65 6e 5b 31 5d 29 3b 0a 20  er.aToken[1]);. 
7380: 20 20 20 20 20 20 20 42 6c 6f 62 20 63 6f 6e 74         Blob cont
7390: 65 6e 74 3b 0a 20 20 20 20 20 20 20 20 62 6c 6f  ent;.        blo
73a0: 62 5f 7a 65 72 6f 28 26 63 6f 6e 74 65 6e 74 29  b_zero(&content)
73b0: 3b 0a 20 20 20 20 20 20 20 20 62 6c 6f 62 5f 65  ;.        blob_e
73c0: 78 74 72 61 63 74 28 78 66 65 72 2e 70 49 6e 2c  xtract(xfer.pIn,
73d0: 20 73 69 7a 65 2c 20 26 63 6f 6e 74 65 6e 74 29   size, &content)
73e0: 3b 0a 20 20 20 20 20 20 20 20 69 66 28 20 63 6f  ;.        if( co
73f0: 6e 66 69 67 75 72 65 5f 69 73 5f 65 78 70 6f 72  nfigure_is_expor
7400: 74 61 62 6c 65 28 7a 4e 61 6d 65 29 20 26 20 6f  table(zName) & o
7410: 72 69 67 43 6f 6e 66 69 67 4d 61 73 6b 20 29 7b  rigConfigMask ){
7420: 0a 20 20 20 20 20 20 20 20 20 20 64 62 5f 6d 75  .          db_mu
7430: 6c 74 69 5f 65 78 65 63 28 0a 20 20 20 20 20 20  lti_exec(.      
7440: 20 20 20 20 20 20 20 20 22 52 45 50 4c 41 43 45          "REPLACE
7450: 20 49 4e 54 4f 20 63 6f 6e 66 69 67 28 6e 61 6d   INTO config(nam
7460: 65 2c 76 61 6c 75 65 29 20 56 41 4c 55 45 53 28  e,value) VALUES(
7470: 25 51 2c 25 51 29 22 2c 0a 20 20 20 20 20 20 20  %Q,%Q)",.       
7480: 20 20 20 20 20 20 20 7a 4e 61 6d 65 2c 20 62 6c         zName, bl
7490: 6f 62 5f 73 74 72 28 26 63 6f 6e 74 65 6e 74 29  ob_str(&content)
74a0: 0a 20 20 20 20 20 20 20 20 20 20 29 3b 0a 20 20  .          );.  
74b0: 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 20 20        }.        
74c0: 6e 43 61 72 64 2b 2b 3b 0a 20 20 20 20 20 20 20  nCard++;.       
74d0: 20 62 6c 6f 62 5f 72 65 73 65 74 28 26 63 6f 6e   blob_reset(&con
74e0: 74 65 6e 74 29 3b 0a 20 20 20 20 20 20 20 20 62  tent);.        b
74f0: 6c 6f 62 5f 73 65 65 6b 28 78 66 65 72 2e 70 49  lob_seek(xfer.pI
7500: 6e 2c 20 31 2c 20 42 4c 4f 42 5f 53 45 45 4b 5f  n, 1, BLOB_SEEK_
7510: 43 55 52 29 3b 0a 20 20 20 20 20 20 7d 65 6c 73  CUR);.      }els
7520: 65 0a 0a 20 20 20 20 20 20 0a 20 20 20 20 20 20  e..      .      
7530: 2f 2a 20 20 20 20 63 6f 6f 6b 69 65 20 54 45 58  /*    cookie TEX
7540: 54 0a 20 20 20 20 20 20 2a 2a 0a 20 20 20 20 20  T.      **.     
7550: 20 2a 2a 20 54 68 65 20 73 65 72 76 65 72 20 6d   ** The server m
7560: 69 67 68 74 20 69 6e 63 6c 75 64 65 20 61 20 63  ight include a c
7570: 6f 6f 6b 69 65 20 69 6e 20 69 74 73 20 72 65 70  ookie in its rep
7580: 6c 79 2e 20 20 54 68 65 20 63 6c 69 65 6e 74 0a  ly.  The client.
7590: 20 20 20 20 20 20 2a 2a 20 73 68 6f 75 6c 64 20        ** should 
75a0: 72 65 6d 65 6d 62 65 72 20 74 68 69 73 20 63 6f  remember this co
75b0: 6f 6b 69 65 20 61 6e 64 20 73 65 6e 64 20 69 74  okie and send it
75c0: 20 62 61 63 6b 20 74 6f 20 74 68 65 20 73 65 72   back to the ser
75d0: 76 65 72 0a 20 20 20 20 20 20 2a 2a 20 69 6e 20  ver.      ** in 
75e0: 69 74 73 20 6e 65 78 74 20 71 75 65 72 79 2e 0a  its next query..
75f0: 20 20 20 20 20 20 2a 2a 0a 20 20 20 20 20 20 2a        **.      *
7600: 2a 20 45 61 63 68 20 63 6f 6f 6b 69 65 20 72 65  * Each cookie re
7610: 63 65 69 76 65 64 20 6f 76 65 72 77 72 69 74 65  ceived overwrite
7620: 73 20 74 68 65 20 70 72 69 6f 72 20 63 6f 6f 6b  s the prior cook
7630: 69 65 20 66 72 6f 6d 20 74 68 65 0a 20 20 20 20  ie from the.    
7640: 20 20 2a 2a 20 73 61 6d 65 20 73 65 72 76 65 72    ** same server
7650: 2e 0a 20 20 20 20 20 20 2a 2f 0a 20 20 20 20 20  ..      */.     
7660: 20 69 66 28 20 62 6c 6f 62 5f 65 71 28 26 78 66   if( blob_eq(&xf
7670: 65 72 2e 61 54 6f 6b 65 6e 5b 30 5d 2c 20 22 63  er.aToken[0], "c
7680: 6f 6f 6b 69 65 22 29 20 26 26 20 78 66 65 72 2e  ookie") && xfer.
7690: 6e 54 6f 6b 65 6e 3d 3d 32 20 29 7b 0a 20 20 20  nToken==2 ){.   
76a0: 20 20 20 20 20 64 62 5f 73 65 74 28 22 63 6f 6f       db_set("coo
76b0: 6b 69 65 22 2c 20 62 6c 6f 62 5f 73 74 72 28 26  kie", blob_str(&
76c0: 78 66 65 72 2e 61 54 6f 6b 65 6e 5b 31 5d 29 2c  xfer.aToken[1]),
76d0: 20 30 29 3b 0a 20 20 20 20 20 20 7d 65 6c 73 65   0);.      }else
76e0: 0a 0a 20 20 20 20 20 20 2f 2a 20 20 20 6d 65 73  ..      /*   mes
76f0: 73 61 67 65 20 4d 45 53 53 41 47 45 0a 20 20 20  sage MESSAGE.   
7700: 20 20 20 2a 2a 0a 20 20 20 20 20 20 2a 2a 20 50     **.      ** P
7710: 72 69 6e 74 20 61 20 6d 65 73 73 61 67 65 2e 20  rint a message. 
7720: 20 53 69 6d 69 6c 61 72 20 74 6f 20 22 65 72 72   Similar to "err
7730: 6f 72 22 20 62 75 74 20 64 6f 65 73 20 6e 6f 74  or" but does not
7740: 20 73 74 6f 70 20 70 72 6f 63 65 73 73 69 6e 67   stop processing
7750: 0a 20 20 20 20 20 20 2a 2f 20 20 20 20 20 20 20  .      */       
7760: 20 0a 20 20 20 20 20 20 69 66 28 20 62 6c 6f 62   .      if( blob
7770: 5f 65 71 28 26 78 66 65 72 2e 61 54 6f 6b 65 6e  _eq(&xfer.aToken
7780: 5b 30 5d 2c 22 6d 65 73 73 61 67 65 22 29 20 26  [0],"message") &
7790: 26 20 78 66 65 72 2e 6e 54 6f 6b 65 6e 3d 3d 32  & xfer.nToken==2
77a0: 20 29 7b 0a 20 20 20 20 20 20 20 20 63 68 61 72   ){.        char
77b0: 20 2a 7a 4d 73 67 20 3d 20 62 6c 6f 62 5f 74 65   *zMsg = blob_te
77c0: 72 6d 69 6e 61 74 65 28 26 78 66 65 72 2e 61 54  rminate(&xfer.aT
77d0: 6f 6b 65 6e 5b 31 5d 29 3b 0a 20 20 20 20 20 20  oken[1]);.      
77e0: 20 20 64 65 66 6f 73 73 69 6c 69 7a 65 28 7a 4d    defossilize(zM
77f0: 73 67 29 3b 0a 20 20 20 20 20 20 20 20 70 72 69  sg);.        pri
7800: 6e 74 66 28 22 53 65 72 76 65 72 20 73 61 79 73  ntf("Server says
7810: 3a 20 25 73 5c 6e 22 2c 20 7a 4d 73 67 29 3b 0a  : %s\n", zMsg);.
7820: 20 20 20 20 20 20 7d 65 6c 73 65 0a 0a 20 20 20        }else..   
7830: 20 20 20 2f 2a 20 20 20 65 72 72 6f 72 20 4d 45     /*   error ME
7840: 53 53 41 47 45 0a 20 20 20 20 20 20 2a 2a 0a 20  SSAGE.      **. 
7850: 20 20 20 20 20 2a 2a 20 52 65 70 6f 72 74 20 61       ** Report a
7860: 6e 20 65 72 72 6f 72 20 61 6e 64 20 61 62 61 6e  n error and aban
7870: 64 6f 6e 20 74 68 65 20 73 79 6e 63 20 73 65 73  don the sync ses
7880: 73 69 6f 6e 0a 20 20 20 20 20 20 2a 2f 20 20 20  sion.      */   
7890: 20 20 20 20 20 0a 20 20 20 20 20 20 69 66 28 20       .      if( 
78a0: 62 6c 6f 62 5f 65 71 28 26 78 66 65 72 2e 61 54  blob_eq(&xfer.aT
78b0: 6f 6b 65 6e 5b 30 5d 2c 22 65 72 72 6f 72 22 29  oken[0],"error")
78c0: 20 26 26 20 78 66 65 72 2e 6e 54 6f 6b 65 6e 3d   && xfer.nToken=
78d0: 3d 32 20 29 7b 0a 20 20 20 20 20 20 20 20 63 68  =2 ){.        ch
78e0: 61 72 20 2a 7a 4d 73 67 20 3d 20 62 6c 6f 62 5f  ar *zMsg = blob_
78f0: 74 65 72 6d 69 6e 61 74 65 28 26 78 66 65 72 2e  terminate(&xfer.
7900: 61 54 6f 6b 65 6e 5b 31 5d 29 3b 0a 20 20 20 20  aToken[1]);.    
7910: 20 20 20 20 64 65 66 6f 73 73 69 6c 69 7a 65 28      defossilize(
7920: 7a 4d 73 67 29 3b 0a 20 20 20 20 20 20 20 20 62  zMsg);.        b
7930: 6c 6f 62 5f 61 70 70 65 6e 64 66 28 26 78 66 65  lob_appendf(&xfe
7940: 72 2e 65 72 72 2c 20 22 73 65 72 76 65 72 20 73  r.err, "server s
7950: 61 79 73 3a 20 25 73 22 2c 20 7a 4d 73 67 29 3b  ays: %s", zMsg);
7960: 0a 20 20 20 20 20 20 7d 65 6c 73 65 0a 0a 20 20  .      }else..  
7970: 20 20 20 20 2f 2a 20 55 6e 6b 6e 6f 77 6e 20 6d      /* Unknown m
7980: 65 73 73 61 67 65 20 2a 2f 0a 20 20 20 20 20 20  essage */.      
7990: 7b 0a 20 20 20 20 20 20 20 20 62 6c 6f 62 5f 61  {.        blob_a
79a0: 70 70 65 6e 64 66 28 26 78 66 65 72 2e 65 72 72  ppendf(&xfer.err
79b0: 2c 20 22 75 6e 6b 6e 6f 77 6e 20 63 6f 6d 6d 61  , "unknown comma
79c0: 6e 64 3a 20 25 62 22 2c 20 26 78 66 65 72 2e 61  nd: %b", &xfer.a
79d0: 54 6f 6b 65 6e 5b 30 5d 29 3b 0a 20 20 20 20 20  Token[0]);.     
79e0: 20 7d 0a 0a 20 20 20 20 20 20 69 66 28 20 62 6c   }..      if( bl
79f0: 6f 62 5f 73 69 7a 65 28 26 78 66 65 72 2e 65 72  ob_size(&xfer.er
7a00: 72 29 20 29 7b 0a 20 20 20 20 20 20 20 20 66 6f  r) ){.        fo
7a10: 73 73 69 6c 5f 66 61 74 61 6c 28 22 25 62 22 2c  ssil_fatal("%b",
7a20: 20 26 78 66 65 72 2e 65 72 72 29 3b 0a 20 20 20   &xfer.err);.   
7a30: 20 20 20 7d 0a 20 20 20 20 20 20 62 6c 6f 62 61     }.      bloba
7a40: 72 72 61 79 5f 72 65 73 65 74 28 78 66 65 72 2e  rray_reset(xfer.
7a50: 61 54 6f 6b 65 6e 2c 20 78 66 65 72 2e 6e 54 6f  aToken, xfer.nTo
7a60: 6b 65 6e 29 3b 0a 20 20 20 20 20 20 62 6c 6f 62  ken);.      blob
7a70: 5f 72 65 73 65 74 28 26 78 66 65 72 2e 6c 69 6e  _reset(&xfer.lin
7a80: 65 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20 70 72  e);.    }.    pr
7a90: 69 6e 74 66 28 7a 56 61 6c 75 65 46 6f 72 6d 61  intf(zValueForma
7aa0: 74 2c 20 22 52 65 63 65 69 76 65 64 3a 22 2c 0a  t, "Received:",.
7ab0: 20 20 20 20 20 20 20 20 20 20 20 20 62 6c 6f 62              blob
7ac0: 5f 73 69 7a 65 28 26 72 65 63 76 29 2c 20 6e 43  _size(&recv), nC
7ad0: 61 72 64 2c 0a 20 20 20 20 20 20 20 20 20 20 20  ard,.           
7ae0: 20 78 66 65 72 2e 6e 46 69 6c 65 52 63 76 64 2c   xfer.nFileRcvd,
7af0: 20 78 66 65 72 2e 6e 44 65 6c 74 61 52 63 76 64   xfer.nDeltaRcvd
7b00: 20 2b 20 78 66 65 72 2e 6e 44 61 6e 67 6c 69 6e   + xfer.nDanglin
7b10: 67 46 69 6c 65 29 3b 0a 20 20 20 20 62 6c 6f 62  gFile);.    blob
7b20: 5f 72 65 73 65 74 28 26 72 65 63 76 29 3b 0a 20  _reset(&recv);. 
7b30: 20 20 20 6e 43 79 63 6c 65 2b 2b 3b 0a 20 20 20     nCycle++;.   
7b40: 20 67 6f 20 3d 20 30 3b 0a 0a 20 20 20 20 2f 2a   go = 0;..    /*
7b50: 20 49 66 20 77 65 20 72 65 63 65 69 76 65 64 20   If we received 
7b60: 6f 6e 65 20 6f 72 20 6d 6f 72 65 20 66 69 6c 65  one or more file
7b70: 73 20 6f 6e 20 74 68 65 20 70 72 65 76 69 6f 75  s on the previou
7b80: 73 20 65 78 63 68 61 6e 67 65 20 62 75 74 0a 20  s exchange but. 
7b90: 20 20 20 2a 2a 20 74 68 65 72 65 20 61 72 65 20     ** there are 
7ba0: 73 74 69 6c 6c 20 70 68 61 6e 74 6f 6d 73 2c 20  still phantoms, 
7bb0: 74 68 65 6e 20 67 6f 20 61 6e 6f 74 68 65 72 20  then go another 
7bc0: 72 6f 75 6e 64 2e 0a 20 20 20 20 2a 2f 0a 20 20  round..    */.  
7bd0: 20 20 6e 46 69 6c 65 52 65 63 76 20 3d 20 78 66    nFileRecv = xf
7be0: 65 72 2e 6e 46 69 6c 65 52 63 76 64 20 2b 20 78  er.nFileRcvd + x
7bf0: 66 65 72 2e 6e 44 65 6c 74 61 52 63 76 64 20 2b  fer.nDeltaRcvd +
7c00: 20 78 66 65 72 2e 6e 44 61 6e 67 6c 69 6e 67 46   xfer.nDanglingF
7c10: 69 6c 65 3b 0a 20 20 20 20 69 66 28 20 28 6e 46  ile;.    if( (nF
7c20: 69 6c 65 52 65 63 76 3e 30 20 7c 7c 20 6e 65 77  ileRecv>0 || new
7c30: 50 68 61 6e 74 6f 6d 29 20 26 26 20 64 62 5f 65  Phantom) && db_e
7c40: 78 69 73 74 73 28 22 53 45 4c 45 43 54 20 31 20  xists("SELECT 1 
7c50: 46 52 4f 4d 20 70 68 61 6e 74 6f 6d 22 29 20 29  FROM phantom") )
7c60: 7b 0a 20 20 20 20 20 20 67 6f 20 3d 20 31 3b 0a  {.      go = 1;.
7c70: 20 20 20 20 20 20 6d 78 50 68 61 6e 74 6f 6d 52        mxPhantomR
7c80: 65 71 20 3d 20 6e 46 69 6c 65 52 65 63 76 2a 32  eq = nFileRecv*2
7c90: 3b 0a 20 20 20 20 20 20 69 66 28 20 6d 78 50 68  ;.      if( mxPh
7ca0: 61 6e 74 6f 6d 52 65 71 3c 32 30 30 20 29 20 6d  antomReq<200 ) m
7cb0: 78 50 68 61 6e 74 6f 6d 52 65 71 20 3d 20 32 30  xPhantomReq = 20
7cc0: 30 3b 0a 20 20 20 20 7d 0a 20 20 20 20 6e 43 61  0;.    }.    nCa
7cd0: 72 64 20 3d 20 30 3b 0a 20 20 20 20 78 66 65 72  rd = 0;.    xfer
7ce0: 2e 6e 46 69 6c 65 52 63 76 64 20 3d 20 30 3b 0a  .nFileRcvd = 0;.
7cf0: 20 20 20 20 78 66 65 72 2e 6e 44 65 6c 74 61 52      xfer.nDeltaR
7d00: 63 76 64 20 3d 20 30 3b 0a 20 20 20 20 78 66 65  cvd = 0;.    xfe
7d10: 72 2e 6e 44 61 6e 67 6c 69 6e 67 46 69 6c 65 20  r.nDanglingFile 
7d20: 3d 20 30 3b 0a 0a 20 20 20 20 2f 2a 20 49 66 20  = 0;..    /* If 
7d30: 77 65 20 68 61 76 65 20 6f 6e 65 20 6f 72 20 6d  we have one or m
7d40: 6f 72 65 20 66 69 6c 65 73 20 71 75 65 75 65 64  ore files queued
7d50: 20 74 6f 20 73 65 6e 64 2c 20 74 68 65 6e 20 67   to send, then g
7d60: 6f 0a 20 20 20 20 2a 2a 20 61 6e 6f 74 68 65 72  o.    ** another
7d70: 20 72 6f 75 6e 64 20 0a 20 20 20 20 2a 2f 0a 20   round .    */. 
7d80: 20 20 20 69 66 28 20 78 66 65 72 2e 6e 46 69 6c     if( xfer.nFil
7d90: 65 53 65 6e 74 2b 78 66 65 72 2e 6e 44 65 6c 74  eSent+xfer.nDelt
7da0: 61 53 65 6e 74 3e 30 20 29 7b 0a 20 20 20 20 20  aSent>0 ){.     
7db0: 20 67 6f 20 3d 20 31 3b 0a 20 20 20 20 7d 0a 20   go = 1;.    }. 
7dc0: 20 7d 3b 0a 20 20 68 74 74 70 5f 63 6c 6f 73 65   };.  http_close
7dd0: 28 29 3b 0a 20 20 64 62 5f 6d 75 6c 74 69 5f 65  ();.  db_multi_e
7de0: 78 65 63 28 22 44 52 4f 50 20 54 41 42 4c 45 20  xec("DROP TABLE 
7df0: 6f 6e 72 65 6d 6f 74 65 22 29 3b 0a 20 20 64 62  onremote");.  db
7e00: 5f 65 6e 64 5f 74 72 61 6e 73 61 63 74 69 6f 6e  _end_transaction
7e10: 28 30 29 3b 0a 7d 0a                             (0);.}.