SVN

From Pickwiki
Jump to navigationJump to search

HomePage>>SourceCode>>BasicSource>>SvnWrapper>>SVN

SVN Source code; once all source is compiled and cataloged, 'SVN -H' will show usage.

* ===================================== SVN =============================================
* @TCL: SVN wrapper, to be run on the UV prompt
* =======================================================================================
* 04.Oct.2013 james: Check /released/ folder too when doing STATUS
* 20.Dec.2012 james: Add CATALOG
* 07.Sep.2012 james: New 'COMMENTS' option to see the latest comment for each program.
* 12.May.2011 james: Handle 'DICT/' in release (flist) files.
* 10.Aug.2010 james: Add 'COMPILE' option to check a release before it goes out.
* 07.Oct.2009 james: Add 'UPDATE'; only add -m if '[[RequiresComment]]'.. IS THIS OK !?
* 30.Jun.2009 james: Add '[[CheckReleaseStatus]]' to make sure nothing was committed after
*                    the Release's Revision Number.
* 17.Nov.2008 chrisg: Added support for CLEANUP command
* 12.Jul.2008 chrisg: Added support for SVN PROPSET command
* 22.Apr.2008 manoj: Fixed a bug that limit the length of a new comment to the comment on 
*                    line 4
* 11.Apr.2008 jim: Use the line 4 comment for the [[CommitComment]] ONLY IF there's one item.
* 11.Feb.2008 jim: [[RequiresFileList]] = @FALSE for DELETE and STATUS; so those still work
*                  if files are in SVN but not in UV.
* 14.Jan.2008 manoj: Added DELETE
* 10.Jan.2008 manoj: Added revert
* 02.Jan.2008 jim: Restructure to improve error handling, etc.
* 31.Aug.2007 manoj: Created
* =======================================================================================
* $Id: SVN 44131 2013-11-12 02:57:36Z jim $
* =======================================================================================
*
* This invokes the svn command, with the proper svn URL and REAL.USER name
*
* Ex:
*
*  svn commit svn+ssh://[email protected]/opt/svn/dsi[[/Database/Universe/trunk/dsisrc]]
*
* Trac Reference: http://trac/dsi/wiki/UniverseInSvn
*
* =======================================================================================

  $include RMS.BP EQU.CMDLINE

  equate SVN$CMD          to "/usr/bin/svn"

  deffun GET.REALUSER( Err )
  deffun OPEN.FILE( FILENAME, FLAGS, FILEVAR, ERR.MSG )

  username = GET.REALUSER( Err ) ; if Err # '' then stopm Err  ;*  Err not yet implemented anyway
  logFile  = "log[[/SVN]].LOG"

  gosub [[ParseCommandLine]]
  gosub [[PrepareSvnCommand]]

  if Err # '' then
    gosub [[DisplayErrors]]
  end else

    gosub [[ExecuteSvnCommand]]
    gosub [[ExecutePostSvnCommands]]

  end
return

* =======================================================================================

