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
107
108
109
110
111
112
113
114
115
116
117
|
-- @@SHEBANG
-- -*- Lua -*-
-- gitano-pre-receive-hook
--
-- Git (with) Augmented network operations -- pre-receive hook handler
--
-- Copyright 2012 Daniel Silverstone <dsilvers@digital-scurf.org>
--
--
-- @@GITANO_LUA_PATH
local gitano = require "gitano"
local gall = require "gall"
local luxio = require "luxio"
local sio = require "luxio.simple"
local sp = require "luxio.subprocess"
-- @@GITANO_BIN_PATH
-- @@GITANO_SHARE_PATH
local start_log_level = gitano.log.get_level()
-- Clamp level at info until we have checked if the caller
-- is an admin or not
gitano.log.cap_level(gitano.log.level.INFO)
gitano.log.syslog.open()
local repo_root = luxio.getenv("GITANO_ROOT")
local username = luxio.getenv("GITANO_USER") or "gitano/anonymous"
local keytag = luxio.getenv("GITANO_KEYTAG") or "unknown"
local project = luxio.getenv("GITANO_PROJECT") or ""
local source = luxio.getenv("GITANO_SOURCE") or "ssh"
-- Now load the administration data
gitano.config.repo_path(repo_root)
local admin_repo = gall.repository.new((repo_root or "") .. "/gitano-admin.git")
if not admin_repo then
gitano.log.fatal("Unable to locate administration repository. Cannot continue");
end
local admin_head = admin_repo:get(admin_repo.HEAD)
if not admin_head then
gitano.log.fatal("Unable to find the HEAD of the administration repository. Cannot continue");
end
local config, msg = gitano.config.parse(admin_head)
if not config then
gitano.log.critical("Unable to parse administration repository.")
gitano.log.critical(" * " .. (msg or "No error?"))
gitano.log.fatal("Cannot continue")
end
-- Now, are we an admin?
if config.groups["gitano-admin"].filtered_members[username] then
-- Yep, so blithely reset logging level
gitano.log.set_level(start_log_level)
end
if not config.global.silent then
-- Not silent, bump to chatty level automatically
gitano.log.bump_level(gitano.log.level.CHAT)
end
local repo, msg = gitano.repository.find(config, project)
if not repo then
gitano.log.critical("Unable to locate repository.")
gitano.log.critical(" * " .. (tostring(msg)))
gitano.log.fatal("Cannot continue")
end
if repo.is_nascent then
gitano.log.fatal("Repository " .. repo.name .. " is nascent")
end
-- pre-receive can prevent updates. Its name is a bit misleading.
-- pre-receive is called once all the objects have been pushed, but before the
-- individual update hooks are called. It gets the same input as post-receive
-- but can opt to reject the entire push. If you need to make decisions based
-- upon multiple refs being updated simultaneously, then this is the badger for
-- you.
local updates = {}
for oldsha, newsha, refname in
(sio.stdin:read("*a")):gmatch("([^ ]+) ([^ ]+) ([^\n]+)") do
gitano.log.ddebug("pre-receive:", oldsha, newsha, refname)
updates[refname] = {oldsha, newsha, oldsha=oldsha, newsha=newsha}
end
if repo:uses_hook("pre-receive") then
gitano.log.debug("Configuring for pre-receive hook")
gitano.actions.set_supple_globals("pre-receive")
local msg = "Running repository pre-receive hook"
gitano.log.info(msg)
gitano.log.syslog.info(msg)
local info = {
username = username,
keytag = keytag,
source = source,
realname = (config.users[username] or {}).real_name or "",
email = (config.users[username] or {}).email_address or "",
}
local ok, msg = gitano.supple.run_hook("pre-receive", repo, info, updates)
if not ok then
gitano.log.crit(msg or "No reason given, but errored somehow.")
end
gitano.log.info("Finished")
end
gitano.log.syslog.close()
return 0
|