package caddyfsgit import ( "io/fs" "errors" "github.com/caddyserver/caddy/v2" "github.com/caddyserver/caddy/v2/caddyconfig/caddyfile" "go.uber.org/zap" "github.com/go-git/go-git/v5" "github.com/go-git/go-git/v5/plumbing" "github.com/thediveo/gitrepofs" ) func init() { caddy.RegisterModule(FS{}) } // Interface guards var ( _ fs.FS = (*FS)(nil) _ caddyfile.Unmarshaler = (*FS)(nil) ) // FS provides a view into a specific git tree. type FS struct { fs.FS `json:"-"` repoPath string `json:"repository,omitempty"` revision string `json:"revision,omitempty"` revCommit plumbing.Hash repo *git.Repository logger *zap.Logger } // CaddyModule returns the Caddy module information. func (FS) CaddyModule() caddy.ModuleInfo { return caddy.ModuleInfo{ ID: "caddy.fs.git", New: func() caddy.Module { return new(FS) }, } } func (fs *FS) Provision(ctx caddy.Context) error { fs.logger = ctx.Logger(fs) fs.logger.Info("Parsed repository path", zap.String("repository", fs.repoPath)) fs.logger.Info("Parsed revision path", zap.String("revision", fs.repoPath)) if fs.repoPath == "" { fs.logger.Error("Repository is unset") return errors.New("repository must be set") } repo, err := git.PlainOpen(fs.repoPath) if err != nil { fs.logger.Error("failed to open git repository", zap.Error(err), zap.String("path", fs.repoPath)) return err } if fs.revision == "" { fs.revision = "HEAD" } fs.repo = repo return nil } func (fs *FS) UnmarshalCaddyfile(d *caddyfile.Dispenser) error { if !d.Next() { // skip block beginning return d.ArgErr() } for nesting := d.Nesting(); d.NextBlock(nesting); { switch d.Val() { case "repository": if !d.Args(&fs.repoPath) { return d.ArgErr() } case "revision": if !d.Args(&fs.revision) { return d.ArgErr() } default: return d.Errf("'%s' not a valid caddy.fs.git option", d.Val()) } } return nil } func (gfs *FS) RepoFS() (fs.FS, error) { hash, err := gfs.repo.ResolveRevision(plumbing.Revision(gfs.revision)) if err != nil { gfs.logger.Error("failed to resolve revision", zap.Error(err)) return nil, err } if *hash ==gfs.revCommit { return gfs.FS, nil } commit, err := gfs.repo.CommitObject(*hash) if err != nil { gfs.logger.Error("failed to get commit object", zap.Error(err)) return nil, err } tree, err := commit.Tree() if err != nil { gfs.logger.Error("failed to get tree object", zap.Error(err)) return nil, err } gfs.revCommit = *hash gfs.FS = gitrepofs.New(gfs.repo, tree, commit.Author.When) return gfs.FS, nil } func (gfs *FS) Open(name string) (fs.File, error) { repofs, err := gfs.RepoFS() if err != nil { return nil, err } return repofs.Open(name) }