package git import ( "compress/flate" "compress/gzip" "encoding/base64" "fmt" "io" "net/http" "os" "os/exec" "path" "regexp" "strconv" "strings" ice "shylinux.com/x/icebergs" "shylinux.com/x/icebergs/base/aaa" "shylinux.com/x/icebergs/base/cli" "shylinux.com/x/icebergs/base/mdb" "shylinux.com/x/icebergs/base/nfs" "shylinux.com/x/icebergs/base/web" kit "shylinux.com/x/toolkits" ) func requestReader(req *http.Request) (io.ReadCloser, error) { switch req.Header.Get("content-encoding") { case "gzip": return gzip.NewReader(req.Body) case "deflate": return flate.NewReader(req.Body), nil } return req.Body, nil } func packetWrite(str string) []byte { s := strconv.FormatInt(int64(len(str)+4), 16) if len(s)%4 != 0 { s = strings.Repeat("0", 4-len(s)%4) + s } return []byte(s + str) } func packetFlush() []byte { return []byte("0000") } var basicAuthRegex = regexp.MustCompile("^([^:]*):(.*)$") func _server_param(m *ice.Message, arg ...string) (string, string) { repos, service := path.Join(arg...), kit.Select(arg[len(arg)-1], m.Option("service")) switch { case strings.HasSuffix(repos, "info/refs"): repos = strings.TrimSuffix(repos, "info/refs") default: repos = strings.TrimSuffix(repos, service) } return kit.Path(m.Conf(SERVER, kit.META_PATH), "repos", repos), strings.TrimPrefix(service, "git-") } func _server_login(m *ice.Message) error { parts := strings.SplitN(m.R.Header.Get("Authorization"), " ", 2) if len(parts) < 2 { return fmt.Errorf("Invalid authorization header, not enought parts") } authType, authData := parts[0], parts[1] if strings.ToLower(authType) != "basic" { return fmt.Errorf("Authentication '%s' was not of 'Basic' type", authType) } data, err := base64.StdEncoding.DecodeString(authData) if err != nil { return err } matches := basicAuthRegex.FindStringSubmatch(string(data)) if matches == nil { return fmt.Errorf("Authorization data '%s' did not match auth regexp", data) } username, password := matches[1], matches[2] if !aaa.UserLogin(m, username, password) { return fmt.Errorf("username or password error") } if aaa.UserRole(m, username) == aaa.VOID { return fmt.Errorf("userrole error") } return nil } func _server_repos(m *ice.Message, arg ...string) { m.Option(cli.CMD_DIR, path.Join(m.Conf(SERVER, kit.META_PATH), REPOS)) p := strings.TrimSuffix(path.Join(arg...), "info/refs") if _, e := os.Stat(path.Join(m.Option(cli.CMD_DIR), p)); os.IsNotExist(e) { m.Cmd(cli.SYSTEM, GIT, INIT, "--bare", p) // 创建仓库 } } func _server_cmd(m *ice.Message, arg ...string) error { repos, service := _server_param(m, arg...) if strings.HasSuffix(path.Join(arg...), "info/refs") { command := exec.Command("/usr/bin/git", service, "--stateless-rpc", "--advertise-refs", ".") command.Dir = repos out, err := command.Output() if err != nil { return err } m.W.Header().Set("Expires", "Fri, 01 Jan 1980 00:00:00 GMT") m.W.Header().Set("Pragma", "no-cache") m.W.Header().Set("Cache-Control", "no-cache, max-age=0, must-revalidate") m.W.Header().Set("Content-Type", fmt.Sprintf("application/x-git-%s-advertisement", service)) m.W.WriteHeader(http.StatusOK) m.W.Write(packetWrite("# service=git-" + service + "\n")) m.W.Write(packetFlush()) m.W.Write(out) return nil } command := exec.Command("/usr/bin/git", service, "--stateless-rpc", ".") command.Dir = repos stdin, err := command.StdinPipe() if err != nil { return err } stdout, err := command.StdoutPipe() if err != nil { return err } defer stdout.Close() if err = command.Start(); err != nil { return err } reader, err := requestReader(m.R) if err != nil { return err } defer reader.Close() io.Copy(stdin, reader) stdin.Close() m.W.Header().Set("Content-Type", fmt.Sprintf("application/x-git-%s-result", service)) io.Copy(m.W, stdout) if err = command.Wait(); err != nil { return err } return nil } const SERVER = "server" func init() { Index.Merge(&ice.Context{Configs: map[string]*ice.Config{ SERVER: {Name: SERVER, Help: "服务器", Value: kit.Data(kit.MDB_PATH, ice.USR_LOCAL)}, }, Commands: map[string]*ice.Command{ web.WEB_LOGIN: {Hand: func(m *ice.Message, c *ice.Context, cmd string, arg ...string) { m.Render(ice.RENDER_VOID) }}, "/repos/": {Name: "/repos/", Help: "代码库", Hand: func(m *ice.Message, c *ice.Context, cmd string, arg ...string) { if m.Option("go-get") == "1" { // 下载地址 p := m.Conf(web.SHARE, kit.Keym(kit.MDB_DOMAIN)) + "/x/" + path.Join(arg...) m.RenderResult(``, strings.TrimPrefix(p, "https://"), p) return } switch _, service := _server_param(m, arg...); service { case "receive-pack": // 上传代码 if err := _server_login(m); err != nil { m.W.Header().Set("WWW-Authenticate", `Basic realm="git server"`) http.Error(m.W, err.Error(), 401) return // 认证失败 } _server_repos(m, arg...) case "upload-pack": // 下载代码 } if err := _server_cmd(m, arg...); err != nil { http.Error(m.W, err.Error(), 500) return // 未知错误 } // githttp.New(kit.Path(m.Conf(SERVER, kit.META_PATH))).ServeHTTP(m.W, m.R) }}, SERVER: {Name: "server path auto create", Help: "服务器", Action: map[string]*ice.Action{ mdb.CREATE: {Name: "create name", Help: "添加", Hand: func(m *ice.Message, arg ...string) { m.Option(cli.CMD_DIR, path.Join(m.Conf(SERVER, kit.META_PATH), REPOS)) m.Cmd(cli.SYSTEM, GIT, INIT, "--bare", m.Option(kit.MDB_NAME)) }}, }, Hand: func(m *ice.Message, c *ice.Context, cmd string, arg ...string) { if m.Option(nfs.DIR_ROOT, path.Join(m.Conf(SERVER, kit.META_PATH), REPOS)); len(arg) == 0 { m.Cmdy(nfs.DIR, "./") return } m.Cmdy("_sum", path.Join(m.Option(nfs.DIR_ROOT), arg[0])) }}, }}) }