Fossil

Check-in [a955c80b43]
Login

Check-in [a955c80b43]

Many hyperlinks are disabled.
Use anonymous login to enable hyperlinks.

Overview
Comment:Added a "no changes" messages if no local files are different; report trying to view a non-existent local file. Tidy-up and added config-options and "what I did" block near top of info.c.
Downloads: Tarball | ZIP archive | SQL archive
Timelines: family | ancestors | descendants | both | ui-local-diff
Files: files | file ages | folders
SHA3-256: a955c80b43c358fb819487b330a395a6c690f9931b6c10a0d4828b88e941d828
User & Date: graham 2020-06-02 16:20:00
Context
2021-02-14
21:28
Brought graham's ui-local-diff branch up to date with respect to trunk. This is not a simple merge, because it had to accommodate several conflicting changes, so it needs to be re-checked for sanity before being merged down to trunk. Also, some of the changes I made simply reduce the size of the resulting diff relative to trunk and need to be reverted; these are primarily 2-line splits of "else/if" so the branch-specific change doesn't repeat the following "if" condition, which is now the "else if" condition. I think this sort of thing aids understanding of the diff, even though it breaks the style guidelines. ... (check-in: 0516f4d3c8 user: wyoung tags: ui-local-diff)
2020-06-02
16:20
Added a "no changes" messages if no local files are different; report trying to view a non-existent local file. Tidy-up and added config-options and "what I did" block near top of info.c. ... (check-in: a955c80b43 user: graham tags: ui-local-diff)
13:52
Merge in Markdown changes from trunk. ... (check-in: bf5a21e036 user: graham tags: ui-local-diff)
Changes
Hide Diffs Unified Diffs Ignore Whitespace Patch

Changes to src/info.c.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
/*TODO Graham's Notes
** o  Should /file behave differently for non-existent local files?
** o  Find a place to add links to /local.
** o  Remove //TODO TESTING HACK TODO
** ?? In hexdump_page(), should content (and downloadName?) be reset/freed?
** ?? In clean_cmd() in checkin.c, should "Blob repo" be blob_reset()?
** ?? Do I need to worry about deleting/emptying TEMP SFILE?
** ?? Is it normal for one artifact to have several check-ins associated with
**    it? In the test fossil (\x\$Test\Fossil) there are several commits under
**    the same artifact...
** ?? A setting to control one- or two-pass mode?
** ?? Add two-pass to the normal loop?
** ?? A setting to control whether Extras are intially shown or not?
** ?? A setting to control max. entries to show initially?
**------------------------------------------------------------------------------
*/
/*
** Copyright (c) 2007 D. Richard Hipp
**
** This program is free software; you can redistribute it and/or
** modify it under the terms of the Simplified BSD License (also
** known as the "2-Clause License" or "FreeBSD License".)

<
<
<
<
<
<
<
<
<
<
<
<
<
<
<
<























1
2
3
4
5
6
7
















/*
** Copyright (c) 2007 D. Richard Hipp
**
** This program is free software; you can redistribute it and/or
** modify it under the terms of the Simplified BSD License (also
** known as the "2-Clause License" or "FreeBSD License".)

34
35
36
37
38
39
40












































41
42
43
44
45
46
47
** This file contains code to implement the "info" command.  The
** "info" command gives command-line access to information about
** the current tree, or a particular artifact or check-in.
*/
#include "config.h"
#include "info.h"
#include <assert.h>













































/*
** Return a string (in memory obtained from malloc) holding a
** comma-separated list of tags that apply to check-in with
** record-id rid.  If the "propagatingOnly" flag is true, then only
** show branch tags (tags that propagate to children).
**







>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>







18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
** This file contains code to implement the "info" command.  The
** "info" command gives command-line access to information about
** the current tree, or a particular artifact or check-in.
*/
#include "config.h"
#include "info.h"
#include <assert.h>

/*-----------------------------------------------------------------------------
** LOCAL DIFF CHANGES
**
** The goal is to show "(working) changes in the current checkout" in much the
** same way as the current "check-in" pages show the changes within a check-in.
**
** The page is on "/local" (no parameters needed; an alias of "/vinfo" and
** "/ci"), but at present there are no links to it within Fossil. Other changes:
** "/localpatch" produces a patch-set for all changes in the current checkout;
** "/localdiff" is an alias for "/fdiff" to show changes for an individual file;
** both "/file" and "/hexdump" have been extended to support local files.
**
** A number of points-of-query are labeld "TODO:LD". Most relate to these
** changes, although some are about existing code.
**
** In "checkin.c", the function "locate_unmanaged_files()" has been made
** non-static so that it can be called from here.
**-----------------------------------------------------------------------------
** Options to control the "local changes" changes. At present, these are
** defines: if these changes are adopted, some may want to be made into
** configuration options.
**
** INTEGER: Controls how many unmanaged files will be shown before the "plus xxx
** other matching files." line is shown (with an option to view all of them).*/
#define LOCAL_DIFF_MAX_EXTRAS       (5)
/*
** STRING: Controls whether the "extras" report is initially shown or hidden. A
** value of "0" hides the report; a value of "" (an empty string) will show it.
*/
#define LOCAL_DIFF_EXTRAS_MODE      ("")
/*
** BOOLEAN: Controls whether one or two passes are made through the list of
** changed files. In two-pass mode, all single-line differences are displayed
** ahead of all differences involving "diff-blocks", making them less likely to
** be overlooked. If disabled, only one pass is made, listing all changes in the
** order found. Possible TODO: Do the same for "normal" diffs. */
#define LOCAL_DIFF_USE_TWO_PASSES   (1)
/*
** BOOLEAN: Controls whether dividers ("<hr/>") added after any "diff-blocks"
** (except the last one)... IMHO doing so makes it easier to see where one block
** ends and the next starts. Possible TODO: Do the same for "normal" diffs. */
#define LOCAL_DIFF_ADD_DIVIDER      (1)
/*---------------------------------------------------------------------------*/

/*
** Return a string (in memory obtained from malloc) holding a
** comma-separated list of tags that apply to check-in with
** record-id rid.  If the "propagatingOnly" flag is true, then only
** show branch tags (tags that propagate to children).
**
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357


358
359
360
361
362
363
364
         |TIMELINE_XMERGE
         |TIMELINE_CHPICK,
       0, 0, 0, rid, rid2, 0);
  db_finalize(&q);
}

/*
** Read the content of file zName (prepended with the checkout directory)
** and put it into the uninitialized blob. The blob is zeroed if the file
** does not exist (if the file cannot be read, blob_read_from_file() aborts
** the program).
*/
static void content_from_file(
  const char *zName,    /* Filename (relative to checkout) of file to be read */
  Blob *pBlob           /* Pointer to blob to receive contents */
){
  const char *zFullPath = mprintf("%s%s", g.zLocalRoot, zName);
  blob_zero(pBlob);
  if( file_size(zFullPath, ExtFILE)>=0 ){
    blob_read_from_file(pBlob, zFullPath, ExtFILE);
  }


}