[[PrepareSvnCommand]]:
  [[RequiresFileList]] = @FALSE
  [[RequiresComment]]  = @FALSE
  * Also add svnOp below (in the [[ShowHelp]] )
  begin case
    case svnOp = "COMMENTS" ; [[RequiresFileList]] = @TRUE  ;*  NOT REALLY AN SVN COMMAND BUT IT'S HANDY !
    case svnOp = "BASIC"    ; [[RequiresFileList]] = @TRUE  ;*  NOT REALLY AN SVN COMMAND BUT IT'S HANDY !
    case svnOp = "COMPILE"  ; [[RequiresFileList]] = @TRUE  ;*  NOT REALLY AN SVN COMMAND BUT IT'S HANDY !
    case svnOp = "CATALOG"  ; [[RequiresFileList]] = @TRUE  ;*  NOT REALLY AN SVN COMMAND BUT IT'S HANDY !
    case svnOp = "COMMIT"   ; [[RequiresFileList]] = @TRUE  ; [[RequiresComment]] = @TRUE
    case svnOp = "ADD"      ; [[RequiresFileList]] = @TRUE
    case svnOp = "UPDATE"   ; [[RequiresFileList]] = @TRUE  ;*  Is this dangerous !? Just for messed up SVN !?!
    case svnOp = "DELETE"   ; [[RequiresFileList]] = @FALSE ;*  Allow delete from SVN with no UV item ?
    case svnOp = "DIFF"     ; [[RequiresFileList]] = @TRUE
    case svnOp = "INFO"     ; [[RequiresFileList]] = @TRUE
    case svnOp = "LISTS"    ; [[RequiresFileList]] = @TRUE  ;*  NOT REALLY AN SVN COMMAND BUT IT'S HANDY !
    case svnOp = "LOG"      ; [[RequiresFileList]] = @TRUE
    case svnOp = "REVERT"   ; [[RequiresFileList]] = @TRUE
    case svnOp = "PROPSET"  ; [[RequiresFileList]] = @TRUE
    
      if [[PropertyNameAndValue]] = '' then 
        Err = svnOp:" requires -P, see help"; return
      end else
        [[PropertyNameAndValue]] = EREPLACE([[PropertyNameAndValue]], "=", " ")
      end

    case svnOp = "STATUS"  ; * [[FileList]] is NOT required; but use it if it's there.

      if [[FileList]] = '' then
        if [[FileName]] # '' then
          [[FileList]] = [[FileName]]
        end else
          [[FileList]] = "DSI.BP DA.BP WEB.BP RMS.BP"
        end
      end else Verbose = @TRUE ;*  Force Verbose if you list specific items.
      if Verbose then [[FileList]] := " -v"  ;*  Always VERBOSE (show lines NOT changed too...)
      [[CommitComment]] = ''   ;*  May have been set from '[[ReadRelease]]' ?
      
    case svnOp = "CLEANUP" ; [[FileList]] = [[FileName]]
    case svnOp = "HELP"    ; [[FileList]] = downcase([[FileName]]) ;*  NO options required; but may have 2nd parameter
    case 1
      Err = "Invalid SVN Operation '":svnOp
  end case

  if [[FileErrs]] # '' then crt [[FileErrs]] ; input j

  if [[RequiresFileList]] then
    Err = [[FileErrs]] ; if Err # '' then return
    if [[FileList]] = '' then Err = svnOp:" requires a File and Item name" ; return
  end

  if [[RequiresComment]] and [[CommitComment]] = '' then
    if CMD.NUM.ARGS = 3 then  ;*  Only 1 file to commit: use it's comment (Tacky ?) :
      [[FileName]] = CMD.ARG.S<1, 2>
      [[ItemID]]   = CMD.ARG.S<1, 3>
      open [[FileName]] to File then
        read [[ProgramItem]] from File, [[ItemID]] then
          [[CommentLine]] = [[ProgramItem]]<4>
          [[CommentUser]] = field( [[CommentLine]], ' ', 3 )
          [[CommitComment]] = [[CommentLine]][ col2() + 1, 999 ]
        end else [[CommitComment]] = ""  ;*  Why would this be ? Throw an error ?
      end
    end
    gosub [[GetCommitComment]]
    if [[CommitComment]] = "" then Err = "Cannot ":svnOp:" with an empty comment"
  end
return

