-
Notifications
You must be signed in to change notification settings - Fork 42
/
scripts.go
133 lines (113 loc) · 3.26 KB
/
scripts.go
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
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
package git
import (
"fmt"
"os"
"strings"
"sync"
"github.com/abiosoft/caddy-git/gitos"
)
var (
// gitBinary holds the absolute path to git executable
gitBinary string
// shell holds the shell to be used. Either sh or bash.
shell string
// initMutex prevents parallel attempt to validate
// git requirements.
initMutex = sync.Mutex{}
)
// Init validates git installation, locates the git executable
// binary in PATH and check for available shell to use.
func Init() error {
// prevent concurrent call
initMutex.Lock()
defer initMutex.Unlock()
// if validation has been done before and binary located in
// PATH, return.
if gitBinary != "" {
return nil
}
// locate git binary in path
var err error
if gitBinary, err = gos.LookPath("git"); err != nil {
return fmt.Errorf("git middleware requires git installed. Cannot find git binary in PATH")
}
// locate bash in PATH. If not found, fallback to sh.
// If neither is found, return error.
shell = "bash"
if _, err = gos.LookPath("bash"); err != nil {
shell = "sh"
if _, err = gos.LookPath("sh"); err != nil {
return fmt.Errorf("git middleware requires either bash or sh")
}
}
return nil
}
// writeScriptFile writes content to a temporary file.
// It changes the temporary file mode to executable and
// closes it to prepare it for execution.
func writeScriptFile(content []byte) (file gitos.File, err error) {
if file, err = gos.TempFile("", "caddy"); err != nil {
return nil, err
}
if _, err = file.Write(content); err != nil {
return nil, err
}
if err = file.Chmod(os.FileMode(0755)); err != nil {
return nil, err
}
return file, file.Close()
}
// gitWrapperScript forms content for git.sh script
func gitWrapperScript() []byte {
scriptTemplate := `#!/usr/bin/env {shell}
# The MIT License (MIT)
# Copyright (c) 2013 Alvin Abad
if [ $# -eq 0 ]; then
echo "Git wrapper script that can specify an ssh-key file
Usage:
git.sh -i ssh-key-file git-command
"
exit 1
fi
# remove temporary file on exit
trap 'rm -f {tmp_dir}/.git_ssh.$$' 0
if [ "$1" = "-i" ]; then
SSH_KEY=$2; shift; shift
echo -e "#!/usr/bin/env {shell}\n \
ssh -i $SSH_KEY \$@" > {tmp_dir}/.git_ssh.$$
chmod +x {tmp_dir}/.git_ssh.$$
export GIT_SSH={tmp_dir}/.git_ssh.$$
fi
# in case the git command is repeated
[ "$1" = "git" ] && shift
# Run the git command
{git_binary} "$@"
`
replacer := strings.NewReplacer(
"{shell}", shell,
"{tmp_dir}", strings.TrimSuffix(gos.TempDir(), "/"),
"{git_binary}", gitBinary,
)
return []byte(replacer.Replace(scriptTemplate))
}
// bashScript forms content of bash script to clone or update a repo using ssh
func bashScript(gitSSHPath string, repo *Repo, params []string) []byte {
scriptTemplate := `#!/usr/bin/env {shell}
mkdir -p ~/.ssh;
touch ~/.ssh/known_hosts;
ssh-keyscan -t rsa,dsa {repo_host} 2>&1 | sort -u - ~/.ssh/known_hosts > ~/.ssh/tmp_hosts;
cat ~/.ssh/tmp_hosts | while read line
do
grep -q "$line" ~/.ssh/known_hosts || echo $line >> ~/.ssh/known_hosts;
done
{git_ssh_path} -i {ssh_key_path} {ssh_params};
`
replacer := strings.NewReplacer(
"{shell}", shell,
"{repo_host}", repo.Host,
"{git_ssh_path}", gitSSHPath,
"{ssh_key_path}", repo.KeyPath,
"{ssh_params}", strings.Join(params, " "),
)
return []byte(replacer.Replace(scriptTemplate))
}