/*
** Append the difference between artifacts to the output
** If zLocal is not NULL, instead compare against the local
** copy of the file it names in the repository.
*/







|
|
|
<

|





|
|

>
>







365
366
367
368
369
370
371
372
373
374

375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
         |TIMELINE_XMERGE
         |TIMELINE_CHPICK,
       0, 0, 0, rid, rid2, 0);
  db_finalize(&q);
}

/*
** Read the content of file zName (prepended with the checkout directory) and
** put it into the uninitialized blob, returning 1. The blob is zeroed if the
** file does not exist or cannot be accessed, in which case it returns 0.

*/
static int content_from_file(
  const char *zName,    /* Filename (relative to checkout) of file to be read */
  Blob *pBlob           /* Pointer to blob to receive contents */
){
  const char *zFullPath = mprintf("%s%s", g.zLocalRoot, zName);
  blob_zero(pBlob);
  if( file_size(zFullPath, ExtFILE)<0 ){
    return 0;
  }
  blob_read_from_file(pBlob, zFullPath, ExtFILE);
  return 1;
}

/*
** Append the difference between artifacts to the output
** If zLocal is not NULL, instead compare against the local
** copy of the file it names in the repository.
*/
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
  int isLink,           /* Is the file a symbolic link? */
  u64 diffFlags,        /* Flags for text_diff().  Zero to omit diffs */
  ReCompiled *pRe,      /* Only show diffs that match this regex, if not NULL */
  int pass              /* 0x01 - Display single-line entries only        */
                        /* 0x02 - Display entries with "diff blocks" only */
                        /* 0x03 - Display both                            */
){
#ifndef GLH_NO_DIVIDER
  /* This remembers whether a side-by-side "diff-block" was shown the last
  ** time through. If it was, we will add "<hr/>" to better separate the
  ** blocks and so single-line entries (when not in two-pass mode) are easier
  ** to spot.
  */
  static int diffShownLastTime = 0;
#endif
  char *zFullName = mprintf("%s%s", g.zLocalRoot, zName);
  int isFilePresent = !file_access(zFullName, F_OK);
  /* Determing up-front whether we would be showing a "diff-block" so we can
  ** filter them according to which pass we are on: the first pass will show







|


|
<







541
542
543
544
545
546
547
548
549
550
551

552
553
554
555
556
557
558
  int isLink,           /* Is the file a symbolic link? */
  u64 diffFlags,        /* Flags for text_diff().  Zero to omit diffs */
  ReCompiled *pRe,      /* Only show diffs that match this regex, if not NULL */
  int pass              /* 0x01 - Display single-line entries only        */
                        /* 0x02 - Display entries with "diff blocks" only */
                        /* 0x03 - Display both                            */
){
#if LOCAL_DIFF_ADD_DIVIDER
  /* This remembers whether a side-by-side "diff-block" was shown the last
  ** time through. If it was, we will add "<hr/>" to better separate the
  ** blocks.

  */
  static int diffShownLastTime = 0;
#endif
  char *zFullName = mprintf("%s%s", g.zLocalRoot, zName);
  int isFilePresent = !file_access(zFullName, F_OK);
  /* Determing up-front whether we would be showing a "diff-block" so we can
  ** filter them according to which pass we are on: the first pass will show
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
                                           )
                 );
  /* We don't use 'diffFlags' in these tests so that whether "Hide diffs" is
  ** in effect or not, the order won't change.
  */
  if(  showDiff && (pass == 1) ){ return; } /* Don't do diff on pass 1 of 2 */
  if( !showDiff && (pass == 2) ){ return; } /* Don't do line on pass 2 of 2 */
#ifndef GLH_NO_DIVIDER
  /* If a SBS diff-block was shown by the previous entry, add a divider */
  if( diffShownLastTime && (diffFlags & DIFF_SIDEBYSIDE) ){
    @ <hr/>
  }
  /* Record whether we will be showing a diff-block this time. We DO factor in
  ** 'diffFlags' here so that in "Hide diffs" mode, we don't get extra lines.
  */
  diffShownLastTime = showDiff && diffFlags;
#endif /*GLH_NO_DIVIDER*/
//--------------------------------------------------- TODO TESTING HACK TODO
if( strncmp(zName,"aa",2)==0 ){
  isChnged = atoi(zName+2);
}
//--------------------------------------------------- TODO TESTING HACK TODO
  @ <p>
  if( !g.perm.Hyperlink ){
    if( isDeleted ){
      if( isFilePresent ){
        @ Deleted %h(zName) (still present as a local file).
        //TODO:Remove? showDiff = 1;
      }else{
        @ Deleted %h(zName).
      }
    }else if( isNew ){
      if( isFilePresent ){
        @ Added %h(zName) but not committed.
      }else{
        @ Missing %h(zName) (was added to checkout).
      }
    }else switch( isChnged ){
      /*TODO
      ** These "special cases" have not all been properly tested (by creating
      ** entries in a in a repository to trigger them), but they do display
      ** as expected when "forced" to appear.
      */
      case 3:
        @ Added %h(zName) due to a merge.
        break;
      case 5:
        @ Added %h(zName) due to an integrate-merge.
        break;
      case 6:   append_status( "gained", "executable", zName, zOld);    break;







|








|
<
<
<
<
<





<










<
<
<
<
<







567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583





584
585
586
587
588

589
590
591
592
593
594
595
596
597
598





599
600
601
602
603
604
605
                                           )
                 );
  /* We don't use 'diffFlags' in these tests so that whether "Hide diffs" is
  ** in effect or not, the order won't change.
  */
  if(  showDiff && (pass == 1) ){ return; } /* Don't do diff on pass 1 of 2 */
  if( !showDiff && (pass == 2) ){ return; } /* Don't do line on pass 2 of 2 */
#if LOCAL_DIFF_ADD_DIVIDER
  /* If a SBS diff-block was shown by the previous entry, add a divider */
  if( diffShownLastTime && (diffFlags & DIFF_SIDEBYSIDE) ){
    @ <hr/>
  }
  /* Record whether we will be showing a diff-block this time. We DO factor in
  ** 'diffFlags' here so that in "Hide diffs" mode, we don't get extra lines.
  */
  diffShownLastTime = showDiff && diffFlags;
#endif





  @ <p>
  if( !g.perm.Hyperlink ){
    if( isDeleted ){
      if( isFilePresent ){
        @ Deleted %h(zName) (still present as a local file).

      }else{
        @ Deleted %h(zName).
      }
    }else if( isNew ){
      if( isFilePresent ){
        @ Added %h(zName) but not committed.
      }else{
        @ Missing %h(zName) (was added to checkout).
      }
    }else switch( isChnged ){





      case 3:
        @ Added %h(zName) due to a merge.
        break;
      case 5:
        @ Added %h(zName) due to an integrate-merge.
        break;
      case 6:   append_status( "gained", "executable", zName, zOld);    break;
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
            @ Merge
            break;
          case 4:
            @ Integrate-merge
            break;
        }
        @ of %h(zName).
        //TODO:Remove? showDiff = 1;
    }
    if( showDiff && diffFlags ){
      append_diff(zOld, NULL, zName, diffFlags, pRe);
    }
  }else{ /* With hyperlinks */
    if( isDeleted ){
      if( isFilePresent ){                    /* DELETEd but still on disk */
        @ Deleted %z(href("%R/finfo?name=%T&m=%!S",zName,zOld))%h(zName)</a>
        @ from %z(href("%R/artifact/%!S",zOld))[%S(zOld)]</a> (still present
        @ as %z(href("%R/file/%T?ci=ckout&annot=removed from checkout",zName))
        @ [local file]</a>).
        //TODO:Remove? showDiff = 1;
      }else{                              /* DELETEd and deleted from disk */
        @ Deleted %z(href("%R/finfo?name=%T&m=%!S",zName,zOld))%h(zName)</a>
        @ from %z(href("%R/artifact/%!S",zOld))[%S(zOld)]</a>.
      }
    }else if( isNew ){
      if( isFilePresent ){                    /* ADDed and present on disk */
        @ Added %z(href("%R/file/%T?ci=ckout",zName))%h(zName)</a>
        @ but not committed.
      }else{                              /* ADDed but not present on disk */
        @ Missing %h(zName) (was added to checkout).
      }
    }else switch( isChnged ){
      /*TODO Not fully tested... see see no-hyperlink version above */
      case 3:                                          /* Added by a merge */
        @ Added
        @ %z(href("%R/file/%T?ci=ckout&annot=added by merge",zName))%h(zName)
        @ </a> to %z(href("%R/artifact/%!S",zOld))[%S(zOld)]</a> due to merge.
        break;
      case 5:                             /* Added by an integration merge */
        @ Added







<











<












<







616
617
618
619
620
621
622

623
624
625
626
627
628
629
630
631
632
633

634
635
636
637
638
639
640
641
642
643
644
645

646
647
648
649
650
651
652
            @ Merge
            break;
          case 4:
            @ Integrate-merge
            break;
        }
        @ of %h(zName).

    }
    if( showDiff && diffFlags ){
      append_diff(zOld, NULL, zName, diffFlags, pRe);
    }
  }else{ /* With hyperlinks */
    if( isDeleted ){
      if( isFilePresent ){                    /* DELETEd but still on disk */
        @ Deleted %z(href("%R/finfo?name=%T&m=%!S",zName,zOld))%h(zName)</a>
        @ from %z(href("%R/artifact/%!S",zOld))[%S(zOld)]</a> (still present
        @ as %z(href("%R/file/%T?ci=ckout&annot=removed from checkout",zName))
        @ [local file]</a>).

      }else{                              /* DELETEd and deleted from disk */
        @ Deleted %z(href("%R/finfo?name=%T&m=%!S",zName,zOld))%h(zName)</a>
        @ from %z(href("%R/artifact/%!S",zOld))[%S(zOld)]</a>.
      }
    }else if( isNew ){
      if( isFilePresent ){                    /* ADDed and present on disk */
        @ Added %z(href("%R/file/%T?ci=ckout",zName))%h(zName)</a>
        @ but not committed.
      }else{                              /* ADDed but not present on disk */
        @ Missing %h(zName) (was added to checkout).
      }
    }else switch( isChnged ){

      case 3:                                          /* Added by a merge */
        @ Added
        @ %z(href("%R/file/%T?ci=ckout&annot=added by merge",zName))%h(zName)
        @ </a> to %z(href("%R/artifact/%!S",zOld))[%S(zOld)]</a> due to merge.
        break;
      case 5:                             /* Added by an integration merge */
        @ Added
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
            @ Integrate-merge
            break;
        }
        @ of %z(href("%R/finfo?name=%T&m=%!S",zName,zOld))%h(zName)</a>
        @ from %z(href("%R/artifact/%!S",zOld))[%S(zOld)]</a> to
        @ %z(href("%R/file/%T?ci=ckout&annot=edited locally",zName))
        @ [local file]</a>
        //TODO:Remove? showDiff = 1;
    }
    if( showDiff ){
      if( diffFlags ){
        append_diff(zOld, NULL, zName, diffFlags, pRe);
      }else if( isChnged ){
        @ &nbsp;&nbsp;
        @ %z(href("%R/localdiff?name=%T&from=%!S",zName,zOld))[diff]</a>







<







671
672
673
674
675
676
677

678
679
680
681
682
683
684
            @ Integrate-merge
            break;
        }
        @ of %z(href("%R/finfo?name=%T&m=%!S",zName,zOld))%h(zName)</a>
        @ from %z(href("%R/artifact/%!S",zOld))[%S(zOld)]</a> to
        @ %z(href("%R/file/%T?ci=ckout&annot=edited locally",zName))
        @ [local file]</a>

    }
    if( showDiff ){
      if( diffFlags ){
        append_diff(zOld, NULL, zName, diffFlags, pRe);
      }else if( isChnged ){
        @ &nbsp;&nbsp;
        @ %z(href("%R/localdiff?name=%T&from=%!S",zName,zOld))[diff]</a>
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
  zCleanFlag = db_get("clean-glob", 0);   /* ...that "clean" clobbers */
  zKeepFlag = db_get("keep-glob", 0);     /* ...that "clean" won't touch */
  pIgnore = glob_create(zIgnoreFlag);     /* Object versions of above */
  pKeep = glob_create(zKeepFlag);
  pClean = glob_create(zCleanFlag);
  nRoot = (int)strlen(g.zLocalRoot);      /* Length of root component */
  Stmt q;
  Blob repo;                              /* TODO May not be needed */
  int maxExtrasToShow = 5;                /* TODO Take from a setting? */
  int extrasFlags = atoi(zExtra);         /* Which entries to show */
  int nExtras;
  int nMatch;
  int nShown;
  int nPlain;
  int nIgnore;
  int nClean;
  int nKeep;

  /*TODO?
  ** It feels sensible to limit the number of "extra" entries shown by default
  ** for cases where "ignore-glob" or "clean-glob" haven't been fully setup.
  ** A minor irritation is that this can lead to "... plus 1 more file", on a
  ** line that COULD have been used to display the omitted file. If we knew in
  ** advance how many entries were going to match, we could temporarily "bump"
  ** the limit by one show all entries would be shown. However, to know the
  ** number of matches in advance we'd have to:
  ** a) Pre-scan SFILE, testing and counting matches against each glob-list,
  **    possibly bump the limit, then re-scan the table repeating the tests
  **    against each glob-list to decide which to show.
  ** b) SFILE could have an extra FLAGS field: during the pre-scan, this could
  **    be updated to indicate which groups each file belong to. This would
  **    save re-testing every file against each glob-list (the main pass could
  **    select "WHERE flags & selector" to get only the matching entries, but
  **    the updates (selecting by "pathname" each time) could be a bit much.
  ** c) vfile_scan() -- where SFILE is populated -- COULD have an option to
  **    do the testing at the time entries are added. This would be the "best"
  **    way, but feels too much disruption to other code for what is only a
  **    minor benefit.
  ** For now, I'll stick with the minor annoyance of "plus 1 more file" :-)
  **
  ** Being able to determine the counts up-front would also allow us to hide
  ** the "extras report" if there were no unmanaged files.
  **
  **TODO?
  ** Does it make sense (and/or is it practiable) to offer an "ADD" button
  ** against files that are unmanaged?
  **
  **TODO?
  ** Does it make sense (and/or ...) to offer ediing of the various blob-lists
  ** from the Extras report? Showing the existing configuration screen would
  ** probably not be a problem (permissions permitting), but what happens if
  ** those settings have been overriden by .fossil-settings/ignore-glob? As we
  ** have access to the local checkout, is it feasible to edit it in the browser
  ** (perhaps piggy-backing /fileedit)?
  */

  locate_unmanaged_files(0, NULL, 0, NULL);   /* Get all unmanaged */
  /*TODO
  ** The first two of these exclusions come from clean_cmd() in checkin.c.
  ** Not sure exactly what they are intended to do (seem to have no effect on
  ** my test repos). Last exclusion is an alternative to the WHERE clause above
  ** so that COUNT(*) returns the correct value. TODO Even though, as noted
  ** above, getting the count ahead of time is of little use (it was used to
  ** bump the display limit if only one entry would be omitted), I'll probably
  ** retain omitting the WHERE, and using DELETE FROM to exclude reserved
  ** names, just in case (c) above was implemented.
  */
  /*TODO deletions from clean_cmd() */
  if( file_tree_name(g.zRepositoryName, &repo, 0, 0) ){
    db_multi_exec("DELETE FROM sfile WHERE pathname=%B", &repo);
  }
  db_multi_exec("DELETE FROM sfile WHERE pathname IN"
                " (SELECT pathname FROM vfile)");
  /*TODO Delete reserved names, rather than WHERE them out. */
  db_multi_exec("DELETE FROM sfile WHERE pathname IN (%s)",
                fossil_all_reserved_names(0));

  /*TODO
  ** If we had a count of matching entries before scanning, this is where
  ** we'd bump the maximum to show so as to avoid "plus 1 file".
  ** ...
  ** If there's only one more than the maximum, let it through...
  ** a line used to say "plus 1 more" may as well display that item!
  if( nExtras == maxExtrasToShow+1 ){ maxExtrasToShow++; }
  ** ...
  */

  /* Handle the special case where zExtra was empty (and got converted to zero).
  ** If so, show "plain" files (those not matching any glob-list) but with an
  ** upper limit to the number shown (set above). If a value WAS given (i.e.
  ** after following a link), display all of the selected entries.
  */
  if( extrasFlags==0 ){
    extrasFlags = EX_PLAIN;
  }else{
    maxExtrasToShow = 0x7fffffff; /* Basically, all of them... */
  }

  /* Describe the files listed. Currently, the only "built-in" options are to







|
|









|













|








|

|



|









|







|

|





|



|












|
<







897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987

988
989
990
991
992
993
994
  zCleanFlag = db_get("clean-glob", 0);   /* ...that "clean" clobbers */
  zKeepFlag = db_get("keep-glob", 0);     /* ...that "clean" won't touch */
  pIgnore = glob_create(zIgnoreFlag);     /* Object versions of above */
  pKeep = glob_create(zKeepFlag);
  pClean = glob_create(zCleanFlag);
  nRoot = (int)strlen(g.zLocalRoot);      /* Length of root component */
  Stmt q;
  Blob repo;                              /* TODO:LD May not be needed */
  int maxExtrasToShow = LOCAL_DIFF_MAX_EXTRAS;
  int extrasFlags = atoi(zExtra);         /* Which entries to show */
  int nExtras;
  int nMatch;
  int nShown;
  int nPlain;
  int nIgnore;
  int nClean;
  int nKeep;

  /*TODO:LD?
  ** It feels sensible to limit the number of "extra" entries shown by default
  ** for cases where "ignore-glob" or "clean-glob" haven't been fully setup.
  ** A minor irritation is that this can lead to "... plus 1 more file", on a
  ** line that COULD have been used to display the omitted file. If we knew in
  ** advance how many entries were going to match, we could temporarily "bump"
  ** the limit by one show all entries would be shown. However, to know the
  ** number of matches in advance we'd have to:
  ** a) Pre-scan SFILE, testing and counting matches against each glob-list,
  **    possibly bump the limit, then re-scan the table repeating the tests
  **    against each glob-list to decide which to show.
  ** b) SFILE could have an extra FLAGS field: during the pre-scan, this could
  **    be updated to indicate which groups each file belong to. This would
  **    save re-testing every file against each glob-list (the main pass could
  **    select "WHERE flags & selector" to get only the matching entries), but
  **    the updates (selecting by "pathname" each time) could be a bit much.
  ** c) vfile_scan() -- where SFILE is populated -- COULD have an option to
  **    do the testing at the time entries are added. This would be the "best"
  **    way, but feels too much disruption to other code for what is only a
  **    minor benefit.
  ** For now, I'll stick with the minor annoyance of "plus 1 more file" :-)
  **
  ** Being able to determine the counts up-front would also allow us to hide
  ** the whole "extras report" if there were no unmanaged files.
  **
  **TODO:LD?
  ** Does it make sense (and/or is it practiable) to offer an "ADD" button
  ** against files that are unmanaged?
  **
  **TODO:LD?
  ** Does it make sense (and/or ...) to offer ediing of the various blob-lists
  ** from the Extras report? Showing the existing configuration screen would
  ** probably not be a problem (permissions permitting), but what happens if
  ** those settings have been overriden by .fossil-settings/ignore-glob? As we
  ** have access to the local checkout, is it feasible to edit it in the browser
  ** (perhaps piggy-backing /fileedit)?
  */

  locate_unmanaged_files(0, NULL, 0, NULL);   /* Get all unmanaged */
  /*TODO:LD
  ** The first two of these exclusions come from clean_cmd() in checkin.c.
  ** Not sure exactly what they are intended to do (seem to have no effect on
  ** my test repos). Last exclusion is an alternative to the WHERE clause above
  ** so that COUNT(*) returns the correct value. TODO Even though, as noted
  ** above, getting the count ahead of time is of little use (it was used to
  ** bump the display limit if only one entry would be omitted), I'll probably
  ** retain omitting the WHERE, and using DELETE FROM to exclude reserved
  ** names, just in case (c) above were to be implemented.
  */
  /*TODO:LD deletions from clean_cmd() */
  if( file_tree_name(g.zRepositoryName, &repo, 0, 0) ){
    db_multi_exec("DELETE FROM sfile WHERE pathname=%B", &repo);
  }
  db_multi_exec("DELETE FROM sfile WHERE pathname IN"
                " (SELECT pathname FROM vfile)");
  /*TODO:LD Delete reserved names, rather than WHERE them out. */
  db_multi_exec("DELETE FROM sfile WHERE pathname IN (%s)",
                fossil_all_reserved_names(0));

  /*TODO:LD
  ** If we had a count of matching entries before scanning, this is where
  ** we'd bump the maximum to show so as to avoid "plus 1 file".
  ** ...
  ** If there's only one more than the maximum, let it through...
  ** a line used to say "plus 1 more" may as well display that item!
  if( nExtras == maxExtrasToShow+1 ){ maxExtrasToShow++; }
  ** ...
  */

  /* Handle the special case where zExtra was empty (and got converted to zero).
  ** If so, show "plain" files (those not matching any glob-list) but with an
  ** upper limit to the number shown (set above). If a value WAS given (i.e.
  ** after following a link), display all of the selected entries. */

  if( extrasFlags==0 ){
    extrasFlags = EX_PLAIN;
  }else{
    maxExtrasToShow = 0x7fffffff; /* Basically, all of them... */
  }

  /* Describe the files listed. Currently, the only "built-in" options are to
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
  }else{
    Blob desc;
    blob_zero(&desc);
    if( extrasFlags & EX_PLAIN  ){ blob_appendf(&desc, " + unmanaged"    ); }
    if( extrasFlags & EX_IGNORE ){ blob_appendf(&desc, " + ignored"      ); }
    if( extrasFlags & EX_CLEAN  ){ blob_appendf(&desc, " + to be cleaned"); }
    if( extrasFlags & EX_KEEP   ){ blob_appendf(&desc, " + to be kept"   ); }
    if( blob_size(&desc) > 3 ){   /* Should never fail... */
      /* Add the string built above, skipping the leading " + " */
      @ (%h(blob_str(&desc)+3))
    }
    blob_reset(&desc);
  }
  @ </b></p>

  db_prepare(&q,
      "SELECT %Q || pathname FROM sfile"
      " ORDER BY 1",  //TODO Order by pathname?
      g.zLocalRoot
  );
  /*
  ** Put the file-list in one paragraph with line-breaks between.
  **TODO
  ** Might a table (with columns for name, ignore/clean/keep) work?
  */
  @ <p>
  nExtras = nMatch = nShown = nPlain = nKeep = nClean = nIgnore = 0;
  while( db_step(&q)==SQLITE_ROW ){
    const char *zName = db_column_text(&q, 0);
    int entryFlags = 0







|









|




|







1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
  }else{
    Blob desc;
    blob_zero(&desc);
    if( extrasFlags & EX_PLAIN  ){ blob_appendf(&desc, " + unmanaged"    ); }
    if( extrasFlags & EX_IGNORE ){ blob_appendf(&desc, " + ignored"      ); }
    if( extrasFlags & EX_CLEAN  ){ blob_appendf(&desc, " + to be cleaned"); }
    if( extrasFlags & EX_KEEP   ){ blob_appendf(&desc, " + to be kept"   ); }
    if( blob_size(&desc) > 3 ){         /* Should never fail... */
      /* Add the string built above, skipping the leading " + " */
      @ (%h(blob_str(&desc)+3))
    }
    blob_reset(&desc);
  }
  @ </b></p>

  db_prepare(&q,
      "SELECT %Q || pathname FROM sfile"
      " ORDER BY 1",  /*TODO:LD Order by pathname, as for differences? */
      g.zLocalRoot
  );
  /*
  ** Put the file-list in one paragraph with line-breaks between.
  **TODO:LD
  ** Might a table (with columns for name, ignore/clean/keep) work?
  */
  @ <p>
  nExtras = nMatch = nShown = nPlain = nKeep = nClean = nIgnore = 0;
  while( db_step(&q)==SQLITE_ROW ){
    const char *zName = db_column_text(&q, 0);
    int entryFlags = 0
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
  ** "1" would list ALL unmanaged files without limiting their number).
  **
  ** If the "ef=" isn't present (as when first navigating to "/local") then a
  ** default setting is used. Set to "0" to initially hide the report.
  */
  zExtra = P("ef");
  if( zExtra==NULL ) {
    zExtra = ""; /*TODO Take the default form a config. option? */
  }
  showExtras = strcmp(zExtra,"0")!=0;

  /* Local mode is selected by either "/local" or with a "name" of "ckout".
  ** First, check we have access to the checkout (and report to the user if we
  ** don't), then refresh the "vfile" table (recording which files in the
  ** checkout have changed etc.). We then change the "name" parameter to "tip"
  ** so that the "header" section displays info about the check-in that the
  ** checkout came from.
  **TODO
  ** It would probably make sense to limit "/local" (and other links that come
  ** from it) to only be permitted when Fossil is running locally in "ui" mode.
  ** It's probably not critical when all you can do is view files in the
  ** checkout (they can already see the checked-in versions), but if a COMMIT
  ** option WERE ever to be implemented, you wouldn't essentially random people
  ** on the internet firing off commits.
  */
  bLocalMode = (g.zPath[0]=='l') || (fossil_strcmp(zName,"ckout")==0);
  if( bLocalMode ){
    vid = g.localOpen ? db_lget_int("checkout", 0) : 0;
    if( vid==0 ){
      /*TODO Is this the right response? */
      style_header("No Local Checkout");
      @ No access to local checkout.
      style_footer();
      return;
    }
    vfile_check_signature(vid, CKSIG_ENOTFILE);
    zName = "tip";







|









|





|





|







1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
  ** "1" would list ALL unmanaged files without limiting their number).
  **
  ** If the "ef=" isn't present (as when first navigating to "/local") then a
  ** default setting is used. Set to "0" to initially hide the report.
  */
  zExtra = P("ef");
  if( zExtra==NULL ) {
    zExtra = LOCAL_DIFF_EXTRAS_MODE;
  }
  showExtras = strcmp(zExtra,"0")!=0;

  /* Local mode is selected by either "/local" or with a "name" of "ckout".
  ** First, check we have access to the checkout (and report to the user if we
  ** don't), then refresh the "vfile" table (recording which files in the
  ** checkout have changed etc.). We then change the "name" parameter to "tip"
  ** so that the "header" section displays info about the check-in that the
  ** checkout came from.
  **TODO:LD
  ** It would probably make sense to limit "/local" (and other links that come
  ** from it) to only be permitted when Fossil is running locally in "ui" mode.
  ** It's probably not critical when all you can do is view files in the
  ** checkout (they can already see the checked-in versions), but if a COMMIT
  ** option WERE ever to be implemented, you wouldn't essentially random people
  ** on the internet firing off commits!
  */
  bLocalMode = (g.zPath[0]=='l') || (fossil_strcmp(zName,"ckout")==0);
  if( bLocalMode ){
    vid = g.localOpen ? db_lget_int("checkout", 0) : 0;
    if( vid==0 ){
      /*TODO:LD Is this the right response? */
      style_header("No Local Checkout");
      @ No access to local checkout.
      style_footer();
      return;
    }
    vfile_check_signature(vid, CKSIG_ENOTFILE);
    zName = "tip";
1521
1522
1523
1524
1525
1526
1527
1528
1529
1530
1531
1532
1533
1534
1535
1536
1537
1538
1539
1540
1541
1542
1543
1544
1545
1546
1547
1548
1549
1550
  }
  if( bLocalMode ){
    if( showExtras ){
      @ %z(chref("button","%R/local?diff=%d%s&ef=0",diffType,zW))Hide Extras</a>
    }else{
      @ %z(chref("button","%R/local?diff=%d%s&ef=",diffType,zW))Show Extras</a>
    }
    /*TODO
    ** There would be a fair chunk of stuff to get right (not least appropriate
    ** restrictions), but it MIGHT be nice to have a COMMIT button here...
    */
  }
  @</div>
  if( pRe ){
    @ <p><b>Only differences that match regular expression "%h(zRe)"
    @ are shown.</b></p>
  }
  if( bLocalMode ){
    if( showExtras ){
      append_extras_report(zExtra, diffType, zW);
    }
    /* Following SQL taken from diff_against_disk() in diffcmd.c */
    /*TODO
    ** That code wrapped the query/processing in a transaction (but, from
    ** memory, other similar uses did not). Is it neeeded?
    */
    db_begin_transaction();
    db_prepare(&q3,
      "SELECT pathname, deleted, chnged , rid==0, rid, islink"
      "  FROM vfile"







|














|







1533
1534
1535
1536
1537
1538
1539
1540
1541
1542
1543
1544
1545
1546
1547
1548
1549
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
  }
  if( bLocalMode ){
    if( showExtras ){
      @ %z(chref("button","%R/local?diff=%d%s&ef=0",diffType,zW))Hide Extras</a>
    }else{
      @ %z(chref("button","%R/local?diff=%d%s&ef=",diffType,zW))Show Extras</a>
    }
    /*TODO:LD
    ** There would be a fair chunk of stuff to get right (not least appropriate
    ** restrictions), but it MIGHT be nice to have a COMMIT button here...
    */
  }
  @</div>
  if( pRe ){
    @ <p><b>Only differences that match regular expression "%h(zRe)"
    @ are shown.</b></p>
  }
  if( bLocalMode ){
    if( showExtras ){
      append_extras_report(zExtra, diffType, zW);
    }
    /* Following SQL taken from diff_against_disk() in diffcmd.c */
    /*TODO:LD
    ** That code wrapped the query/processing in a transaction (but, from
    ** memory, other similar uses did not). Is it neeeded?
    */
    db_begin_transaction();
    db_prepare(&q3,
      "SELECT pathname, deleted, chnged , rid==0, rid, islink"
      "  FROM vfile"
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
1569
1570
1571
1572

1573
1574
1575
1576
1577
1578
1579
1580
1581
1582
1583
1584

1585
1586
1587



1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
    ** differences. If enabled, the first pass will only show one-line entries
    ** and the second pass will only show those with diff-blocks. This has the
    ** side-effect of altering the order entries are shown in (but within each
    ** group the original order is maintained).
    **
    ** If disabled, (pass gets set to 3), only one pass is made on which all
    ** entries are shown in their "normal" order.
    **TODO
    ** Add this to the original (non-local) loop?
    */
//--------------------------------------------------- TODO TESTING HACK TODO
    int bTwoPass = P("op")==NULL; //TODO Taken from a config option?
//--------------------------------------------------- TODO TESTING HACK TODO
    int pass = bTwoPass?1:3;

    do{
      while( db_step(&q3)==SQLITE_ROW ){
        const char *zPathname = db_column_text(&q3,0);
        int isDeleted = db_column_int(&q3, 1);
        int isChnged = db_column_int(&q3,2);
        int isNew = db_column_int(&q3,3);
        int srcid = db_column_int(&q3, 4);
        int isLink = db_column_int(&q3, 5);
        char *zUuid = db_text(0, "SELECT uuid FROM blob WHERE rid=%d", srcid);
        append_local_file_change_line( zPathname, zUuid,
                      isDeleted, isChnged, isNew, isLink, diffFlags,pRe,pass );
        free(zUuid);

      }
      db_reset(&q3);
    }while( ++pass < 3 ); /* Either "1, 2, stop" or "3, stop". */



    db_finalize(&q3);
    db_end_transaction(1);  /* ROLLBACK */
  }else{ /* Normal, non-local-mode: show diffs against parent */
    /*TODO: Implement the optional two-pass code? */
    db_prepare(&q3,
      "SELECT name,"
      "       mperm,"
      "       (SELECT uuid FROM blob WHERE rid=mlink.pid),"
      "       (SELECT uuid FROM blob WHERE rid=mlink.fid),"
      "       (SELECT name FROM filename WHERE filename.fnid=mlink.pfnid)"
      "  FROM mlink JOIN filename ON filename.fnid=mlink.fnid"







|


<
<
<
|
>












>



>
>
>



|







1571
1572
1573
1574
1575
1576
1577
1578
1579
1580



1581
1582
1583
1584
1585
1586
1587
1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599
1600
1601
1602
1603
1604
1605
1606
1607
1608
1609
1610
1611
1612
    ** differences. If enabled, the first pass will only show one-line entries
    ** and the second pass will only show those with diff-blocks. This has the
    ** side-effect of altering the order entries are shown in (but within each
    ** group the original order is maintained).
    **
    ** If disabled, (pass gets set to 3), only one pass is made on which all
    ** entries are shown in their "normal" order.
    **TODO:LD
    ** Add this to the original (non-local) loop?
    */



    int pass = LOCAL_DIFF_USE_TWO_PASSES?1:3;
    int anyDifferences = 0;
    do{
      while( db_step(&q3)==SQLITE_ROW ){
        const char *zPathname = db_column_text(&q3,0);
        int isDeleted = db_column_int(&q3, 1);
        int isChnged = db_column_int(&q3,2);
        int isNew = db_column_int(&q3,3);
        int srcid = db_column_int(&q3, 4);
        int isLink = db_column_int(&q3, 5);
        char *zUuid = db_text(0, "SELECT uuid FROM blob WHERE rid=%d", srcid);
        append_local_file_change_line( zPathname, zUuid,
                      isDeleted, isChnged, isNew, isLink, diffFlags,pRe,pass );
        free(zUuid);
        anyDifferences = 1;
      }
      db_reset(&q3);
    }while( ++pass < 3 ); /* Either "1, 2, stop" or "3, stop". */
    if( !anyDifferences ){
      @ <p>No changes in the local checkout.</p>
    }
    db_finalize(&q3);
    db_end_transaction(1);  /* ROLLBACK */
  }else{ /* Normal, non-local-mode: show diffs against parent */
    /*TODO:LD: Implement the optional two-pass code? */
    db_prepare(&q3,
      "SELECT name,"
      "       mperm,"
      "       (SELECT uuid FROM blob WHERE rid=mlink.pid),"
      "       (SELECT uuid FROM blob WHERE rid=mlink.fid),"
      "       (SELECT name FROM filename WHERE filename.fnid=mlink.pfnid)"
      "  FROM mlink JOIN filename ON filename.fnid=mlink.fnid"
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
1654
  int vid;

  login_check_credentials();
  if( !g.perm.Read ){ login_needed(g.anon.Read); return; }

  vid = g.localOpen ? db_lget_int("checkout", 0) : 0;
  if( vid==0 ){
    /*TODO Is this the right response? */
    style_header("No Local Checkout");
    @ No access to local checkout.
    style_footer();
    return;
  }
  vfile_check_signature(vid, CKSIG_ENOTFILE);

  cgi_set_content_type("text/plain");

  db_begin_transaction();
  /*TODO
  ** This query is the same as in ci_page() for local-mode (as well as in
  ** diff_against_disk() in diffcmd.c, where it was originally taken from).
  ** Should they be "coalesced" in some way?
  */
  db_prepare(&q3,
    "SELECT pathname, deleted, chnged , rid==0, rid, islink"
    "  FROM vfile"







|










|







1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
  int vid;

  login_check_credentials();
  if( !g.perm.Read ){ login_needed(g.anon.Read); return; }

  vid = g.localOpen ? db_lget_int("checkout", 0) : 0;
  if( vid==0 ){
    /*TODO:LD Is this the right response? */
    style_header("No Local Checkout");
    @ No access to local checkout.
    style_footer();
    return;
  }
  vfile_check_signature(vid, CKSIG_ENOTFILE);

  cgi_set_content_type("text/plain");

  db_begin_transaction();
  /*TODO:LD
  ** This query is the same as in ci_page() for local-mode (as well as in
  ** diff_against_disk() in diffcmd.c, where it was originally taken from).
  ** Should they be "coalesced" in some way?
  */
  db_prepare(&q3,
    "SELECT pathname, deleted, chnged , rid==0, rid, islink"
    "  FROM vfile"
1973
1974
1975
1976
1977
1978
1979
1980
1981
1982
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
  if( diffType!=1 ){
    style_submenu_element("Unified Diff",
                          "%R/vdiff?%s&diff=1%s%T%s",
                          zQuery,
                          zGlob ? "&glob=" : "", zGlob ? zGlob : "", zW);
  }
  if( zBranch==0 ){
    //TODO Is there an extra "&" here?
    style_submenu_element("Invert",
                          "%R/vdiff?from=%T&to=%T&%s%T%s", zTo, zFrom,
                          zGlob ? "&glob=" : "", zGlob ? zGlob : "", zW);
  }
  if( zGlob ){
    //TODO Is there an extra "&" here?
    style_submenu_element("Clear glob",
                          "%R/vdiff?%s&%s", zQuery, zW);
  }else{
    style_submenu_element("Patch", "%R/vpatch?from=%T&to=%T%s", zFrom, zTo, zW);
  }
  if( diffType!=0 ){
    style_submenu_checkbox("w", "Ignore Whitespace", 0, 0);







|





|







1987
1988
1989
1990
1991
1992
1993
1994
1995
1996
1997
1998
1999
2000
2001
2002
2003
2004
2005
2006
2007
  if( diffType!=1 ){
    style_submenu_element("Unified Diff",
                          "%R/vdiff?%s&diff=1%s%T%s",
                          zQuery,
                          zGlob ? "&glob=" : "", zGlob ? zGlob : "", zW);
  }
  if( zBranch==0 ){
    //TODO:LD Is there an extra "&" here?
    style_submenu_element("Invert",
                          "%R/vdiff?from=%T&to=%T&%s%T%s", zTo, zFrom,
                          zGlob ? "&glob=" : "", zGlob ? zGlob : "", zW);
  }
  if( zGlob ){
    //TODO:LD Is there an extra "&" here?
    style_submenu_element("Clear glob",
                          "%R/vdiff?%s&%s", zQuery, zW);
  }else{
    style_submenu_element("Patch", "%R/vpatch?from=%T&to=%T%s", zFrom, zTo, zW);
  }
  if( diffType!=0 ){
    style_submenu_checkbox("w", "Ignore Whitespace", 0, 0);
2737
2738
2739
2740
2741
2742
2743
2744
2745
2746
2747
2748
2749
2750
2751
              g.zTop, zUuid);
      }else{
        style_submenu_element("Shun", "%s/shun?shun=%s#addshun", g.zTop, zUuid);
      }
    }
  }
  style_header("Hex Artifact Content");
  /* TODO
  ** Could the call to style_header() be moved so these two exclusion
  ** blocks could be merged? I don't think any of them make sense for
  ** a local file.
  */
  if( !bLocalMode ){
    zUuid = db_text("?","SELECT uuid FROM blob WHERE rid=%d", rid);
    @ <h2>Artifact







|







2751
2752
2753
2754
2755
2756
2757
2758
2759
2760
2761
2762
2763
2764
2765
              g.zTop, zUuid);
      }else{
        style_submenu_element("Shun", "%s/shun?shun=%s#addshun", g.zTop, zUuid);
      }
    }
  }
  style_header("Hex Artifact Content");
  /*TODO:LD
  ** Could the call to style_header() be moved so these two exclusion
  ** blocks could be merged? I don't think any of them make sense for
  ** a local file.
  */
  if( !bLocalMode ){
    zUuid = db_text("?","SELECT uuid FROM blob WHERE rid=%d", rid);
    @ <h2>Artifact
2768
2769
2770
2771
2772
2773
2774
2775
2776
2777
2778
2779
2780
2781
2782
  if( bLocalMode ){
      content_from_file(zLocalName, &content);
  }else{
    content_get(rid, &content);
  }
  @ <blockquote><pre>
  hexdump(&content);
  /* TODO: Should content (and downloadName?) be reset/freed? */
  @ </pre></blockquote>
  style_footer();
}

/*
** Look for "ci" and "filename" query parameters.  If found, try to
** use them to extract the record ID of an artifact for the file.







|







2782
2783
2784
2785
2786
2787
2788
2789
2790
2791
2792
2793
2794
2795
2796
  if( bLocalMode ){
      content_from_file(zLocalName, &content);
  }else{
    content_get(rid, &content);
  }
  @ <blockquote><pre>
  hexdump(&content);
  /* TODO:LD: Should content (and downloadName?) be reset/freed? */
  @ </pre></blockquote>
  style_footer();
}

/*
** Look for "ci" and "filename" query parameters.  If found, try to
** use them to extract the record ID of an artifact for the file.
3020
3021
3022
3023
3024
3025
3026

3027
3028
3029
3030
3031
3032
3033
    /* If there is no ci= query parameter, then prefer to interpret
    ** name= as a hash for /artifact and /whatis.  But for not for /file.
    ** For /file, a name= without a ci= while prefer to use the default
    ** "tip" value for ci=. */
    rid = name_to_rid(zName);
  }
  if( fossil_strcmp(zCI,"ckout")==0 ){

    bLocalMode = 1;
    rid = -1;     /* Dummy value to make it look found */
  }else if( rid==0 ){
    rid = artifact_from_ci_and_filename(0);
  }

  if( rid==0 ){  /* Artifact not found */







>







3034
3035
3036
3037
3038
3039
3040
3041
3042
3043
3044
3045
3046
3047
3048
    /* If there is no ci= query parameter, then prefer to interpret
    ** name= as a hash for /artifact and /whatis.  But for not for /file.
    ** For /file, a name= without a ci= while prefer to use the default
    ** "tip" value for ci=. */
    rid = name_to_rid(zName);
  }
  if( fossil_strcmp(zCI,"ckout")==0 ){
    /* "ci=ckout" is an extension for viewing files in the local checkout. */
    bLocalMode = 1;
    rid = -1;     /* Dummy value to make it look found */
  }else if( rid==0 ){
    rid = artifact_from_ci_and_filename(0);
  }

  if( rid==0 ){  /* Artifact not found */
3060
3061
3062
3063
3064
3065
3066
3067
3068
3069
3070
3071
3072
3073
3074
    objdescFlags |= OBJDESC_DETAIL;
  }
  zUuid = db_text("?", "SELECT uuid FROM blob WHERE rid=%d", rid);

  asText = P("txt")!=0;
  if( isFile ){
    if( bLocalMode ){
      /*TODO
      ** Is this the best way of handling annotations to the description?
      ** If "annot=message" is part of the URL, the message is appended
      ** to the description of the file. Only used for "local" files to
      ** distinguish such files from part of the repository.
      */
      const char *annot = P("annot");
      @ <h2>File %z(href("%R/finfo?name=%T&m=tip",zName))%h(zName)</a>







|







3075
3076
3077
3078
3079
3080
3081
3082
3083
3084
3085
3086
3087
3088
3089
    objdescFlags |= OBJDESC_DETAIL;
  }
  zUuid = db_text("?", "SELECT uuid FROM blob WHERE rid=%d", rid);

  asText = P("txt")!=0;
  if( isFile ){
    if( bLocalMode ){
      /*TODO:LD
      ** Is this the best way of handling annotations to the description?
      ** If "annot=message" is part of the URL, the message is appended
      ** to the description of the file. Only used for "local" files to
      ** distinguish such files from part of the repository.
      */
      const char *annot = P("annot");
      @ <h2>File %z(href("%R/finfo?name=%T&m=tip",zName))%h(zName)</a>
3210
3211
3212
3213
3214
3215
3216
3217
3218
3219
3220
3221
3222
3223
3224
3225


3226
3227
3228
3229
3230
3231
3232
    style_submenu_element("Parsed", "%R/info/%s", zUuid);
  }
  if( descOnly ){
    style_submenu_element("Content", "%R/artifact/%s", zUuid);
  }else{
    @ <hr />
    if( bLocalMode ){
      /*TODO
      ** Should we handle non-existent local files differently? Currently,
      ** they are shown the same as if the file was present but empty. This
      ** should never happen through "normal" operation, but someone might
      ** craft a link to one. Perhaps have content_from_file() perform an
      ** existence-check (rather than relying on blob_read_from_file() which
      ** it calls returning an empty blob)?
      */
      content_from_file(zName, &content);


    }else{
      content_get(rid, &content);
    }
    if( renderAsWiki ){
      wiki_render_by_mimetype(&content, zMime);
    }else if( renderAsHtml ){
      @ <iframe src="%R/raw/%s(zUuid)"







<
<
<
<
<
<
<
<
|
>
>







3225
3226
3227
3228
3229
3230
3231








3232
3233
3234
3235
3236
3237
3238
3239
3240
3241
    style_submenu_element("Parsed", "%R/info/%s", zUuid);
  }
  if( descOnly ){
    style_submenu_element("Content", "%R/artifact/%s", zUuid);
  }else{
    @ <hr />
    if( bLocalMode ){








      if( !content_from_file(zName, &content) ){
        fossil_warning("Cannot find/access %s.", zName);
      }
    }else{
      content_get(rid, &content);
    }
    if( renderAsWiki ){
      wiki_render_by_mimetype(&content, zMime);
    }else if( renderAsHtml ){
      @ <iframe src="%R/raw/%s(zUuid)"