[[ExecuteSvnCommand]]:
  *  Force the username by passing to the "sudo" command below
  svnCommand = SVN$CMD:' ':downcase(svnOp):' ':[[PropertyNameAndValue]]:' ':[[FileList]]
  if [[RequiresComment]] and [[CommitComment]] # '' then
    svnCommand := ' -m "':[[CommitComment]]:'"'
  end

  begin case
    case svnOp = "COMMENTS"  ;*  Special hack: just to use the code that reads the Release

      crt
      crt " (comments are always from the CURRENT version...)"
      crt
      [[FileList]] = trim( [[FileList]] )
      [[NumFiles]] = dcount([[FileList]], ' ')

      [[LongestName]] = 0
      for N.F = 1 to [[NumFiles]]
        [[OneProgram]] = field( [[FileList]], ' ', N.F )
        if len([[OneProgram]]) > [[LongestName]] then [[LongestName]] = len([[OneProgram]])
      next N.F
      [[NameFmt]] = 'L#':[[LongestName]]

      for N.F = 1 to [[NumFiles]]
        [[OneProgram]] = field( [[FileList]], ' ', N.F )
        [[ProgramFile]] = field( [[OneProgram]], '/', 1)
        [[ProgramID]]   = field( [[OneProgram]], '/', 2)
        if not( OPEN.FILE( [[ProgramFile]], '', PROGRAM.FILE,  Err ) ) then stopm Err
        read [[ProgramCode]] from PROGRAM.FILE, [[ProgramID]] else crt "No item named:  ":[[ProgramID]] ; return
        crt [[OneProgram]] [[NameFmt]]:' ':[[ProgramCode]]<4>
      next N.F
      crt

    *  Special hack: just to use the code that reads the Release
    case svnOp = "COMPILE" or svnOp = "BASIC" or svnOp = "CATALOG"

      [[FileList]] = trim( [[FileList]] )
      [[NumFiles]] = dcount([[FileList]], ' ')
      for N.F = 1 to [[NumFiles]]
        [[OneProgram]] = field( [[FileList]], ' ', N.F )
        convert '/' to ' ' in [[OneProgram]]
        if svnOp = "COMPILE" or svnOp = "BASIC" then [[TclCommand]] = "BASIC" else [[TclCommand]] = "CATALOG"
        execute [[TclCommand]]:" ":[[OneProgram]]
      next N.F

    case svnOp = "LISTS"  ;*  Special hack: just to use the code that reads the Release

      gosub [[SaveProgramLists]]

    case 1

      svnScript  = "(cd /u/dsisrc; sudo -u ":username:" ":svnCommand:" )"
      call APPEND.UNIX.LOG (svnScript, logFile, aulErr)

      if [[DisplayCmd]] then
        crt svnScript ; crt
        input j ; j = upcase(j)
        if j = 'X' or j = 'Q' then stop
      end

      if not([[NoExecute]]) then
        execute "SH -c \":svnScript:"\" capturing [[SvnResponse]]
      end
      [[OutputResponse]] = [[SvnResponse]]
      convert @FM to char(10) in [[OutputResponse]]
      crt [[OutputResponse]]

  end case

return

[[ExecutePostSvnCommands]]:
  *  Now do anything special we need 'post SVN' :
  begin case

    case svnOp = "DIFF"
      if [[SvnResponse]] = '' then crt "The current version does not differ from what's checked in." ; crt

    case svnOp = "COMMIT"
      if [[ReleaseNumber]] # '' then  ;*  Separate 'if' because [[SvnLine]] will only be defined for -R :
        if [[SvnLine]] # 0 then       ;*  Release needs the SVN number:
          gosub [[AddReleaseNumber]]
        end
      end

    case svnOp = "STATUS"
      if [[ReleaseNumber]] # '' then  ;*  [[SvnLine]] will only be defined for -R :
        gosub [[CheckReleaseStatus]]
      end

  end case
return

[[SaveProgramLists]]:
  if not(OPEN.FILE("&SAVEDLISTS&", "", SAVEDLISTS.FILE, Err)) then stopm Err
  [[FileList]] = trim( [[FileList]] )
  [[NumFiles]] = dcount([[FileList]], ' ')
  F.P = 0 ; [[CurrentFile]] = '~' ; [[ItemList]] = ''
  loop
    F.P += 1
   while F.P <= [[NumFiles]] do
    [[OneProgram]] = field( [[FileList]], ' ', F.P )
    [[ProgramFile]] = field( [[OneProgram]], '/', 1 )
    [[ProgramItem]] = field( [[OneProgram]], '/', 2 )
    if [[ProgramFile]] # [[CurrentFile]] then
      gosub [[SaveLastList]]
      [[CurrentFile]] = [[ProgramFile]]
    end
    [[ItemList]]<-1> = [[ProgramItem]]
  repeat
  gosub [[SaveLastList]]
