summaryrefslogtreecommitdiff
path: root/plugins/rsync.lua
blob: 8f8c8da198a5dd62d55e5b4b35df03f7131bde40 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
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
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
-- rsync Plugin
--
-- This plugin is part of Trove.  Trove is Codethink's Baserock central server
-- and uses Gitano as the Git service.  This plugin adds support to Trove for
-- supporting 'rsync' as a command in Gitano.  This means that every repository
-- has an rsync tree attached to it which can be used by remote ends sshing
-- into the Trove.
--
-- Since the functionality is generically useful however, it is supported here
-- in Gitano upstream.
--
-- Copyright 2014 Daniel Silverstone <daniel.silverstone@codethink.co.uk>

local gitano = require "gitano"

local sp = require "luxio.subprocess"

local rsync_short_help = "An rsync endpoint within Gitano"
local rsync_helptext = [[
Users are not expected to use this command directly, but instead to
instruct their local rsync to use a remote of user@gitanohost:reponame
when pushing or pulling rsync content.
]]

local function rsync_detect_repo(config, cmdline)
   local repo, msg
   local repopath = cmdline[#cmdline]

   if #cmdline < 4 then
      -- Validate will fail with a nice error, give up now
      return nil, cmdline
   end

   if repopath:match("^/") then
      repopath = repopath:sub(2)
   end

   local origpath = repopath

   -- Basically, while there's still something to the repopath
   -- and we've not yet found a repo, strip an element and try again...
   while (not repo or repo.is_nascent) and repopath ~= ""do
      gitano.log.error("Trying " .. repopath)
      repo, msg = gitano.repository.find(config, repopath)
      if not repo or repo.is_nascent then
         repopath = repopath:match("^(.*)/[^/]*$") or ""
      end
   end

   if not repo then
      gitano.log.error("Unable to find a repository for " .. cmdline[#cmdline])
      return nil, nil
   end

   if repo.is_nascent then
      gitano.log.error("Repository " .. repo.name .. " is nascent")
      gitano.log.error("Cannot use rsync command with nascent repositories")
      return nil, nil
   end

   -- Okay, so repopath represented a repository, let's convert the path
   -- into something which we can work with...
   cmdline[#cmdline] = repo:fs_path() .. "/rsync" .. origpath:sub(#repopath+1)
   if origpath:match("/$") and not (cmdline[#cmdline]):match("/$") then
      cmdline[#cmdline] = cmdline[#cmdline] .. "/"
   end

   gitano.util.mkdir_p(repo:fs_path() .. "/rsync")

   -- And give back the repo for ruleset running and the cmdline for the rsync
   return repo, cmdline
end

local function rsync_validate(config, repo, cmdline)
   if #cmdline < 4 then
      gitano.log.error("usage: rsync --server <rsync arguments> . <server-side-path>")
      return false
   end
   if cmdline[2] ~= "--server" then
      gitano.log.error("Second cmdline element must always be --server")
      return false
   end
   return true
end

local function rsync_prep(config, repo, cmdline, context)
   if cmdline[3] == "--sender" then
      context.operation = "read"
   else
      context.operation = "write"
   end
   return repo:run_lace(context)
end

local function rsync_run(config, repo, cmdline, env)
   local cmdcopy = {env=env}
   for i = 1, #cmdline do cmdcopy[i] = cmdline[i] end
   local proc = sp.spawn(cmdcopy)
   return proc:wait()
end

assert(gitano.command.register("rsync",
                               rsync_short_help, rsync_helptext,
                               rsync_validate, rsync_prep, rsync_run,
                               true, true, false,
                               rsync_detect_repo))