| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191 | package decoderimport (	"errors"	"io"	"io/fs"	"os"	"path/filepath"	"strings"	"github.com/go-git/go-git/plumbing/format/gitignore"	"github.com/langgenius/dify-plugin-daemon/pkg/entities/plugin_entities")var (	ErrNotDir = errors.New("not a directory"))type FSPluginDecoder struct {	PluginDecoder	PluginDecoderHelper	// root directory of the plugin	root string	fs fs.FS}func NewFSPluginDecoder(root string) (*FSPluginDecoder, error) {	decoder := &FSPluginDecoder{		root: root,	}	err := decoder.Open()	if err != nil {		return nil, err	}	// read the manifest file	if _, err := decoder.Manifest(); err != nil {		return nil, err	}	return decoder, nil}func (d *FSPluginDecoder) Open() error {	d.fs = os.DirFS(d.root)	// try to stat the root directory	s, err := os.Stat(d.root)	if err != nil {		return err	}	if !s.IsDir() {		return ErrNotDir	}	return nil}func (d *FSPluginDecoder) Walk(fn func(filename string, dir string) error) error {	// read .difyignore file	ignorePatterns := []gitignore.Pattern{}	// Try .difyignore first, fallback to .gitignore if not found	ignoreBytes, err := d.ReadFile(".difyignore")	if err != nil {		ignoreBytes, err = d.ReadFile(".gitignore")	}	if err == nil {		ignoreLines := strings.Split(string(ignoreBytes), "\n")		for _, line := range ignoreLines {			line = strings.TrimSpace(line)			if line == "" || strings.HasPrefix(line, "#") {				continue			}			ignorePatterns = append(ignorePatterns, gitignore.ParsePattern(line, nil))		}	}	return filepath.WalkDir(d.root, func(path string, info fs.DirEntry, err error) error {		if err != nil {			return err		}		// get relative path from root		relPath, err := filepath.Rel(d.root, path)		if err != nil {			return err		}		// skip root directory		if relPath == "." {			return nil		}		// check if path matches any ignore pattern		pathParts := strings.Split(relPath, string(filepath.Separator))		for _, pattern := range ignorePatterns {			if result := pattern.Match(pathParts, info.IsDir()); result == gitignore.Exclude {				if info.IsDir() {					return filepath.SkipDir				}				return nil			}		}		// get directory path relative to root		dir := filepath.Dir(relPath)		if dir == "." {			dir = ""		}		// skip if the path is a directory		if info.IsDir() {			return nil		}		return fn(info.Name(), dir)	})}func (d *FSPluginDecoder) Close() error {	return nil}func (d *FSPluginDecoder) Stat(filename string) (fs.FileInfo, error) {	return os.Stat(filepath.Join(d.root, filename))}func (d *FSPluginDecoder) ReadFile(filename string) ([]byte, error) {	return os.ReadFile(filepath.Join(d.root, filename))}func (d *FSPluginDecoder) ReadDir(dirname string) ([]string, error) {	var files []string	err := filepath.WalkDir(		filepath.Join(d.root, dirname),		func(path string, info fs.DirEntry, err error) error {			if err != nil {				return err			}			if !info.IsDir() {				relPath, err := filepath.Rel(d.root, path)				if err != nil {					return err				}				files = append(files, relPath)			}			return nil		},	)	if err != nil {		return nil, err	}	return files, nil}func (d *FSPluginDecoder) FileReader(filename string) (io.ReadCloser, error) {	return os.Open(filepath.Join(d.root, filename))}func (d *FSPluginDecoder) Signature() (string, error) {	return "", nil}func (d *FSPluginDecoder) CreateTime() (int64, error) {	return 0, nil}func (d *FSPluginDecoder) Manifest() (plugin_entities.PluginDeclaration, error) {	return d.PluginDecoderHelper.Manifest(d)}func (d *FSPluginDecoder) Assets() (map[string][]byte, error) {	return d.PluginDecoderHelper.Assets(d)}func (d *FSPluginDecoder) Checksum() (string, error) {	return d.PluginDecoderHelper.Checksum(d)}func (d *FSPluginDecoder) UniqueIdentity() (plugin_entities.PluginUniqueIdentifier, error) {	return d.PluginDecoderHelper.UniqueIdentity(d)}func (d *FSPluginDecoder) CheckAssetsValid() error {	return d.PluginDecoderHelper.CheckAssetsValid(d)}
 |