return

[[SaveLastList]]:
  if [[CurrentFile]] # '~' then
    [[ListName]] = [[CurrentFile]]:"_":[[ReleaseNumber]]
    crt "Writing list ":[[ListName]]
    write [[ItemList]] on SAVEDLISTS.FILE, [[ListName]]
    [[ItemList]] = ''
  end
return

* =======================================================================================

[[AddReleaseNumber]]:
  [[RevisionNumber]] = ''
  [[NumResponseLines]] = dcount( [[SvnResponse]], @FM )
  for N.R.L = [[NumResponseLines]] to 1 step -1  ;*  Go backward since it's at the end:
    [[ResponseLn]] = [[SvnResponse]]<N.R.L>
    if [[ResponseLn]][1,18] = "Committed revision" then
      [[RevisionNumber]] = trim( field([[ResponseLn]][19,999], '.', 1) )
    end
  next N.R.L
  if [[RevisionNumber]] # '' then
    [[ReleaseText]]<SvnLine> = "SVN: ":[[RevisionNumber]]
    convert @FM to char(10) in [[ReleaseText]]
    *  Was "/home/release/":[[ReleaseNumber]]:"/flist" but maybe in 'released' now?:
    call WRITE.UNIX( [[ReleaseText]], flistPath, Err)
  end
return

*  SVN STATUS -R xxxx : Check if the SVN # of each item is > the Release's #

[[CheckReleaseStatus]]:
  [[RevisionNumber]] = field( [[ReleaseText]]<SvnLine>, ' ', 2)
  if [[RevisionNumber]] = '' then return

  if [[ReleaseLocation]] # '' then crt ; crt [[ReleaseLocation]] ; crt

  crt ; crt "Checking that each file is before Revision ":[[RevisionNumber]]:'...' ; crt
  [[NumResponseLines]] = dcount( [[SvnResponse]], @FM )
  for N.R.L = 1 to [[NumResponseLines]]
    [[ResponseLn]] = [[SvnResponse]]<N.R.L>
    Revs = trim( [[ResponseLn]][12, 9999] )  ;*  Kind of a hack ? How else though ?
    [[LastCommitted]] = field(Revs, ' ', 2)
    if [[LastCommitted]] > [[RevisionNumber]] then
      Path = field( Revs, ' ', 4)
      crt Path:" has a Revision Number (":[[LastCommitted]]:") AFTER this Releases Revision # ":[[RevisionNumber]]:" ! (one-off Commit ?)"
    end
  next N.R.L
  crt
return

[[DisplayErrors]]:
  if Err # '' then
    crt
    crt 'ERROR ! : ':Err<1>
    [[NumErrs]] = dcount(Err, @FM)
    for N.E = 2 to [[NumErrs]]
      crt '          ':Err<N.E>
    next N.E
    crt
    crt "   ('SVN -H' for more info)"
    crt
  end
return

[[ReadRelease]]:
  [[ReleaseLocation]] = ''
  flistPath = "/home/release/":[[ReleaseNumber]]:"/flist"
  call READ.UNIX( [[ReleaseText]], flistPath, Err)
  if Err # '' then
    [[ReleaseLocation]] = ' ** RELEASED **'
    flistPath = "/home/release/released/":[[ReleaseNumber]]:"/flist"
    call READ.UNIX( [[ReleaseText]], flistPath, Err2)
    if Err2 # '' then
      crt Err ; crt Err2 ; input j ; return
    end
    Err = ''  ;*  Must be in 'released'... so we're good.
  end

  convert char(10):char(13) to @FM in [[ReleaseText]]
  [[SvnLine]] = 0 ;* Track the 'SVN:' line to append the # if we are Committing and there's not one already
  [[NumLines]] = dcount( [[ReleaseText]], @FM )
  for N.L = 1 to [[NumLines]]
    LN = trim( [[ReleaseText]]<N.L> )
    begin case
      case LN[1,2] = '**' and [[CommitComment]] = ''
        [[CommitComment]] = LN[3, 99999]       ;*  Special single comment line used for Commit
      case LN[1,1] = '*' or LN[1,1] = '#'  ;*  Ignore comments
      case index(LN, ':', 1) # 0           ;*  Handle special tags :
        if LN[1,4] = "SVN:" then           ;*  Is this the "SVN:" line ?
          [[SvnLine]] = N.L                    ;*  Remember where
        end
      case LN # ''                         ;*  Finally handle normal lines:
        Item = LN ; gosub [[AddItem]]
    end case
  next N.L
