diff options
author | Runxi Yu <me@runxiyu.org> | 2025-04-05 17:21:14 +0800 |
---|---|---|
committer | Runxi Yu <me@runxiyu.org> | 2025-04-05 17:21:14 +0800 |
commit | 9b17278aece47aca17d32a37f67b7078708e78be (patch) | |
tree | 8badae3ed3e9594fae3f9de6e23468165a69e5d4 /git2c/cmd2.go | |
parent | Remove the extra .gitignore in man/ (diff) | |
download | forge-9b17278aece47aca17d32a37f67b7078708e78be.tar.gz forge-9b17278aece47aca17d32a37f67b7078708e78be.tar.zst forge-9b17278aece47aca17d32a37f67b7078708e78be.zip |
Refactor git2d comms to ./git2c
Diffstat (limited to 'git2c/cmd2.go')
-rw-r--r-- | git2c/cmd2.go | 89 |
1 files changed, 89 insertions, 0 deletions
diff --git a/git2c/cmd2.go b/git2c/cmd2.go new file mode 100644 index 0000000..0671372 --- /dev/null +++ b/git2c/cmd2.go @@ -0,0 +1,89 @@ +package git2c + +import ( + "errors" + "fmt" + "io" +) + +func (c *Client) Cmd2(repoPath, pathSpec string) ([]TreeEntry, string, error) { + if err := c.writer.WriteData([]byte(repoPath)); err != nil { + return nil, "", fmt.Errorf("sending repo path failed: %w", err) + } + if err := c.writer.WriteUint(2); err != nil { + return nil, "", fmt.Errorf("sending command failed: %w", err) + } + if err := c.writer.WriteData([]byte(pathSpec)); err != nil { + return nil, "", fmt.Errorf("sending path failed: %w", err) + } + + status, err := c.reader.ReadUint() + if err != nil { + return nil, "", fmt.Errorf("reading status failed: %w", err) + } + + switch status { + case 0: + kind, err := c.reader.ReadUint() + if err != nil { + return nil, "", fmt.Errorf("reading object kind failed: %w", err) + } + + switch kind { + case 1: + // Tree + count, err := c.reader.ReadUint() + if err != nil { + return nil, "", fmt.Errorf("reading entry count failed: %w", err) + } + + var files []TreeEntry + for i := 0; i < int(count); i++ { + typeCode, err := c.reader.ReadUint() + if err != nil { + return nil, "", fmt.Errorf("error reading entry type: %w", err) + } + mode, err := c.reader.ReadUint() + if err != nil { + return nil, "", fmt.Errorf("error reading entry mode: %w", err) + } + size, err := c.reader.ReadUint() + if err != nil { + return nil, "", fmt.Errorf("error reading entry size: %w", err) + } + name, err := c.reader.ReadData() + if err != nil { + return nil, "", fmt.Errorf("error reading entry name: %w", err) + } + + files = append(files, TreeEntry{ + Name: string(name), + Mode: fmt.Sprintf("%06o", mode), + Size: size, + IsFile: typeCode == 2, + IsSubtree: typeCode == 1, + }) + } + + return files, "", nil + + case 2: + // Blob + content, err := c.reader.ReadData() + if err != nil && !errors.Is(err, io.EOF) { + return nil, "", fmt.Errorf("error reading file content: %w", err) + } + + return nil, string(content), nil + + default: + return nil, "", fmt.Errorf("unknown kind: %d", kind) + } + + case 3: + return nil, "", fmt.Errorf("path not found: %s", pathSpec) + + default: + return nil, "", fmt.Errorf("unknown status code: %d", status) + } +} |