Hex Artifact Content
Not logged in

Artifact fdfe66efd6db70448c251b9aa67feee1b1b273f6:

File src/vfile.c part of check-in [573a464cb7] - Complete rework of the xfer mechanism. Compiles but not yet working. by drh on 2007-08-10 00:08:25.

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 50 72 6f 63 65 64 75 72 65  .**.** Procedure
0370: 73 20 66 6f 72 20 6d 61 6e 61 67 69 6e 67 20 74  s for managing t
0380: 68 65 20 56 46 49 4c 45 20 74 61 62 6c 65 2e 0a  he VFILE table..
0390: 2a 2f 0a 23 69 6e 63 6c 75 64 65 20 22 63 6f 6e  */.#include "con
03a0: 66 69 67 2e 68 22 0a 23 69 6e 63 6c 75 64 65 20  fig.h".#include 
03b0: 22 76 66 69 6c 65 2e 68 22 0a 23 69 6e 63 6c 75  "vfile.h".#inclu
03c0: 64 65 20 3c 61 73 73 65 72 74 2e 68 3e 0a 23 69  de <assert.h>.#i
03d0: 6e 63 6c 75 64 65 20 3c 73 79 73 2f 74 79 70 65  nclude <sys/type
03e0: 73 2e 68 3e 0a 23 69 6e 63 6c 75 64 65 20 3c 64  s.h>.#include <d
03f0: 69 72 65 6e 74 2e 68 3e 0a 0a 2f 2a 0a 2a 2a 20  irent.h>../*.** 
0400: 47 69 76 65 6e 20 61 20 55 55 49 44 2c 20 72 65  Given a UUID, re
0410: 74 75 72 6e 20 74 68 65 20 63 6f 72 72 65 73 70  turn the corresp
0420: 6f 6e 64 69 6e 67 20 72 65 63 6f 72 64 20 49 44  onding record ID
0430: 2e 20 20 49 66 20 74 68 65 20 55 55 49 44 0a 2a  .  If the UUID.*
0440: 2a 20 64 6f 65 73 20 6e 6f 74 20 65 78 69 73 74  * does not exist
0450: 2c 20 74 68 65 6e 20 72 65 74 75 72 6e 20 30 2e  , then return 0.
0460: 0a 2a 2a 0a 2a 2a 20 46 6f 72 20 74 68 69 73 20  .**.** For this 
0470: 72 6f 75 74 69 6e 65 2c 20 74 68 65 20 55 55 49  routine, the UUI
0480: 44 20 6d 75 73 74 20 62 65 20 65 78 61 63 74 2e  D must be exact.
0490: 20 20 46 6f 72 20 61 20 6d 61 74 63 68 20 61 67    For a match ag
04a0: 61 69 6e 73 74 0a 2a 2a 20 75 73 65 72 20 69 6e  ainst.** user in
04b0: 70 75 74 20 77 69 74 68 20 6d 69 78 65 64 20 63  put with mixed c
04c0: 61 73 65 2c 20 75 73 65 20 72 65 73 6f 6c 76 65  ase, use resolve
04d0: 5f 75 75 69 64 28 29 2e 0a 2a 2a 0a 2a 2a 20 49  _uuid()..**.** I
04e0: 66 20 74 68 65 20 55 55 49 44 20 69 73 20 6e 6f  f the UUID is no
04f0: 74 20 66 6f 75 6e 64 20 61 6e 64 20 70 68 61 6e  t found and phan
0500: 74 6f 6d 69 7a 65 20 69 73 20 31 2c 20 74 68 65  tomize is 1, the
0510: 6e 20 61 74 74 65 6d 70 74 20 74 6f 20 0a 2a 2a  n attempt to .**
0520: 20 63 72 65 61 74 65 20 61 20 70 68 61 6e 74 6f   create a phanto
0530: 6d 20 72 65 63 6f 72 64 2e 0a 2a 2f 0a 69 6e 74  m record..*/.int
0540: 20 75 75 69 64 5f 74 6f 5f 72 69 64 28 63 6f 6e   uuid_to_rid(con
0550: 73 74 20 63 68 61 72 20 2a 7a 55 75 69 64 2c 20  st char *zUuid, 
0560: 69 6e 74 20 70 68 61 6e 74 6f 6d 69 7a 65 29 7b  int phantomize){
0570: 0a 20 20 69 6e 74 20 72 69 64 2c 20 73 7a 3b 0a  .  int rid, sz;.
0580: 20 20 63 68 61 72 20 7a 5b 55 55 49 44 5f 53 49    char z[UUID_SI
0590: 5a 45 2b 31 5d 3b 0a 20 20 0a 20 20 73 7a 20 3d  ZE+1];.  .  sz =
05a0: 20 73 74 72 6c 65 6e 28 7a 55 75 69 64 29 3b 0a   strlen(zUuid);.
05b0: 20 20 69 66 28 20 73 7a 21 3d 55 55 49 44 5f 53    if( sz!=UUID_S
05c0: 49 5a 45 20 7c 7c 20 21 76 61 6c 69 64 61 74 65  IZE || !validate
05d0: 31 36 28 7a 55 75 69 64 2c 20 73 7a 29 20 29 7b  16(zUuid, sz) ){
05e0: 0a 20 20 20 20 72 65 74 75 72 6e 20 30 3b 0a 20  .    return 0;. 
05f0: 20 7d 0a 20 20 73 74 72 63 70 79 28 7a 2c 20 7a   }.  strcpy(z, z
0600: 55 75 69 64 29 3b 0a 20 20 63 61 6e 6f 6e 69 63  Uuid);.  canonic
0610: 61 6c 31 36 28 7a 2c 20 73 7a 29 3b 0a 20 20 72  al16(z, sz);.  r
0620: 69 64 20 3d 20 64 62 5f 69 6e 74 28 30 2c 20 22  id = db_int(0, "
0630: 53 45 4c 45 43 54 20 72 69 64 20 46 52 4f 4d 20  SELECT rid FROM 
0640: 62 6c 6f 62 20 57 48 45 52 45 20 75 75 69 64 3d  blob WHERE uuid=
0650: 25 51 22 2c 20 7a 29 3b 0a 20 20 69 66 28 20 72  %Q", z);.  if( r
0660: 69 64 3d 3d 30 20 26 26 20 70 68 61 6e 74 6f 6d  id==0 && phantom
0670: 69 7a 65 20 29 7b 0a 20 20 20 20 72 69 64 20 3d  ize ){.    rid =
0680: 20 63 6f 6e 74 65 6e 74 5f 70 75 74 28 30 2c 20   content_put(0, 
0690: 7a 55 75 69 64 2c 20 30 29 3b 0a 20 20 7d 0a 20  zUuid, 0);.  }. 
06a0: 20 72 65 74 75 72 6e 20 72 69 64 3b 0a 7d 0a 0a   return rid;.}..
06b0: 2f 2a 0a 2a 2a 20 42 75 69 6c 64 20 61 20 63 61  /*.** Build a ca
06c0: 74 61 6c 6f 67 20 6f 66 20 61 6c 6c 20 66 69 6c  talog of all fil
06d0: 65 73 20 69 6e 20 61 20 62 61 73 65 6c 69 6e 65  es in a baseline
06e0: 2e 0a 2a 2a 20 57 65 20 73 63 61 6e 20 74 68 65  ..** We scan the
06f0: 20 62 61 73 65 6c 69 6e 65 20 66 69 6c 65 20 66   baseline file f
0700: 6f 72 20 6c 69 6e 65 73 20 6f 66 20 74 68 65 20  or lines of the 
0710: 66 6f 72 6d 3a 0a 2a 2a 0a 2a 2a 20 20 20 20 20  form:.**.**     
0720: 46 20 4e 41 4d 45 20 55 55 49 44 0a 2a 2a 0a 2a  F NAME UUID.**.*
0730: 2a 20 45 61 63 68 20 73 75 63 68 20 6c 69 6e 65  * Each such line
0740: 20 6d 61 6b 65 73 20 61 6e 20 65 6e 74 72 79 20   makes an entry 
0750: 69 6e 20 74 68 65 20 56 46 49 4c 45 20 74 61 62  in the VFILE tab
0760: 6c 65 2e 0a 2a 2f 0a 76 6f 69 64 20 76 66 69 6c  le..*/.void vfil
0770: 65 5f 62 75 69 6c 64 28 69 6e 74 20 76 69 64 2c  e_build(int vid,
0780: 20 42 6c 6f 62 20 2a 70 29 7b 0a 20 20 69 6e 74   Blob *p){.  int
0790: 20 72 69 64 3b 0a 20 20 63 68 61 72 20 2a 7a 4e   rid;.  char *zN
07a0: 61 6d 65 2c 20 2a 7a 55 75 69 64 3b 0a 20 20 53  ame, *zUuid;.  S
07b0: 74 6d 74 20 69 6e 73 3b 0a 20 20 42 6c 6f 62 20  tmt ins;.  Blob 
07c0: 6c 69 6e 65 2c 20 74 6f 6b 65 6e 2c 20 6e 61 6d  line, token, nam
07d0: 65 2c 20 75 75 69 64 3b 0a 20 20 69 6e 74 20 73  e, uuid;.  int s
07e0: 65 65 6e 48 65 61 64 65 72 20 3d 20 30 3b 0a 20  eenHeader = 0;. 
07f0: 20 64 62 5f 62 65 67 69 6e 5f 74 72 61 6e 73 61   db_begin_transa
0800: 63 74 69 6f 6e 28 29 3b 0a 20 20 64 62 5f 6d 75  ction();.  db_mu
0810: 6c 74 69 5f 65 78 65 63 28 22 44 45 4c 45 54 45  lti_exec("DELETE
0820: 20 46 52 4f 4d 20 76 66 69 6c 65 20 57 48 45 52   FROM vfile WHER
0830: 45 20 76 69 64 3d 25 64 22 2c 20 76 69 64 29 3b  E vid=%d", vid);
0840: 0a 20 20 64 62 5f 70 72 65 70 61 72 65 28 26 69  .  db_prepare(&i
0850: 6e 73 2c 0a 20 20 20 20 22 49 4e 53 45 52 54 20  ns,.    "INSERT 
0860: 49 4e 54 4f 20 76 66 69 6c 65 28 76 69 64 2c 72  INTO vfile(vid,r
0870: 69 64 2c 6d 72 69 64 2c 70 61 74 68 6e 61 6d 65  id,mrid,pathname
0880: 29 20 22 0a 20 20 20 20 22 20 56 41 4c 55 45 53  ) ".    " VALUES
0890: 28 3a 76 69 64 2c 3a 69 64 2c 3a 69 64 2c 3a 6e  (:vid,:id,:id,:n
08a0: 61 6d 65 29 22 29 3b 0a 20 20 64 62 5f 62 69 6e  ame)");.  db_bin
08b0: 64 5f 69 6e 74 28 26 69 6e 73 2c 20 22 3a 76 69  d_int(&ins, ":vi
08c0: 64 22 2c 20 76 69 64 29 3b 0a 20 20 77 68 69 6c  d", vid);.  whil
08d0: 65 28 20 62 6c 6f 62 5f 6c 69 6e 65 28 70 2c 20  e( blob_line(p, 
08e0: 26 6c 69 6e 65 29 20 29 7b 0a 20 20 20 20 63 68  &line) ){.    ch
08f0: 61 72 20 2a 7a 20 3d 20 62 6c 6f 62 5f 62 75 66  ar *z = blob_buf
0900: 66 65 72 28 26 6c 69 6e 65 29 3b 0a 20 20 20 20  fer(&line);.    
0910: 69 66 28 20 7a 5b 30 5d 3d 3d 27 2d 27 20 29 7b  if( z[0]=='-' ){
0920: 0a 20 20 20 20 20 20 69 66 28 20 73 65 65 6e 48  .      if( seenH
0930: 65 61 64 65 72 20 29 20 62 72 65 61 6b 3b 0a 20  eader ) break;. 
0940: 20 20 20 20 20 77 68 69 6c 65 28 20 62 6c 6f 62       while( blob
0950: 5f 6c 69 6e 65 28 70 2c 20 26 6c 69 6e 65 29 3e  _line(p, &line)>
0960: 31 20 29 7b 7d 0a 20 20 20 20 20 20 69 66 28 20  1 ){}.      if( 
0970: 62 6c 6f 62 5f 6c 69 6e 65 28 70 2c 20 26 6c 69  blob_line(p, &li
0980: 6e 65 29 3d 3d 30 20 29 20 62 72 65 61 6b 3b 0a  ne)==0 ) break;.
0990: 20 20 20 20 7d 0a 20 20 20 20 73 65 65 6e 48 65      }.    seenHe
09a0: 61 64 65 72 20 3d 20 31 3b 0a 20 20 20 20 69 66  ader = 1;.    if
09b0: 28 20 7a 5b 30 5d 21 3d 27 46 27 20 7c 7c 20 7a  ( z[0]!='F' || z
09c0: 5b 31 5d 21 3d 27 20 27 20 29 20 63 6f 6e 74 69  [1]!=' ' ) conti
09d0: 6e 75 65 3b 0a 20 20 20 20 62 6c 6f 62 5f 74 6f  nue;.    blob_to
09e0: 6b 65 6e 28 26 6c 69 6e 65 2c 20 26 74 6f 6b 65  ken(&line, &toke
09f0: 6e 29 3b 20 20 2f 2a 20 53 6b 69 70 20 74 68 65  n);  /* Skip the
0a00: 20 22 46 22 20 74 6f 6b 65 6e 20 2a 2f 0a 20 20   "F" token */.  
0a10: 20 20 69 66 28 20 62 6c 6f 62 5f 74 6f 6b 65 6e    if( blob_token
0a20: 28 26 6c 69 6e 65 2c 20 26 6e 61 6d 65 29 3d 3d  (&line, &name)==
0a30: 30 20 29 20 62 72 65 61 6b 3b 0a 20 20 20 20 69  0 ) break;.    i
0a40: 66 28 20 62 6c 6f 62 5f 74 6f 6b 65 6e 28 26 6c  f( blob_token(&l
0a50: 69 6e 65 2c 20 26 75 75 69 64 29 3d 3d 30 20 29  ine, &uuid)==0 )
0a60: 20 62 72 65 61 6b 3b 0a 20 20 20 20 7a 4e 61 6d   break;.    zNam
0a70: 65 20 3d 20 62 6c 6f 62 5f 73 74 72 28 26 6e 61  e = blob_str(&na
0a80: 6d 65 29 3b 0a 20 20 20 20 64 65 66 6f 73 73 69  me);.    defossi
0a90: 6c 69 7a 65 28 7a 4e 61 6d 65 29 3b 0a 20 20 20  lize(zName);.   
0aa0: 20 7a 55 75 69 64 20 3d 20 62 6c 6f 62 5f 73 74   zUuid = blob_st
0ab0: 72 28 26 75 75 69 64 29 3b 0a 20 20 20 20 72 69  r(&uuid);.    ri
0ac0: 64 20 3d 20 75 75 69 64 5f 74 6f 5f 72 69 64 28  d = uuid_to_rid(
0ad0: 7a 55 75 69 64 2c 20 30 29 3b 0a 20 20 20 20 69  zUuid, 0);.    i
0ae0: 66 28 20 72 69 64 3e 30 20 26 26 20 66 69 6c 65  f( rid>0 && file
0af0: 5f 69 73 5f 73 69 6d 70 6c 65 5f 70 61 74 68 6e  _is_simple_pathn
0b00: 61 6d 65 28 7a 4e 61 6d 65 29 20 29 7b 0a 20 20  ame(zName) ){.  
0b10: 20 20 20 20 64 62 5f 62 69 6e 64 5f 69 6e 74 28      db_bind_int(
0b20: 26 69 6e 73 2c 20 22 3a 69 64 22 2c 20 72 69 64  &ins, ":id", rid
0b30: 29 3b 0a 20 20 20 20 20 20 64 62 5f 62 69 6e 64  );.      db_bind
0b40: 5f 74 65 78 74 28 26 69 6e 73 2c 20 22 3a 6e 61  _text(&ins, ":na
0b50: 6d 65 22 2c 20 7a 4e 61 6d 65 29 3b 0a 20 20 20  me", zName);.   
0b60: 20 20 20 64 62 5f 73 74 65 70 28 26 69 6e 73 29     db_step(&ins)
0b70: 3b 0a 20 20 20 20 20 20 64 62 5f 72 65 73 65 74  ;.      db_reset
0b80: 28 26 69 6e 73 29 3b 0a 20 20 20 20 7d 0a 20 20  (&ins);.    }.  
0b90: 20 20 62 6c 6f 62 5f 72 65 73 65 74 28 26 6e 61    blob_reset(&na
0ba0: 6d 65 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 72 65  me);.    blob_re
0bb0: 73 65 74 28 26 75 75 69 64 29 3b 0a 20 20 7d 0a  set(&uuid);.  }.
0bc0: 20 20 64 62 5f 66 69 6e 61 6c 69 7a 65 28 26 69    db_finalize(&i
0bd0: 6e 73 29 3b 0a 20 20 64 62 5f 65 6e 64 5f 74 72  ns);.  db_end_tr
0be0: 61 6e 73 61 63 74 69 6f 6e 28 30 29 3b 0a 7d 0a  ansaction(0);.}.
0bf0: 0a 2f 2a 0a 2a 2a 20 43 68 65 63 6b 20 74 68 65  ./*.** Check the
0c00: 20 66 69 6c 65 20 73 69 67 6e 61 74 75 72 65 20   file signature 
0c10: 6f 66 20 74 68 65 20 64 69 73 6b 20 69 6d 61 67  of the disk imag
0c20: 65 20 66 6f 72 20 65 76 65 72 79 20 56 46 49 4c  e for every VFIL
0c30: 45 20 6f 66 20 76 69 64 2e 0a 2a 2a 0a 2a 2a 20  E of vid..**.** 
0c40: 53 65 74 20 74 68 65 20 56 46 49 4c 45 2e 43 48  Set the VFILE.CH
0c50: 4e 47 45 44 20 66 69 65 6c 64 20 6f 6e 20 65 76  NGED field on ev
0c60: 65 72 79 20 66 69 6c 65 20 74 68 61 74 20 68 61  ery file that ha
0c70: 73 20 63 68 61 6e 67 65 64 2e 20 20 41 6c 73 6f  s changed.  Also
0c80: 20 0a 2a 2a 20 73 65 74 20 56 46 49 4c 45 2e 43   .** set VFILE.C
0c90: 48 4e 47 45 44 20 6f 6e 20 65 76 65 72 79 20 66  HNGED on every f
0ca0: 6f 6c 64 65 72 20 74 68 61 74 20 63 6f 6e 74 61  older that conta
0cb0: 69 6e 73 20 61 20 66 69 6c 65 20 6f 72 20 66 6f  ins a file or fo
0cc0: 6c 64 65 72 20 0a 2a 2a 20 74 68 61 74 20 68 61  lder .** that ha
0cd0: 73 20 63 68 61 6e 67 65 64 2e 0a 2a 2a 0a 2a 2a  s changed..**.**
0ce0: 20 49 66 20 56 46 49 4c 45 2e 44 45 4c 45 54 45   If VFILE.DELETE
0cf0: 44 20 69 73 20 6e 75 6c 6c 20 6f 72 20 69 66 20  D is null or if 
0d00: 56 46 49 4c 45 2e 52 49 44 20 69 73 20 7a 65 72  VFILE.RID is zer
0d10: 6f 2c 20 74 68 65 6e 20 77 65 20 63 61 6e 20 61  o, then we can a
0d20: 73 73 75 6d 65 0a 2a 2a 20 74 68 65 20 66 69 6c  ssume.** the fil
0d30: 65 20 68 61 73 20 63 68 61 6e 67 65 64 20 77 69  e has changed wi
0d40: 74 68 6f 75 74 20 68 61 76 69 6e 67 20 74 68 65  thout having the
0d50: 20 63 68 65 63 6b 20 74 68 65 20 6f 6e 2d 64 69   check the on-di
0d60: 73 6b 20 69 6d 61 67 65 2e 0a 2a 2f 0a 76 6f 69  sk image..*/.voi
0d70: 64 20 76 66 69 6c 65 5f 63 68 65 63 6b 5f 73 69  d vfile_check_si
0d80: 67 6e 61 74 75 72 65 28 69 6e 74 20 76 69 64 29  gnature(int vid)
0d90: 7b 0a 20 20 53 74 6d 74 20 71 3b 0a 20 20 42 6c  {.  Stmt q;.  Bl
0da0: 6f 62 20 66 69 6c 65 43 6b 73 75 6d 2c 20 6f 72  ob fileCksum, or
0db0: 69 67 43 6b 73 75 6d 3b 0a 0a 20 20 64 62 5f 62  igCksum;..  db_b
0dc0: 65 67 69 6e 5f 74 72 61 6e 73 61 63 74 69 6f 6e  egin_transaction
0dd0: 28 29 3b 0a 20 20 64 62 5f 70 72 65 70 61 72 65  ();.  db_prepare
0de0: 28 26 71 2c 20 22 53 45 4c 45 43 54 20 69 64 2c  (&q, "SELECT id,
0df0: 20 25 51 20 7c 7c 20 70 61 74 68 6e 61 6d 65 2c   %Q || pathname,
0e00: 22 0a 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ".              
0e10: 20 20 20 22 20 20 20 20 20 20 20 76 66 69 6c 65     "       vfile
0e20: 2e 6d 72 69 64 2c 20 64 65 6c 65 74 65 64 2c 20  .mrid, deleted, 
0e30: 63 68 6e 67 65 64 2c 20 75 75 69 64 22 0a 20 20  chnged, uuid".  
0e40: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 22                 "
0e50: 20 20 46 52 4f 4d 20 76 66 69 6c 65 20 4c 45 46    FROM vfile LEF
0e60: 54 20 4a 4f 49 4e 20 62 6c 6f 62 20 4f 4e 20 76  T JOIN blob ON v
0e70: 66 69 6c 65 2e 6d 72 69 64 3d 62 6c 6f 62 2e 72  file.mrid=blob.r
0e80: 69 64 22 0a 20 20 20 20 20 20 20 20 20 20 20 20  id".            
0e90: 20 20 20 20 20 22 20 57 48 45 52 45 20 76 69 64       " WHERE vid
0ea0: 3d 25 64 20 22 2c 20 67 2e 7a 4c 6f 63 61 6c 52  =%d ", g.zLocalR
0eb0: 6f 6f 74 2c 20 76 69 64 29 3b 0a 20 20 77 68 69  oot, vid);.  whi
0ec0: 6c 65 28 20 64 62 5f 73 74 65 70 28 26 71 29 3d  le( db_step(&q)=
0ed0: 3d 53 51 4c 49 54 45 5f 52 4f 57 20 29 7b 0a 20  =SQLITE_ROW ){. 
0ee0: 20 20 20 69 6e 74 20 69 64 2c 20 72 69 64 2c 20     int id, rid, 
0ef0: 69 73 44 65 6c 65 74 65 64 3b 0a 20 20 20 20 63  isDeleted;.    c
0f00: 6f 6e 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65  onst char *zName
0f10: 3b 0a 20 20 20 20 69 6e 74 20 63 68 6e 67 65 64  ;.    int chnged
0f20: 20 3d 20 30 3b 0a 20 20 20 20 69 6e 74 20 6f 6c   = 0;.    int ol
0f30: 64 43 68 6e 67 65 64 3b 0a 0a 20 20 20 20 69 64  dChnged;..    id
0f40: 20 3d 20 64 62 5f 63 6f 6c 75 6d 6e 5f 69 6e 74   = db_column_int
0f50: 28 26 71 2c 20 30 29 3b 0a 20 20 20 20 7a 4e 61  (&q, 0);.    zNa
0f60: 6d 65 20 3d 20 64 62 5f 63 6f 6c 75 6d 6e 5f 74  me = db_column_t
0f70: 65 78 74 28 26 71 2c 20 31 29 3b 0a 20 20 20 20  ext(&q, 1);.    
0f80: 72 69 64 20 3d 20 64 62 5f 63 6f 6c 75 6d 6e 5f  rid = db_column_
0f90: 69 6e 74 28 26 71 2c 20 32 29 3b 0a 20 20 20 20  int(&q, 2);.    
0fa0: 69 73 44 65 6c 65 74 65 64 20 3d 20 64 62 5f 63  isDeleted = db_c
0fb0: 6f 6c 75 6d 6e 5f 69 6e 74 28 26 71 2c 20 33 29  olumn_int(&q, 3)
0fc0: 3b 0a 20 20 20 20 6f 6c 64 43 68 6e 67 65 64 20  ;.    oldChnged 
0fd0: 3d 20 64 62 5f 63 6f 6c 75 6d 6e 5f 69 6e 74 28  = db_column_int(
0fe0: 26 71 2c 20 34 29 3b 0a 20 20 20 20 69 66 28 20  &q, 4);.    if( 
0ff0: 6f 6c 64 43 68 6e 67 65 64 3e 3d 32 20 29 7b 0a  oldChnged>=2 ){.
1000: 20 20 20 20 20 20 63 68 6e 67 65 64 20 3d 20 6f        chnged = o
1010: 6c 64 43 68 6e 67 65 64 3b 0a 20 20 20 20 7d 65  ldChnged;.    }e
1020: 6c 73 65 20 69 66 28 20 69 73 44 65 6c 65 74 65  lse if( isDelete
1030: 64 20 7c 7c 20 72 69 64 3d 3d 30 20 29 7b 0a 20  d || rid==0 ){. 
1040: 20 20 20 20 20 63 68 6e 67 65 64 20 3d 20 31 3b       chnged = 1;
1050: 0a 20 20 20 20 7d 0a 20 20 20 20 69 66 28 20 63  .    }.    if( c
1060: 68 6e 67 65 64 21 3d 31 20 29 7b 0a 20 20 20 20  hnged!=1 ){.    
1070: 20 20 64 62 5f 65 70 68 65 6d 65 72 61 6c 5f 62    db_ephemeral_b
1080: 6c 6f 62 28 26 71 2c 20 35 2c 20 26 6f 72 69 67  lob(&q, 5, &orig
1090: 43 6b 73 75 6d 29 3b 0a 20 20 20 20 20 20 69 66  Cksum);.      if
10a0: 28 20 73 68 61 31 73 75 6d 5f 66 69 6c 65 28 7a  ( sha1sum_file(z
10b0: 4e 61 6d 65 2c 20 26 66 69 6c 65 43 6b 73 75 6d  Name, &fileCksum
10c0: 29 20 29 7b 0a 20 20 20 20 20 20 20 20 62 6c 6f  ) ){.        blo
10d0: 62 5f 7a 65 72 6f 28 26 66 69 6c 65 43 6b 73 75  b_zero(&fileCksu
10e0: 6d 29 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20  m);.      }.    
10f0: 20 20 69 66 28 20 62 6c 6f 62 5f 63 6f 6d 70 61    if( blob_compa
1100: 72 65 28 26 66 69 6c 65 43 6b 73 75 6d 2c 20 26  re(&fileCksum, &
1110: 6f 72 69 67 43 6b 73 75 6d 29 20 29 7b 0a 20 20  origCksum) ){.  
1120: 20 20 20 20 20 20 63 68 6e 67 65 64 20 3d 20 31        chnged = 1
1130: 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20  ;.      }.      
1140: 62 6c 6f 62 5f 72 65 73 65 74 28 26 6f 72 69 67  blob_reset(&orig
1150: 43 6b 73 75 6d 29 3b 0a 20 20 20 20 20 20 62 6c  Cksum);.      bl
1160: 6f 62 5f 72 65 73 65 74 28 26 66 69 6c 65 43 6b  ob_reset(&fileCk
1170: 73 75 6d 29 3b 0a 20 20 20 20 7d 0a 20 20 20 20  sum);.    }.    
1180: 69 66 28 20 63 68 6e 67 65 64 21 3d 6f 6c 64 43  if( chnged!=oldC
1190: 68 6e 67 65 64 20 29 7b 0a 20 20 20 20 20 20 64  hnged ){.      d
11a0: 62 5f 6d 75 6c 74 69 5f 65 78 65 63 28 22 55 50  b_multi_exec("UP
11b0: 44 41 54 45 20 76 66 69 6c 65 20 53 45 54 20 63  DATE vfile SET c
11c0: 68 6e 67 65 64 3d 25 64 20 57 48 45 52 45 20 69  hnged=%d WHERE i
11d0: 64 3d 25 64 22 2c 20 63 68 6e 67 65 64 2c 20 69  d=%d", chnged, i
11e0: 64 29 3b 0a 20 20 20 20 7d 0a 20 20 7d 0a 20 20  d);.    }.  }.  
11f0: 64 62 5f 66 69 6e 61 6c 69 7a 65 28 26 71 29 3b  db_finalize(&q);
1200: 0a 20 20 64 62 5f 65 6e 64 5f 74 72 61 6e 73 61  .  db_end_transa
1210: 63 74 69 6f 6e 28 30 29 3b 0a 7d 0a 0a 2f 2a 0a  ction(0);.}../*.
1220: 2a 2a 20 57 72 69 74 65 20 61 6c 6c 20 66 69 6c  ** Write all fil
1230: 65 73 20 66 72 6f 6d 20 76 69 64 20 74 6f 20 74  es from vid to t
1240: 68 65 20 64 69 73 6b 2e 20 20 4f 72 20 69 66 20  he disk.  Or if 
1250: 76 69 64 3d 3d 30 20 61 6e 64 20 69 64 21 3d 30  vid==0 and id!=0
1260: 0a 2a 2a 20 77 72 69 74 65 20 6a 75 73 74 20 74  .** write just t
1270: 68 65 20 73 70 65 63 69 66 69 63 20 66 69 6c 65  he specific file
1280: 20 77 68 65 72 65 20 56 46 49 4c 45 2e 49 44 3d   where VFILE.ID=
1290: 69 64 2e 0a 2a 2f 0a 76 6f 69 64 20 76 66 69 6c  id..*/.void vfil
12a0: 65 5f 74 6f 5f 64 69 73 6b 28 69 6e 74 20 76 69  e_to_disk(int vi
12b0: 64 2c 20 69 6e 74 20 69 64 2c 20 69 6e 74 20 76  d, int id, int v
12c0: 65 72 62 6f 73 65 29 7b 0a 20 20 53 74 6d 74 20  erbose){.  Stmt 
12d0: 71 3b 0a 20 20 42 6c 6f 62 20 63 6f 6e 74 65 6e  q;.  Blob conten
12e0: 74 3b 0a 20 20 69 6e 74 20 6e 52 65 70 6f 73 20  t;.  int nRepos 
12f0: 3d 20 73 74 72 6c 65 6e 28 67 2e 7a 4c 6f 63 61  = strlen(g.zLoca
1300: 6c 52 6f 6f 74 29 3b 0a 0a 20 20 69 66 28 20 76  lRoot);..  if( v
1310: 69 64 3e 30 20 26 26 20 69 64 3d 3d 30 20 29 7b  id>0 && id==0 ){
1320: 0a 20 20 20 20 64 62 5f 70 72 65 70 61 72 65 28  .    db_prepare(
1330: 26 71 2c 20 22 53 45 4c 45 43 54 20 69 64 2c 20  &q, "SELECT id, 
1340: 25 51 20 7c 7c 20 70 61 74 68 6e 61 6d 65 2c 20  %Q || pathname, 
1350: 6d 72 69 64 22 0a 20 20 20 20 20 20 20 20 20 20  mrid".          
1360: 20 20 20 20 20 20 20 20 20 22 20 20 46 52 4f 4d           "  FROM
1370: 20 76 66 69 6c 65 22 0a 20 20 20 20 20 20 20 20   vfile".        
1380: 20 20 20 20 20 20 20 20 20 20 20 22 20 57 48 45             " WHE
1390: 52 45 20 76 69 64 3d 25 64 20 41 4e 44 20 6d 72  RE vid=%d AND mr
13a0: 69 64 3e 30 22 2c 0a 20 20 20 20 20 20 20 20 20  id>0",.         
13b0: 20 20 20 20 20 20 20 20 20 20 67 2e 7a 4c 6f 63            g.zLoc
13c0: 61 6c 52 6f 6f 74 2c 20 76 69 64 29 3b 0a 20 20  alRoot, vid);.  
13d0: 7d 65 6c 73 65 7b 0a 20 20 20 20 61 73 73 65 72  }else{.    asser
13e0: 74 28 20 76 69 64 3d 3d 30 20 26 26 20 69 64 3e  t( vid==0 && id>
13f0: 30 20 29 3b 0a 20 20 20 20 64 62 5f 70 72 65 70  0 );.    db_prep
1400: 61 72 65 28 26 71 2c 20 22 53 45 4c 45 43 54 20  are(&q, "SELECT 
1410: 69 64 2c 20 25 51 20 7c 7c 20 70 61 74 68 6e 61  id, %Q || pathna
1420: 6d 65 2c 20 6d 72 69 64 22 0a 20 20 20 20 20 20  me, mrid".      
1430: 20 20 20 20 20 20 20 20 20 20 20 20 20 22 20 20               "  
1440: 46 52 4f 4d 20 76 66 69 6c 65 22 0a 20 20 20 20  FROM vfile".    
1450: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 22                 "
1460: 20 57 48 45 52 45 20 69 64 3d 25 64 20 41 4e 44   WHERE id=%d AND
1470: 20 6d 72 69 64 3e 30 22 2c 0a 20 20 20 20 20 20   mrid>0",.      
1480: 20 20 20 20 20 20 20 20 20 20 20 20 20 67 2e 7a               g.z
1490: 4c 6f 63 61 6c 52 6f 6f 74 2c 20 69 64 29 3b 0a  LocalRoot, id);.
14a0: 20 20 7d 0a 20 20 77 68 69 6c 65 28 20 64 62 5f    }.  while( db_
14b0: 73 74 65 70 28 26 71 29 3d 3d 53 51 4c 49 54 45  step(&q)==SQLITE
14c0: 5f 52 4f 57 20 29 7b 0a 20 20 20 20 69 6e 74 20  _ROW ){.    int 
14d0: 69 64 2c 20 72 69 64 3b 0a 20 20 20 20 63 6f 6e  id, rid;.    con
14e0: 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65 3b 0a  st char *zName;.
14f0: 0a 20 20 20 20 69 64 20 3d 20 64 62 5f 63 6f 6c  .    id = db_col
1500: 75 6d 6e 5f 69 6e 74 28 26 71 2c 20 30 29 3b 0a  umn_int(&q, 0);.
1510: 20 20 20 20 7a 4e 61 6d 65 20 3d 20 64 62 5f 63      zName = db_c
1520: 6f 6c 75 6d 6e 5f 74 65 78 74 28 26 71 2c 20 31  olumn_text(&q, 1
1530: 29 3b 0a 20 20 20 20 72 69 64 20 3d 20 64 62 5f  );.    rid = db_
1540: 63 6f 6c 75 6d 6e 5f 69 6e 74 28 26 71 2c 20 32  column_int(&q, 2
1550: 29 3b 0a 20 20 20 20 63 6f 6e 74 65 6e 74 5f 67  );.    content_g
1560: 65 74 28 72 69 64 2c 20 26 63 6f 6e 74 65 6e 74  et(rid, &content
1570: 29 3b 0a 20 20 20 20 69 66 28 20 76 65 72 62 6f  );.    if( verbo
1580: 73 65 20 29 20 70 72 69 6e 74 66 28 22 25 73 5c  se ) printf("%s\
1590: 6e 22 2c 20 26 7a 4e 61 6d 65 5b 6e 52 65 70 6f  n", &zName[nRepo
15a0: 73 5d 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 77 72  s]);.    blob_wr
15b0: 69 74 65 5f 74 6f 5f 66 69 6c 65 28 26 63 6f 6e  ite_to_file(&con
15c0: 74 65 6e 74 2c 20 7a 4e 61 6d 65 29 3b 0a 20 20  tent, zName);.  
15d0: 7d 0a 20 20 64 62 5f 66 69 6e 61 6c 69 7a 65 28  }.  db_finalize(
15e0: 26 71 29 3b 0a 7d 0a 0a 0a 2f 2a 0a 2a 2a 20 44  &q);.}.../*.** D
15f0: 65 6c 65 74 65 20 66 72 6f 6d 20 74 68 65 20 64  elete from the d
1600: 69 73 6b 20 65 76 65 72 79 20 66 69 6c 65 20 69  isk every file i
1610: 6e 20 56 46 49 4c 45 20 76 69 64 2e 0a 2a 2f 0a  n VFILE vid..*/.
1620: 76 6f 69 64 20 76 66 69 6c 65 5f 75 6e 6c 69 6e  void vfile_unlin
1630: 6b 28 69 6e 74 20 76 69 64 29 7b 0a 20 20 53 74  k(int vid){.  St
1640: 6d 74 20 71 3b 0a 20 20 64 62 5f 70 72 65 70 61  mt q;.  db_prepa
1650: 72 65 28 26 71 2c 20 22 53 45 4c 45 43 54 20 25  re(&q, "SELECT %
1660: 51 20 7c 7c 20 70 61 74 68 6e 61 6d 65 20 46 52  Q || pathname FR
1670: 4f 4d 20 76 66 69 6c 65 22 0a 20 20 20 20 20 20  OM vfile".      
1680: 20 20 20 20 20 20 20 20 20 20 20 22 20 57 48 45             " WHE
1690: 52 45 20 76 69 64 3d 25 64 20 41 4e 44 20 6d 72  RE vid=%d AND mr
16a0: 69 64 3e 30 22 2c 20 67 2e 7a 4c 6f 63 61 6c 52  id>0", g.zLocalR
16b0: 6f 6f 74 2c 20 76 69 64 29 3b 0a 20 20 77 68 69  oot, vid);.  whi
16c0: 6c 65 28 20 64 62 5f 73 74 65 70 28 26 71 29 3d  le( db_step(&q)=
16d0: 3d 53 51 4c 49 54 45 5f 52 4f 57 20 29 7b 0a 20  =SQLITE_ROW ){. 
16e0: 20 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a     const char *z
16f0: 4e 61 6d 65 3b 0a 0a 20 20 20 20 7a 4e 61 6d 65  Name;..    zName
1700: 20 3d 20 64 62 5f 63 6f 6c 75 6d 6e 5f 74 65 78   = db_column_tex
1710: 74 28 26 71 2c 20 30 29 3b 0a 20 20 20 20 75 6e  t(&q, 0);.    un
1720: 6c 69 6e 6b 28 7a 4e 61 6d 65 29 3b 0a 20 20 7d  link(zName);.  }
1730: 0a 20 20 64 62 5f 66 69 6e 61 6c 69 7a 65 28 26  .  db_finalize(&
1740: 71 29 3b 0a 7d 0a 0a 2f 2a 0a 2a 2a 20 4c 6f 61  q);.}../*.** Loa
1750: 64 20 69 6e 74 6f 20 74 61 62 6c 65 20 53 46 49  d into table SFI
1760: 4c 45 20 74 68 65 20 6e 61 6d 65 20 6f 66 20 65  LE the name of e
1770: 76 65 72 79 20 6f 72 64 69 6e 61 72 79 20 66 69  very ordinary fi
1780: 6c 65 20 69 6e 0a 2a 2a 20 74 68 65 20 64 69 72  le in.** the dir
1790: 65 63 74 6f 72 79 20 70 50 61 74 68 2e 20 20 53  ectory pPath.  S
17a0: 75 62 64 69 72 65 63 74 6f 72 69 65 73 20 61 72  ubdirectories ar
17b0: 65 20 73 63 61 6e 6e 65 64 20 72 65 63 75 72 73  e scanned recurs
17c0: 69 76 65 6c 79 2e 0a 2a 2a 20 4f 6d 69 74 20 66  ively..** Omit f
17d0: 69 6c 65 73 20 6e 61 6d 65 64 20 69 6e 20 56 46  iles named in VF
17e0: 49 4c 45 2e 76 69 64 0a 2a 2f 0a 76 6f 69 64 20  ILE.vid.*/.void 
17f0: 76 66 69 6c 65 5f 73 63 61 6e 28 69 6e 74 20 76  vfile_scan(int v
1800: 69 64 2c 20 42 6c 6f 62 20 2a 70 50 61 74 68 29  id, Blob *pPath)
1810: 7b 0a 20 20 44 49 52 20 2a 64 3b 0a 20 20 69 6e  {.  DIR *d;.  in
1820: 74 20 6f 72 69 67 53 69 7a 65 3b 0a 20 20 63 6f  t origSize;.  co
1830: 6e 73 74 20 63 68 61 72 20 2a 7a 44 69 72 3b 0a  nst char *zDir;.
1840: 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 46    const char *zF
1850: 6f 72 6d 61 74 3b 0a 20 20 73 74 72 75 63 74 20  ormat;.  struct 
1860: 64 69 72 65 6e 74 20 2a 70 45 6e 74 72 79 3b 0a  dirent *pEntry;.
1870: 20 20 73 74 61 74 69 63 20 63 6f 6e 73 74 20 63    static const c
1880: 68 61 72 20 2a 7a 53 71 6c 20 3d 20 22 53 45 4c  har *zSql = "SEL
1890: 45 43 54 20 31 20 46 52 4f 4d 20 76 66 69 6c 65  ECT 1 FROM vfile
18a0: 20 22 0a 20 20 20 20 20 20 20 20 20 20 20 20 20   ".             
18b0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 22                 "
18c0: 20 57 48 45 52 45 20 70 61 74 68 6e 61 6d 65 3d   WHERE pathname=
18d0: 25 42 20 41 4e 44 20 4e 4f 54 20 64 65 6c 65 74  %B AND NOT delet
18e0: 65 64 22 3b 0a 0a 20 20 6f 72 69 67 53 69 7a 65  ed";..  origSize
18f0: 20 3d 20 62 6c 6f 62 5f 73 69 7a 65 28 70 50 61   = blob_size(pPa
1900: 74 68 29 3b 0a 20 20 7a 44 69 72 20 3d 20 62 6c  th);.  zDir = bl
1910: 6f 62 5f 73 74 72 28 70 50 61 74 68 29 3b 0a 20  ob_str(pPath);. 
1920: 20 69 66 28 20 7a 44 69 72 5b 30 5d 3d 3d 30 20   if( zDir[0]==0 
1930: 29 7b 0a 20 20 20 20 20 7a 44 69 72 20 3d 20 22  ){.     zDir = "
1940: 2e 22 3b 0a 20 20 20 20 20 7a 46 6f 72 6d 61 74  .";.     zFormat
1950: 20 3d 20 22 25 73 22 3b 0a 20 20 7d 65 6c 73 65   = "%s";.  }else
1960: 7b 0a 20 20 20 20 20 7a 46 6f 72 6d 61 74 20 3d  {.     zFormat =
1970: 20 22 2f 25 73 22 3b 0a 20 20 7d 0a 20 20 64 20   "/%s";.  }.  d 
1980: 3d 20 6f 70 65 6e 64 69 72 28 7a 44 69 72 29 3b  = opendir(zDir);
1990: 0a 20 20 69 66 28 20 64 20 29 7b 0a 20 20 20 20  .  if( d ){.    
19a0: 77 68 69 6c 65 28 20 28 70 45 6e 74 72 79 3d 72  while( (pEntry=r
19b0: 65 61 64 64 69 72 28 64 29 29 21 3d 30 20 29 7b  eaddir(d))!=0 ){
19c0: 0a 20 20 20 20 20 20 63 68 61 72 20 2a 7a 50 61  .      char *zPa
19d0: 74 68 3b 0a 20 20 20 20 20 20 69 66 28 20 70 45  th;.      if( pE
19e0: 6e 74 72 79 2d 3e 64 5f 6e 61 6d 65 5b 30 5d 3d  ntry->d_name[0]=
19f0: 3d 27 2e 27 20 29 20 63 6f 6e 74 69 6e 75 65 3b  ='.' ) continue;
1a00: 0a 20 20 20 20 20 20 62 6c 6f 62 5f 61 70 70 65  .      blob_appe
1a10: 6e 64 66 28 70 50 61 74 68 2c 20 7a 46 6f 72 6d  ndf(pPath, zForm
1a20: 61 74 2c 20 70 45 6e 74 72 79 2d 3e 64 5f 6e 61  at, pEntry->d_na
1a30: 6d 65 29 3b 0a 20 20 20 20 20 20 7a 50 61 74 68  me);.      zPath
1a40: 20 3d 20 62 6c 6f 62 5f 73 74 72 28 70 50 61 74   = blob_str(pPat
1a50: 68 29 3b 0a 20 20 20 20 20 20 69 66 28 20 66 69  h);.      if( fi
1a60: 6c 65 5f 69 73 64 69 72 28 7a 50 61 74 68 29 3d  le_isdir(zPath)=
1a70: 3d 31 20 29 7b 0a 20 20 20 20 20 20 20 20 76 66  =1 ){.        vf
1a80: 69 6c 65 5f 73 63 61 6e 28 76 69 64 2c 20 70 50  ile_scan(vid, pP
1a90: 61 74 68 29 3b 0a 20 20 20 20 20 20 7d 65 6c 73  ath);.      }els
1aa0: 65 20 69 66 28 20 66 69 6c 65 5f 69 73 66 69 6c  e if( file_isfil
1ab0: 65 28 7a 50 61 74 68 29 20 26 26 20 21 64 62 5f  e(zPath) && !db_
1ac0: 65 78 69 73 74 73 28 7a 53 71 6c 2c 70 50 61 74  exists(zSql,pPat
1ad0: 68 29 20 29 7b 0a 20 20 20 20 20 20 20 20 64 62  h) ){.        db
1ae0: 5f 6d 75 6c 74 69 5f 65 78 65 63 28 22 49 4e 53  _multi_exec("INS
1af0: 45 52 54 20 49 4e 54 4f 20 73 66 69 6c 65 20 56  ERT INTO sfile V
1b00: 41 4c 55 45 53 28 25 42 29 22 2c 20 70 50 61 74  ALUES(%B)", pPat
1b10: 68 29 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20  h);.      }.    
1b20: 20 20 62 6c 6f 62 5f 72 65 73 69 7a 65 28 70 50    blob_resize(pP
1b30: 61 74 68 2c 20 6f 72 69 67 53 69 7a 65 29 3b 0a  ath, origSize);.
1b40: 20 20 20 20 7d 0a 20 20 7d 0a 20 20 63 6c 6f 73      }.  }.  clos
1b50: 65 64 69 72 28 64 29 3b 0a 7d 0a 0a 2f 2a 0a 2a  edir(d);.}../*.*
1b60: 2a 20 43 6f 6d 70 75 74 65 20 61 6e 20 61 67 67  * Compute an agg
1b70: 72 65 67 61 74 65 20 4d 44 35 20 63 68 65 63 6b  regate MD5 check
1b80: 73 75 6d 20 6f 76 65 72 20 74 68 65 20 64 69 73  sum over the dis
1b90: 6b 20 69 6d 61 67 65 20 6f 66 20 65 76 65 72 79  k image of every
1ba0: 0a 2a 2a 20 66 69 6c 65 20 69 6e 20 76 69 64 2e  .** file in vid.
1bb0: 20 20 54 68 65 20 66 69 6c 65 20 6e 61 6d 65 73    The file names
1bc0: 20 61 72 65 20 70 61 72 74 20 6f 66 20 74 68 65   are part of the
1bd0: 20 63 68 65 63 6b 73 75 6d 2e 0a 2a 2a 0a 2a 2a   checksum..**.**
1be0: 20 54 68 69 73 20 66 75 6e 63 74 69 6f 6e 20 6f   This function o
1bf0: 70 65 72 61 74 65 73 20 64 69 66 66 65 72 65 6e  perates differen
1c00: 74 6c 79 20 69 66 20 74 68 65 20 47 6c 6f 62 61  tly if the Globa
1c10: 6c 2e 61 43 6f 6d 6d 69 74 46 69 6c 65 0a 2a 2a  l.aCommitFile.**
1c20: 20 76 61 72 69 61 62 6c 65 20 69 73 20 6e 6f 74   variable is not
1c30: 20 4e 55 4c 4c 2e 20 49 6e 20 74 68 61 74 20 63   NULL. In that c
1c40: 61 73 65 2c 20 74 68 65 20 64 69 73 6b 20 69 6d  ase, the disk im
1c50: 61 67 65 20 69 73 20 75 73 65 64 20 66 6f 72 0a  age is used for.
1c60: 2a 2a 20 65 61 63 68 20 66 69 6c 65 20 69 6e 20  ** each file in 
1c70: 61 43 6f 6d 6d 69 74 46 69 6c 65 5b 5d 20 61 6e  aCommitFile[] an
1c80: 64 20 74 68 65 20 72 65 70 6f 73 69 74 6f 72 79  d the repository
1c90: 20 69 6d 61 67 65 20 28 73 65 65 0a 2a 2a 20 76   image (see.** v
1ca0: 66 69 6c 65 5f 61 67 67 72 65 67 61 74 65 5f 63  file_aggregate_c
1cb0: 68 65 63 6b 73 75 6d 5f 72 65 70 6f 73 69 74 6f  hecksum_reposito
1cc0: 72 79 28 29 20 69 73 20 75 73 65 64 20 66 6f 72  ry() is used for
1cd0: 20 61 6c 6c 20 6f 74 68 65 72 73 29 2e 0a 2a 2a   all others)..**
1ce0: 20 4e 65 77 6c 79 20 61 64 64 65 64 20 66 69 6c   Newly added fil
1cf0: 65 73 20 74 68 61 74 20 61 72 65 20 6e 6f 74 20  es that are not 
1d00: 63 6f 6e 74 61 69 6e 65 64 20 69 6e 20 74 68 65  contained in the
1d10: 20 72 65 70 6f 73 69 74 6f 72 79 20 61 72 65 0a   repository are.
1d20: 2a 2a 20 6f 6d 69 74 74 65 64 20 66 72 6f 6d 20  ** omitted from 
1d30: 74 68 65 20 63 68 65 63 6b 73 75 6d 20 69 66 20  the checksum if 
1d40: 74 68 65 79 20 61 72 65 20 6e 6f 74 20 69 6e 20  they are not in 
1d50: 47 6c 6f 62 61 6c 2e 61 43 6f 6d 6d 69 74 46 69  Global.aCommitFi
1d60: 6c 65 2e 0a 2a 2a 0a 2a 2a 20 52 65 74 75 72 6e  le..**.** Return
1d70: 20 74 68 65 20 72 65 73 75 6c 74 69 6e 67 20 63   the resulting c
1d80: 68 65 63 6b 73 75 6d 20 69 6e 20 62 6c 6f 62 20  hecksum in blob 
1d90: 70 4f 75 74 2e 0a 2a 2f 0a 76 6f 69 64 20 76 66  pOut..*/.void vf
1da0: 69 6c 65 5f 61 67 67 72 65 67 61 74 65 5f 63 68  ile_aggregate_ch
1db0: 65 63 6b 73 75 6d 5f 64 69 73 6b 28 69 6e 74 20  ecksum_disk(int 
1dc0: 76 69 64 2c 20 42 6c 6f 62 20 2a 70 4f 75 74 29  vid, Blob *pOut)
1dd0: 7b 0a 20 20 46 49 4c 45 20 2a 69 6e 3b 0a 20 20  {.  FILE *in;.  
1de0: 53 74 6d 74 20 71 3b 0a 20 20 63 68 61 72 20 7a  Stmt q;.  char z
1df0: 42 75 66 5b 34 30 39 36 5d 3b 0a 0a 20 20 64 62  Buf[4096];..  db
1e00: 5f 6d 75 73 74 5f 62 65 5f 77 69 74 68 69 6e 5f  _must_be_within_
1e10: 74 72 65 65 28 29 3b 0a 20 20 64 62 5f 70 72 65  tree();.  db_pre
1e20: 70 61 72 65 28 26 71 2c 20 0a 20 20 20 20 20 20  pare(&q, .      
1e30: 22 53 45 4c 45 43 54 20 25 51 20 7c 7c 20 70 61  "SELECT %Q || pa
1e40: 74 68 6e 61 6d 65 2c 20 70 61 74 68 6e 61 6d 65  thname, pathname
1e50: 2c 20 66 69 6c 65 5f 69 73 5f 73 65 6c 65 63 74  , file_is_select
1e60: 65 64 28 69 64 29 2c 20 72 69 64 20 46 52 4f 4d  ed(id), rid FROM
1e70: 20 76 66 69 6c 65 22 0a 20 20 20 20 20 20 22 20   vfile".      " 
1e80: 57 48 45 52 45 20 4e 4f 54 20 64 65 6c 65 74 65  WHERE NOT delete
1e90: 64 20 41 4e 44 20 76 69 64 3d 25 64 22 0a 20 20  d AND vid=%d".  
1ea0: 20 20 20 20 22 20 4f 52 44 45 52 20 42 59 20 70      " ORDER BY p
1eb0: 61 74 68 6e 61 6d 65 22 2c 0a 20 20 20 20 20 20  athname",.      
1ec0: 67 2e 7a 4c 6f 63 61 6c 52 6f 6f 74 2c 20 76 69  g.zLocalRoot, vi
1ed0: 64 0a 20 20 29 3b 0a 20 20 6d 64 35 73 75 6d 5f  d.  );.  md5sum_
1ee0: 69 6e 69 74 28 29 3b 0a 20 20 77 68 69 6c 65 28  init();.  while(
1ef0: 20 64 62 5f 73 74 65 70 28 26 71 29 3d 3d 53 51   db_step(&q)==SQ
1f00: 4c 49 54 45 5f 52 4f 57 20 29 7b 0a 20 20 20 20  LITE_ROW ){.    
1f10: 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a 46 75 6c  const char *zFul
1f20: 6c 70 61 74 68 20 3d 20 64 62 5f 63 6f 6c 75 6d  lpath = db_colum
1f30: 6e 5f 74 65 78 74 28 26 71 2c 20 30 29 3b 0a 20  n_text(&q, 0);. 
1f40: 20 20 20 63 6f 6e 73 74 20 63 68 61 72 20 2a 7a     const char *z
1f50: 4e 61 6d 65 20 3d 20 64 62 5f 63 6f 6c 75 6d 6e  Name = db_column
1f60: 5f 74 65 78 74 28 26 71 2c 20 31 29 3b 0a 20 20  _text(&q, 1);.  
1f70: 20 20 69 6e 74 20 69 73 53 65 6c 65 63 74 65 64    int isSelected
1f80: 20 3d 20 64 62 5f 63 6f 6c 75 6d 6e 5f 69 6e 74   = db_column_int
1f90: 28 26 71 2c 20 32 29 3b 0a 0a 20 20 20 20 69 66  (&q, 2);..    if
1fa0: 28 20 69 73 53 65 6c 65 63 74 65 64 20 29 7b 0a  ( isSelected ){.
1fb0: 20 20 20 20 20 20 6d 64 35 73 75 6d 5f 73 74 65        md5sum_ste
1fc0: 70 5f 74 65 78 74 28 7a 4e 61 6d 65 2c 20 2d 31  p_text(zName, -1
1fd0: 29 3b 0a 20 20 20 20 20 20 69 6e 20 3d 20 66 6f  );.      in = fo
1fe0: 70 65 6e 28 7a 46 75 6c 6c 70 61 74 68 2c 22 72  pen(zFullpath,"r
1ff0: 62 22 29 3b 0a 20 20 20 20 20 20 69 66 28 20 69  b");.      if( i
2000: 6e 3d 3d 30 20 29 7b 0a 20 20 20 20 20 20 20 20  n==0 ){.        
2010: 6d 64 35 73 75 6d 5f 73 74 65 70 5f 74 65 78 74  md5sum_step_text
2020: 28 22 20 30 5c 6e 22 2c 20 2d 31 29 3b 0a 20 20  (" 0\n", -1);.  
2030: 20 20 20 20 20 20 63 6f 6e 74 69 6e 75 65 3b 0a        continue;.
2040: 20 20 20 20 20 20 7d 0a 20 20 20 20 20 20 66 73        }.      fs
2050: 65 65 6b 28 69 6e 2c 20 30 4c 2c 20 53 45 45 4b  eek(in, 0L, SEEK
2060: 5f 45 4e 44 29 3b 0a 20 20 20 20 20 20 73 70 72  _END);.      spr
2070: 69 6e 74 66 28 7a 42 75 66 2c 20 22 20 25 6c 64  intf(zBuf, " %ld
2080: 5c 6e 22 2c 20 66 74 65 6c 6c 28 69 6e 29 29 3b  \n", ftell(in));
2090: 0a 20 20 20 20 20 20 66 73 65 65 6b 28 69 6e 2c  .      fseek(in,
20a0: 20 30 4c 2c 20 53 45 45 4b 5f 53 45 54 29 3b 0a   0L, SEEK_SET);.
20b0: 20 20 20 20 20 20 6d 64 35 73 75 6d 5f 73 74 65        md5sum_ste
20c0: 70 5f 74 65 78 74 28 7a 42 75 66 2c 20 2d 31 29  p_text(zBuf, -1)
20d0: 3b 0a 20 20 20 20 20 20 66 6f 72 28 3b 3b 29 7b  ;.      for(;;){
20e0: 0a 20 20 20 20 20 20 20 20 69 6e 74 20 6e 3b 0a  .        int n;.
20f0: 20 20 20 20 20 20 20 20 6e 20 3d 20 66 72 65 61          n = frea
2100: 64 28 7a 42 75 66 2c 20 31 2c 20 73 69 7a 65 6f  d(zBuf, 1, sizeo
2110: 66 28 7a 42 75 66 29 2c 20 69 6e 29 3b 0a 20 20  f(zBuf), in);.  
2120: 20 20 20 20 20 20 69 66 28 20 6e 3c 3d 30 20 29        if( n<=0 )
2130: 20 62 72 65 61 6b 3b 0a 20 20 20 20 20 20 20 20   break;.        
2140: 6d 64 35 73 75 6d 5f 73 74 65 70 5f 74 65 78 74  md5sum_step_text
2150: 28 7a 42 75 66 2c 20 6e 29 3b 0a 20 20 20 20 20  (zBuf, n);.     
2160: 20 7d 0a 20 20 20 20 20 20 66 63 6c 6f 73 65 28   }.      fclose(
2170: 69 6e 29 3b 0a 20 20 20 20 7d 65 6c 73 65 7b 0a  in);.    }else{.
2180: 20 20 20 20 20 20 69 6e 74 20 72 69 64 20 3d 20        int rid = 
2190: 64 62 5f 63 6f 6c 75 6d 6e 5f 69 6e 74 28 26 71  db_column_int(&q
21a0: 2c 20 33 29 3b 0a 20 20 20 20 20 20 63 68 61 72  , 3);.      char
21b0: 20 7a 42 75 66 5b 31 30 30 5d 3b 0a 20 20 20 20   zBuf[100];.    
21c0: 20 20 42 6c 6f 62 20 66 69 6c 65 3b 0a 0a 20 20    Blob file;..  
21d0: 20 20 20 20 69 66 28 20 72 69 64 3e 30 20 29 7b      if( rid>0 ){
21e0: 0a 20 20 20 20 20 20 20 20 6d 64 35 73 75 6d 5f  .        md5sum_
21f0: 73 74 65 70 5f 74 65 78 74 28 7a 4e 61 6d 65 2c  step_text(zName,
2200: 20 2d 31 29 3b 0a 20 20 20 20 20 20 20 20 62 6c   -1);.        bl
2210: 6f 62 5f 7a 65 72 6f 28 26 66 69 6c 65 29 3b 0a  ob_zero(&file);.
2220: 20 20 20 20 20 20 20 20 63 6f 6e 74 65 6e 74 5f          content_
2230: 67 65 74 28 72 69 64 2c 20 26 66 69 6c 65 29 3b  get(rid, &file);
2240: 0a 20 20 20 20 20 20 20 20 73 70 72 69 6e 74 66  .        sprintf
2250: 28 7a 42 75 66 2c 20 22 20 25 64 5c 6e 22 2c 20  (zBuf, " %d\n", 
2260: 62 6c 6f 62 5f 73 69 7a 65 28 26 66 69 6c 65 29  blob_size(&file)
2270: 29 3b 0a 20 20 20 20 20 20 20 20 6d 64 35 73 75  );.        md5su
2280: 6d 5f 73 74 65 70 5f 74 65 78 74 28 7a 42 75 66  m_step_text(zBuf
2290: 2c 20 2d 31 29 3b 0a 20 20 20 20 20 20 20 20 6d  , -1);.        m
22a0: 64 35 73 75 6d 5f 73 74 65 70 5f 62 6c 6f 62 28  d5sum_step_blob(
22b0: 26 66 69 6c 65 29 3b 0a 20 20 20 20 20 20 20 20  &file);.        
22c0: 62 6c 6f 62 5f 72 65 73 65 74 28 26 66 69 6c 65  blob_reset(&file
22d0: 29 3b 0a 20 20 20 20 20 20 7d 0a 20 20 20 20 7d  );.      }.    }
22e0: 0a 20 20 7d 0a 20 20 64 62 5f 66 69 6e 61 6c 69  .  }.  db_finali
22f0: 7a 65 28 26 71 29 3b 0a 20 20 6d 64 35 73 75 6d  ze(&q);.  md5sum
2300: 5f 66 69 6e 69 73 68 28 70 4f 75 74 29 3b 0a 7d  _finish(pOut);.}
2310: 0a 0a 2f 2a 0a 2a 2a 20 43 6f 6d 70 75 74 65 20  ../*.** Compute 
2320: 61 6e 20 61 67 67 72 65 67 61 74 65 20 4d 44 35  an aggregate MD5
2330: 20 63 68 65 63 6b 73 75 6d 20 6f 76 65 72 20 74   checksum over t
2340: 68 65 20 72 65 70 6f 73 69 74 6f 72 79 20 69 6d  he repository im
2350: 61 67 65 20 6f 66 20 65 76 65 72 79 0a 2a 2a 20  age of every.** 
2360: 66 69 6c 65 20 69 6e 20 76 69 64 2e 20 20 54 68  file in vid.  Th
2370: 65 20 66 69 6c 65 20 6e 61 6d 65 73 20 61 72 65  e file names are
2380: 20 70 61 72 74 20 6f 66 20 74 68 65 20 63 68 65   part of the che
2390: 63 6b 73 75 6d 2e 0a 2a 2a 0a 2a 2a 20 52 65 74  cksum..**.** Ret
23a0: 75 72 6e 20 74 68 65 20 72 65 73 75 6c 74 69 6e  urn the resultin
23b0: 67 20 63 68 65 63 6b 73 75 6d 20 69 6e 20 62 6c  g checksum in bl
23c0: 6f 62 20 70 4f 75 74 2e 0a 2a 2f 0a 76 6f 69 64  ob pOut..*/.void
23d0: 20 76 66 69 6c 65 5f 61 67 67 72 65 67 61 74 65   vfile_aggregate
23e0: 5f 63 68 65 63 6b 73 75 6d 5f 72 65 70 6f 73 69  _checksum_reposi
23f0: 74 6f 72 79 28 69 6e 74 20 76 69 64 2c 20 42 6c  tory(int vid, Bl
2400: 6f 62 20 2a 70 4f 75 74 29 7b 0a 20 20 42 6c 6f  ob *pOut){.  Blo
2410: 62 20 66 69 6c 65 3b 0a 20 20 53 74 6d 74 20 71  b file;.  Stmt q
2420: 3b 0a 20 20 63 68 61 72 20 7a 42 75 66 5b 31 30  ;.  char zBuf[10
2430: 30 5d 3b 0a 0a 20 20 64 62 5f 6d 75 73 74 5f 62  0];..  db_must_b
2440: 65 5f 77 69 74 68 69 6e 5f 74 72 65 65 28 29 3b  e_within_tree();
2450: 0a 20 20 0a 20 20 64 62 5f 70 72 65 70 61 72 65  .  .  db_prepare
2460: 28 26 71 2c 20 22 53 45 4c 45 43 54 20 70 61 74  (&q, "SELECT pat
2470: 68 6e 61 6d 65 2c 20 72 69 64 20 46 52 4f 4d 20  hname, rid FROM 
2480: 76 66 69 6c 65 22 0a 20 20 20 20 20 20 20 20 20  vfile".         
2490: 20 20 20 20 20 20 20 20 22 20 57 48 45 52 45 20          " WHERE 
24a0: 4e 4f 54 20 64 65 6c 65 74 65 64 20 41 4e 44 20  NOT deleted AND 
24b0: 72 69 64 3e 30 20 41 4e 44 20 76 69 64 3d 25 64  rid>0 AND vid=%d
24c0: 22 0a 20 20 20 20 20 20 20 20 20 20 20 20 20 20  ".              
24d0: 20 20 20 22 20 4f 52 44 45 52 20 42 59 20 70 61     " ORDER BY pa
24e0: 74 68 6e 61 6d 65 22 2c 0a 20 20 20 20 20 20 20  thname",.       
24f0: 20 20 20 20 20 20 20 20 20 20 76 69 64 29 3b 0a            vid);.
2500: 20 20 62 6c 6f 62 5f 7a 65 72 6f 28 26 66 69 6c    blob_zero(&fil
2510: 65 29 3b 0a 20 20 6d 64 35 73 75 6d 5f 69 6e 69  e);.  md5sum_ini
2520: 74 28 29 3b 0a 20 20 77 68 69 6c 65 28 20 64 62  t();.  while( db
2530: 5f 73 74 65 70 28 26 71 29 3d 3d 53 51 4c 49 54  _step(&q)==SQLIT
2540: 45 5f 52 4f 57 20 29 7b 0a 20 20 20 20 63 6f 6e  E_ROW ){.    con
2550: 73 74 20 63 68 61 72 20 2a 7a 4e 61 6d 65 20 3d  st char *zName =
2560: 20 64 62 5f 63 6f 6c 75 6d 6e 5f 74 65 78 74 28   db_column_text(
2570: 26 71 2c 20 30 29 3b 0a 20 20 20 20 69 6e 74 20  &q, 0);.    int 
2580: 72 69 64 20 3d 20 64 62 5f 63 6f 6c 75 6d 6e 5f  rid = db_column_
2590: 69 6e 74 28 26 71 2c 20 31 29 3b 0a 20 20 20 20  int(&q, 1);.    
25a0: 6d 64 35 73 75 6d 5f 73 74 65 70 5f 74 65 78 74  md5sum_step_text
25b0: 28 7a 4e 61 6d 65 2c 20 2d 31 29 3b 0a 20 20 20  (zName, -1);.   
25c0: 20 63 6f 6e 74 65 6e 74 5f 67 65 74 28 72 69 64   content_get(rid
25d0: 2c 20 26 66 69 6c 65 29 3b 0a 20 20 20 20 73 70  , &file);.    sp
25e0: 72 69 6e 74 66 28 7a 42 75 66 2c 20 22 20 25 64  rintf(zBuf, " %d
25f0: 5c 6e 22 2c 20 62 6c 6f 62 5f 73 69 7a 65 28 26  \n", blob_size(&
2600: 66 69 6c 65 29 29 3b 0a 20 20 20 20 6d 64 35 73  file));.    md5s
2610: 75 6d 5f 73 74 65 70 5f 74 65 78 74 28 7a 42 75  um_step_text(zBu
2620: 66 2c 20 2d 31 29 3b 0a 20 20 20 20 6d 64 35 73  f, -1);.    md5s
2630: 75 6d 5f 73 74 65 70 5f 62 6c 6f 62 28 26 66 69  um_step_blob(&fi
2640: 6c 65 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 72 65  le);.    blob_re
2650: 73 65 74 28 26 66 69 6c 65 29 3b 0a 20 20 7d 0a  set(&file);.  }.
2660: 20 20 64 62 5f 66 69 6e 61 6c 69 7a 65 28 26 71    db_finalize(&q
2670: 29 3b 0a 20 20 6d 64 35 73 75 6d 5f 66 69 6e 69  );.  md5sum_fini
2680: 73 68 28 70 4f 75 74 29 3b 0a 7d 0a 0a 2f 2a 0a  sh(pOut);.}../*.
2690: 2a 2a 20 43 6f 6d 70 75 74 65 20 61 6e 20 61 67  ** Compute an ag
26a0: 67 72 65 67 61 74 65 20 4d 44 35 20 63 68 65 63  gregate MD5 chec
26b0: 6b 73 75 6d 20 6f 76 65 72 20 74 68 65 20 72 65  ksum over the re
26c0: 70 6f 73 69 74 6f 72 79 20 69 6d 61 67 65 20 6f  pository image o
26d0: 66 20 65 76 65 72 79 0a 2a 2a 20 66 69 6c 65 20  f every.** file 
26e0: 69 6e 20 6d 61 6e 69 66 65 73 74 20 76 69 64 2e  in manifest vid.
26f0: 20 20 54 68 65 20 66 69 6c 65 20 6e 61 6d 65 73    The file names
2700: 20 61 72 65 20 70 61 72 74 20 6f 66 20 74 68 65   are part of the
2710: 20 63 68 65 63 6b 73 75 6d 2e 0a 2a 2a 0a 2a 2a   checksum..**.**
2720: 20 52 65 74 75 72 6e 20 74 68 65 20 72 65 73 75   Return the resu
2730: 6c 74 69 6e 67 20 63 68 65 63 6b 73 75 6d 20 69  lting checksum i
2740: 6e 20 62 6c 6f 62 20 70 4f 75 74 2e 0a 2a 2f 0a  n blob pOut..*/.
2750: 76 6f 69 64 20 76 66 69 6c 65 5f 61 67 67 72 65  void vfile_aggre
2760: 67 61 74 65 5f 63 68 65 63 6b 73 75 6d 5f 6d 61  gate_checksum_ma
2770: 6e 69 66 65 73 74 28 69 6e 74 20 76 69 64 2c 20  nifest(int vid, 
2780: 42 6c 6f 62 20 2a 70 4f 75 74 2c 20 42 6c 6f 62  Blob *pOut, Blob
2790: 20 2a 70 4d 61 6e 4f 75 74 29 7b 0a 20 20 69 6e   *pManOut){.  in
27a0: 74 20 69 2c 20 66 69 64 3b 0a 20 20 42 6c 6f 62  t i, fid;.  Blob
27b0: 20 66 69 6c 65 2c 20 6d 66 69 6c 65 3b 0a 20 20   file, mfile;.  
27c0: 4d 61 6e 69 66 65 73 74 20 6d 3b 0a 20 20 63 68  Manifest m;.  ch
27d0: 61 72 20 7a 42 75 66 5b 31 30 30 5d 3b 0a 0a 20  ar zBuf[100];.. 
27e0: 20 64 62 5f 6d 75 73 74 5f 62 65 5f 77 69 74 68   db_must_be_with
27f0: 69 6e 5f 74 72 65 65 28 29 3b 0a 20 20 63 6f 6e  in_tree();.  con
2800: 74 65 6e 74 5f 67 65 74 28 76 69 64 2c 20 26 6d  tent_get(vid, &m
2810: 66 69 6c 65 29 3b 0a 20 20 69 66 28 20 6d 61 6e  file);.  if( man
2820: 69 66 65 73 74 5f 70 61 72 73 65 28 26 6d 2c 20  ifest_parse(&m, 
2830: 26 6d 66 69 6c 65 29 3d 3d 30 20 29 7b 0a 20 20  &mfile)==0 ){.  
2840: 20 20 62 6c 6f 62 5f 7a 65 72 6f 28 70 4f 75 74    blob_zero(pOut
2850: 29 3b 0a 20 20 20 20 72 65 74 75 72 6e 3b 0a 20  );.    return;. 
2860: 20 7d 0a 20 20 66 6f 72 28 69 3d 30 3b 20 69 3c   }.  for(i=0; i<
2870: 6d 2e 6e 46 69 6c 65 3b 20 69 2b 2b 29 7b 0a 20  m.nFile; i++){. 
2880: 20 20 20 66 69 64 20 3d 20 75 75 69 64 5f 74 6f     fid = uuid_to
2890: 5f 72 69 64 28 6d 2e 61 46 69 6c 65 5b 69 5d 2e  _rid(m.aFile[i].
28a0: 7a 55 75 69 64 2c 20 30 29 3b 0a 20 20 20 20 6d  zUuid, 0);.    m
28b0: 64 35 73 75 6d 5f 73 74 65 70 5f 74 65 78 74 28  d5sum_step_text(
28c0: 6d 2e 61 46 69 6c 65 5b 69 5d 2e 7a 4e 61 6d 65  m.aFile[i].zName
28d0: 2c 20 2d 31 29 3b 0a 20 20 20 20 63 6f 6e 74 65  , -1);.    conte
28e0: 6e 74 5f 67 65 74 28 66 69 64 2c 20 26 66 69 6c  nt_get(fid, &fil
28f0: 65 29 3b 0a 20 20 20 20 73 70 72 69 6e 74 66 28  e);.    sprintf(
2900: 7a 42 75 66 2c 20 22 20 25 64 5c 6e 22 2c 20 62  zBuf, " %d\n", b
2910: 6c 6f 62 5f 73 69 7a 65 28 26 66 69 6c 65 29 29  lob_size(&file))
2920: 3b 0a 20 20 20 20 6d 64 35 73 75 6d 5f 73 74 65  ;.    md5sum_ste
2930: 70 5f 74 65 78 74 28 7a 42 75 66 2c 20 2d 31 29  p_text(zBuf, -1)
2940: 3b 0a 20 20 20 20 6d 64 35 73 75 6d 5f 73 74 65  ;.    md5sum_ste
2950: 70 5f 62 6c 6f 62 28 26 66 69 6c 65 29 3b 0a 20  p_blob(&file);. 
2960: 20 20 20 62 6c 6f 62 5f 72 65 73 65 74 28 26 66     blob_reset(&f
2970: 69 6c 65 29 3b 0a 20 20 7d 0a 20 20 69 66 28 20  ile);.  }.  if( 
2980: 70 4d 61 6e 4f 75 74 20 29 7b 0a 20 20 20 20 62  pManOut ){.    b
2990: 6c 6f 62 5f 7a 65 72 6f 28 70 4d 61 6e 4f 75 74  lob_zero(pManOut
29a0: 29 3b 0a 20 20 20 20 62 6c 6f 62 5f 61 70 70 65  );.    blob_appe
29b0: 6e 64 28 70 4d 61 6e 4f 75 74 2c 20 6d 2e 7a 52  nd(pManOut, m.zR
29c0: 65 70 6f 43 6b 73 75 6d 2c 20 2d 31 29 3b 0a 20  epoCksum, -1);. 
29d0: 20 7d 0a 20 20 6d 61 6e 69 66 65 73 74 5f 63 6c   }.  manifest_cl
29e0: 65 61 72 28 26 6d 29 3b 0a 20 20 6d 64 35 73 75  ear(&m);.  md5su
29f0: 6d 5f 66 69 6e 69 73 68 28 70 4f 75 74 29 3b 0a  m_finish(pOut);.
2a00: 7d 0a 0a 2f 2a 0a 2a 2a 20 43 4f 4d 4d 41 4e 44  }../*.** COMMAND
2a10: 3a 20 74 65 73 74 2d 61 67 67 2d 63 6b 73 75 6d  : test-agg-cksum
2a20: 0a 2a 2f 0a 76 6f 69 64 20 74 65 73 74 5f 61 67  .*/.void test_ag
2a30: 67 5f 63 6b 73 75 6d 5f 63 6d 64 28 76 6f 69 64  g_cksum_cmd(void
2a40: 29 7b 0a 20 20 69 6e 74 20 76 69 64 3b 0a 20 20  ){.  int vid;.  
2a50: 42 6c 6f 62 20 68 61 73 68 2c 20 68 61 73 68 32  Blob hash, hash2
2a60: 3b 0a 20 20 64 62 5f 6d 75 73 74 5f 62 65 5f 77  ;.  db_must_be_w
2a70: 69 74 68 69 6e 5f 74 72 65 65 28 29 3b 0a 20 20  ithin_tree();.  
2a80: 76 69 64 20 3d 20 64 62 5f 6c 67 65 74 5f 69 6e  vid = db_lget_in
2a90: 74 28 22 63 68 65 63 6b 6f 75 74 22 2c 20 30 29  t("checkout", 0)
2aa0: 3b 0a 20 20 76 66 69 6c 65 5f 61 67 67 72 65 67  ;.  vfile_aggreg
2ab0: 61 74 65 5f 63 68 65 63 6b 73 75 6d 5f 64 69 73  ate_checksum_dis
2ac0: 6b 28 76 69 64 2c 20 26 68 61 73 68 29 3b 0a 20  k(vid, &hash);. 
2ad0: 20 70 72 69 6e 74 66 28 22 64 69 73 6b 3a 20 20   printf("disk:  
2ae0: 20 20 20 25 73 5c 6e 22 2c 20 62 6c 6f 62 5f 73     %s\n", blob_s
2af0: 74 72 28 26 68 61 73 68 29 29 3b 0a 20 20 62 6c  tr(&hash));.  bl
2b00: 6f 62 5f 72 65 73 65 74 28 26 68 61 73 68 29 3b  ob_reset(&hash);
2b10: 0a 20 20 76 66 69 6c 65 5f 61 67 67 72 65 67 61  .  vfile_aggrega
2b20: 74 65 5f 63 68 65 63 6b 73 75 6d 5f 72 65 70 6f  te_checksum_repo
2b30: 73 69 74 6f 72 79 28 76 69 64 2c 20 26 68 61 73  sitory(vid, &has
2b40: 68 29 3b 0a 20 20 70 72 69 6e 74 66 28 22 61 72  h);.  printf("ar
2b50: 63 68 69 76 65 3a 20 20 25 73 5c 6e 22 2c 20 62  chive:  %s\n", b
2b60: 6c 6f 62 5f 73 74 72 28 26 68 61 73 68 29 29 3b  lob_str(&hash));
2b70: 0a 20 20 62 6c 6f 62 5f 72 65 73 65 74 28 26 68  .  blob_reset(&h
2b80: 61 73 68 29 3b 0a 20 20 76 66 69 6c 65 5f 61 67  ash);.  vfile_ag
2b90: 67 72 65 67 61 74 65 5f 63 68 65 63 6b 73 75 6d  gregate_checksum
2ba0: 5f 6d 61 6e 69 66 65 73 74 28 76 69 64 2c 20 26  _manifest(vid, &
2bb0: 68 61 73 68 2c 20 26 68 61 73 68 32 29 3b 0a 20  hash, &hash2);. 
2bc0: 20 70 72 69 6e 74 66 28 22 6d 61 6e 69 66 65 73   printf("manifes
2bd0: 74 3a 20 25 73 5c 6e 22 2c 20 62 6c 6f 62 5f 73  t: %s\n", blob_s
2be0: 74 72 28 26 68 61 73 68 29 29 3b 0a 20 20 70 72  tr(&hash));.  pr
2bf0: 69 6e 74 66 28 22 72 65 63 6f 72 64 65 64 3a 20  intf("recorded: 
2c00: 25 73 5c 6e 22 2c 20 62 6c 6f 62 5f 73 74 72 28  %s\n", blob_str(
2c10: 26 68 61 73 68 32 29 29 3b 0a 7d 0a              &hash2));.}.