return

[[AddItem]]:
  [[FileName]] = field(Item, '/', 1) ; [[ItemID]] = field(Item, '/', 2)
  open [[FileName]] to File then
    [[FileList]] := Item:' '  ;*  Keep list because some things don't need to be in UV ?:
    read dummy from File, [[ItemID]] else
      [[FileErrs]]<-1> = [[ItemID]]:" is not in ":[[FileName]]
    end
  end else
    if [[FileName]] = 'DICT' then  ;*  Special case: make it act like a BP ?
      [[FileList]] := Item:' '     ;*  Just like above ?
    end else
      [[FileErrs]]<-1> = [[FileName]]:" is not a valid BP file/directory."
    end
  end
return

[[GetCommitComment]]:
  * Add spaces, to allow for the "new" comment to be longer than '[[CommitComment]]'. A MR.IN characteristic.
  commitLength = len([[CommitComment]])
  [[DisplayComment]] = [[CommitComment]]:spaces(100-commitLength)
  crt "Enter comment:": ; call MR.IN( [[DisplayComment]], '', OUT )
  if OUT = "ESC" or OUT = 'HOME' then stop
  [[CommitComment]] = trim([[DisplayComment]])
return

* =======================================================================================

*  Set all Option flags and set [[FileList]] and [[FileErrs]]. Note we don't handle the errors
*  now because some options don't require them anyway.

[[ParseCommandLine]]:
  [[CmdLineOptions]] = "SINGLEPARAM"  ;*  for -M option
  call PARSE.CMDLINE( @SENTENCE, mat CMDLINE, [[CmdLineOptions]] )
  if CMD.HELP then gosub Help ; stop
  svnOp = CMD.ARG.S<1, 1>
  [[FileName]] = CMD.ARG.S<1, 2>
  locate 'D' in CMD.FLAG.S<1> setting dum then [[DisplayCmd]] = @TRUE else [[DisplayCmd]] = @FALSE
  locate 'L' in CMD.FLAG.S<1> setting [[SelectListLoc]] then
    [[SelectListName]] = CMD.PARAM.S<1, [[SelectListLoc]], 1>
  end else [[SelectListName]] = ''
  locate 'N' in CMD.FLAG.S<1> setting dum then [[NoExecute]] = @TRUE else [[NoExecute]] = @FALSE
  locate 'M' in CMD.FLAG.S<1> setting [[MessagePos]] then
    [[CommitComment]] = CMD.PARAM.S<1, [[MessagePos]], 1>   ;*  'SINGLEPARAM' is set so there's only one line
  end else [[CommitComment]] = ''
  locate 'R' in CMD.FLAG.S<1> setting [[ReleasePos]] then
    [[ReleaseNumber]] = CMD.PARAM.S<1, [[ReleasePos]], 1>
  end else [[ReleaseNumber]] = ''
  locate 'V' in CMD.FLAG.S<1> setting dum then Verbose = @TRUE else Verbose = @FALSE
  locate 'P' in CMD.FLAG.S<1> setting [[PropertyNameAndValueLoc]] then
    [[PropertyNameAndValue]] = CMD.PARAM.S<1, [[PropertyNameAndValueLoc]], 1>
  end else [[PropertyNameAndValue]] = ''
  [[FileList]]  = ""  ;*  Space separated; each item is a path from /u/dsisrc :
  [[FileErrs]]  = ""

  begin case      ;*  Lots of ways to get the [[FileList]]:
    case [[ReleaseNumber]]  # '' ; gosub [[ReadRelease]]
    case [[SelectListName]] # '' ;* Use the SELECT list if provided
      getlist [[SelectListName]] else Err = "Unable to read Select List ": [[SelectListName]] ; return
      loop while readnext [[ItemID]] do
        Item = [[FileName]]:'/':[[ItemID]] ; gosub [[AddItem]]
      repeat
    case 1   ;* Get the individual files if no "-L" option was specified
      for N.A = 3 to CMD.NUM.ARGS   ;* We get the files as arguments 3 and more
        Item = [[FileName]]:'/':CMD.ARG.S<1,N.A> ; gosub [[AddItem]]
      next N.A
  end case

  if svnOp = '' then gosub Help
return

Help:
  crt 'SVN <command> { <BP_Dir> <item> ... | -L <save-list> | -R <release#> } [-N][-M "message"][-V][-D][-P <[[PropName]]=PropVal>]'
  crt
  crt " where <command> is COMPILE, COMMIT, ADD, DELETE, DIFF, INFO, LOG, REVERT, STATUS, HELP, PROPSET. Options:"
  crt "  -N : No Execute; for testing just to see the <command>"
  crt "  -V : Verbose for STATUS; default for listed items; NOT default for 'all' option"
  crt "  -D : Display the <command> before executing it."
  crt "  -P : Add more arguments after <command>, useful for PROPSET (example)"
  crt
  crt " For 'COMMIT' when 'M is not used, you will be prompted for a comment if the release has no '**' comment."
  crt
  crt "Examples:"
  crt "  SVN COMMIT DSI.BP AR.FREEZE AR.UNFREEZE"
  crt "  SVN COMMIT DSI.BP -L DSI.BP.FLIST"
  crt "  SVN COMMIT -R 570                    : Commit EVERY program in Release 570 using the '**' line for the Comment"
  crt "  SVN COMPILE -R 570                   : Compile EVERY program in Release 570: NOT REALLY SVN but it's handy!"
  crt "  SVN LISTS   -R 570                   : Save Select Lists for program files: NOT REALLY SVN but it's handy!"
  crt "  SVN STATUS DSI.BP                    : Show all uncommitted 'M'odifications, 'D'eletes, 'A'dds, etc."
  crt "  SVN DIFF DSI.BP AR.FREEZE            : Show the difference between the local copy and what's in SVN"
  crt "  SVN INFO DSI.BP AR.FREEZE PMTS       : Show misc info about what's in SVN"
  crt "  SVN CLEANUP /u/dsisrc[[/DSI]].BP         : Execute cleanup on DSI.BP directory (if something is locked, etc)"
  crt "  SVN LOG DSI.BP AR.FREEZE             : List of historical changes"
  crt
  crt "To add a new file, you need to execute ADD first before Committing (i.e. standard 'svn' procedure):"
  crt "  SVN ADD    DSI.BP NEWFILENAME"
  crt "  SVN COMMIT DSI.BP NEWFILENAME"
  crt
  crt "To delete a file, you need to execute DELETE first, and then 'commit the delete' (again, standard svn):"
  crt "(note: this can be used if you get 'Already Exists' errors ? Just backup your files first !)"
  crt "  SVN DELETE DSI.BP ANYFILENAME"
  crt "  SVN COMMIT DSI.BP ANYFILENAME"
  crt
  crt "To set properties like svn:keywords: "
  crt "  SVN PROPSET DSI.BP ANYFILENAME -P svn:keywords=Id"   
  crt "  SVN PROPSET DSI.BP ANYFILENAME -P svn:keywords=Id Rev Date Author URL"
  crt
  crt "To FIX SVN if you get 'Out of Date' errors (!?) :"
  crt "  SVN UPDATE -R 570                    : UPDATE EVERY program in Release 570 to the current SVN #"
  crt "                                         (CAN THIS OVERWRITE YOUR CHANGES !?)"
  crt
  crt "For more svn options try: svn -help and svn <command> help"
  crt "(see also Confluence page: http://intranet.docmagic.com/display/Development/Universe+SVN+Integration)"
return