| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426 | 
							- package decoder
 
- import (
 
- 	"errors"
 
- 	"fmt"
 
- 	"path/filepath"
 
- 	"strings"
 
- 	"github.com/langgenius/dify-plugin-daemon/internal/utils/parser"
 
- 	"github.com/langgenius/dify-plugin-daemon/pkg/entities/plugin_entities"
 
- )
 
- type PluginDecoderHelper struct {
 
- 	pluginDeclaration *plugin_entities.PluginDeclaration
 
- 	checksum          string
 
- }
 
- func (p *PluginDecoderHelper) Manifest(decoder PluginDecoder) (plugin_entities.PluginDeclaration, error) {
 
- 	if p.pluginDeclaration != nil {
 
- 		return *p.pluginDeclaration, nil
 
- 	}
 
- 	// read the manifest file
 
- 	manifest, err := decoder.ReadFile("manifest.yaml")
 
- 	if err != nil {
 
- 		return plugin_entities.PluginDeclaration{}, err
 
- 	}
 
- 	dec, err := parser.UnmarshalYamlBytes[plugin_entities.PluginDeclaration](manifest)
 
- 	if err != nil {
 
- 		return plugin_entities.PluginDeclaration{}, err
 
- 	}
 
- 	// try to load plugins
 
- 	plugins := dec.Plugins
 
- 	for _, tool := range plugins.Tools {
 
- 		// read yaml
 
- 		pluginYaml, err := decoder.ReadFile(tool)
 
- 		if err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read tool file: %s", tool))
 
- 		}
 
- 		pluginDec, err := parser.UnmarshalYamlBytes[plugin_entities.ToolProviderDeclaration](pluginYaml)
 
- 		if err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal plugin file: %s", tool))
 
- 		}
 
- 		// read tools
 
- 		for _, tool_file := range pluginDec.ToolFiles {
 
- 			toolFileContent, err := decoder.ReadFile(tool_file)
 
- 			if err != nil {
 
- 				return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read tool file: %s", tool_file))
 
- 			}
 
- 			toolFileDec, err := parser.UnmarshalYamlBytes[plugin_entities.ToolDeclaration](toolFileContent)
 
- 			if err != nil {
 
- 				return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal tool file: %s", tool_file))
 
- 			}
 
- 			pluginDec.Tools = append(pluginDec.Tools, toolFileDec)
 
- 		}
 
- 		dec.Tool = &pluginDec
 
- 	}
 
- 	for _, endpoint := range plugins.Endpoints {
 
- 		// read yaml
 
- 		pluginYaml, err := decoder.ReadFile(endpoint)
 
- 		if err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read endpoint file: %s", endpoint))
 
- 		}
 
- 		pluginDec, err := parser.UnmarshalYamlBytes[plugin_entities.EndpointProviderDeclaration](pluginYaml)
 
- 		if err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal plugin file: %s", endpoint))
 
- 		}
 
- 		// read detailed endpoints
 
- 		endpointsFiles := pluginDec.EndpointFiles
 
- 		for _, endpoint_file := range endpointsFiles {
 
- 			endpointFileContent, err := decoder.ReadFile(endpoint_file)
 
- 			if err != nil {
 
- 				return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read endpoint file: %s", endpoint_file))
 
- 			}
 
- 			endpointFileDec, err := parser.UnmarshalYamlBytes[plugin_entities.EndpointDeclaration](endpointFileContent)
 
- 			if err != nil {
 
- 				return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal endpoint file: %s", endpoint_file))
 
- 			}
 
- 			pluginDec.Endpoints = append(pluginDec.Endpoints, endpointFileDec)
 
- 		}
 
- 		dec.Endpoint = &pluginDec
 
- 	}
 
- 	for _, model := range plugins.Models {
 
- 		// read yaml
 
- 		pluginYaml, err := decoder.ReadFile(model)
 
- 		if err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read model file: %s", model))
 
- 		}
 
- 		pluginDec, err := parser.UnmarshalYamlBytes[plugin_entities.ModelProviderDeclaration](pluginYaml)
 
- 		if err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal plugin file: %s", model))
 
- 		}
 
- 		// read model position file
 
- 		if pluginDec.PositionFiles != nil {
 
- 			pluginDec.Position = &plugin_entities.ModelPosition{}
 
- 			llmFileName, ok := pluginDec.PositionFiles["llm"]
 
- 			if ok {
 
- 				llmFile, err := decoder.ReadFile(llmFileName)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read llm position file: %s", llmFileName))
 
- 				}
 
- 				position, err := parser.UnmarshalYamlBytes[[]string](llmFile)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal llm position file: %s", llmFileName))
 
- 				}
 
- 				pluginDec.Position.LLM = &position
 
- 			}
 
- 			textEmbeddingFileName, ok := pluginDec.PositionFiles["text_embedding"]
 
- 			if ok {
 
- 				textEmbeddingFile, err := decoder.ReadFile(textEmbeddingFileName)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read text embedding position file: %s", textEmbeddingFileName))
 
- 				}
 
- 				position, err := parser.UnmarshalYamlBytes[[]string](textEmbeddingFile)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal text embedding position file: %s", textEmbeddingFileName))
 
- 				}
 
- 				pluginDec.Position.TextEmbedding = &position
 
- 			}
 
- 			rerankFileName, ok := pluginDec.PositionFiles["rerank"]
 
- 			if ok {
 
- 				rerankFile, err := decoder.ReadFile(rerankFileName)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read rerank position file: %s", rerankFileName))
 
- 				}
 
- 				position, err := parser.UnmarshalYamlBytes[[]string](rerankFile)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal rerank position file: %s", rerankFileName))
 
- 				}
 
- 				pluginDec.Position.Rerank = &position
 
- 			}
 
- 			ttsFileName, ok := pluginDec.PositionFiles["tts"]
 
- 			if ok {
 
- 				ttsFile, err := decoder.ReadFile(ttsFileName)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read tts position file: %s", ttsFileName))
 
- 				}
 
- 				position, err := parser.UnmarshalYamlBytes[[]string](ttsFile)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal tts position file: %s", ttsFileName))
 
- 				}
 
- 				pluginDec.Position.TTS = &position
 
- 			}
 
- 			speech2textFileName, ok := pluginDec.PositionFiles["speech2text"]
 
- 			if ok {
 
- 				speech2textFile, err := decoder.ReadFile(speech2textFileName)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read speech2text position file: %s", speech2textFileName))
 
- 				}
 
- 				position, err := parser.UnmarshalYamlBytes[[]string](speech2textFile)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal speech2text position file: %s", speech2textFileName))
 
- 				}
 
- 				pluginDec.Position.Speech2text = &position
 
- 			}
 
- 			moderationFileName, ok := pluginDec.PositionFiles["moderation"]
 
- 			if ok {
 
- 				moderationFile, err := decoder.ReadFile(moderationFileName)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read moderation position file: %s", moderationFileName))
 
- 				}
 
- 				position, err := parser.UnmarshalYamlBytes[[]string](moderationFile)
 
- 				if err != nil {
 
- 					return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal moderation position file: %s", moderationFileName))
 
- 				}
 
- 				pluginDec.Position.Moderation = &position
 
- 			}
 
- 		}
 
- 		// read models
 
- 		if err := decoder.Walk(func(filename, dir string) error {
 
- 			modelPatterns := pluginDec.ModelFiles
 
- 			// using glob to match if dir/filename is in models
 
- 			modelFileName := filepath.Join(dir, filename)
 
- 			if strings.HasSuffix(modelFileName, "_position.yaml") {
 
- 				return nil
 
- 			}
 
- 			for _, model_pattern := range modelPatterns {
 
- 				matched, err := filepath.Match(model_pattern, modelFileName)
 
- 				if err != nil {
 
- 					return err
 
- 				}
 
- 				if matched {
 
- 					// read model file
 
- 					modelFile, err := decoder.ReadFile(modelFileName)
 
- 					if err != nil {
 
- 						return err
 
- 					}
 
- 					modelDec, err := parser.UnmarshalYamlBytes[plugin_entities.ModelDeclaration](modelFile)
 
- 					if err != nil {
 
- 						return err
 
- 					}
 
- 					pluginDec.Models = append(pluginDec.Models, modelDec)
 
- 				}
 
- 			}
 
- 			return nil
 
- 		}); err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, err
 
- 		}
 
- 		dec.Model = &pluginDec
 
- 	}
 
- 	for _, agentStrategy := range plugins.AgentStrategies {
 
- 		// read yaml
 
- 		pluginYaml, err := decoder.ReadFile(agentStrategy)
 
- 		if err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read agent strategy file: %s", agentStrategy))
 
- 		}
 
- 		pluginDec, err := parser.UnmarshalYamlBytes[plugin_entities.AgentStrategyProviderDeclaration](pluginYaml)
 
- 		if err != nil {
 
- 			return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal plugin file: %s", agentStrategy))
 
- 		}
 
- 		for _, strategyFile := range pluginDec.StrategyFiles {
 
- 			strategyFileContent, err := decoder.ReadFile(strategyFile)
 
- 			if err != nil {
 
- 				return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to read agent strategy file: %s", strategyFile))
 
- 			}
 
- 			strategyDec, err := parser.UnmarshalYamlBytes[plugin_entities.AgentStrategyDeclaration](strategyFileContent)
 
- 			if err != nil {
 
- 				return plugin_entities.PluginDeclaration{}, errors.Join(err, fmt.Errorf("failed to unmarshal agent strategy file: %s", strategyFile))
 
- 			}
 
- 			pluginDec.Strategies = append(pluginDec.Strategies, strategyDec)
 
- 		}
 
- 		dec.AgentStrategy = &pluginDec
 
- 	}
 
- 	dec.FillInDefaultValues()
 
- 	// verify signature
 
- 	// for ZipPluginDecoder, use the third party signature verification if it is enabled
 
- 	if zipDecoder, ok := decoder.(*ZipPluginDecoder); ok {
 
- 		config := zipDecoder.thirdPartySignatureVerificationConfig
 
- 		if config != nil && config.Enabled && len(config.PublicKeyPaths) > 0 {
 
- 			dec.Verified = VerifyPluginWithPublicKeyPaths(decoder, config.PublicKeyPaths) == nil
 
- 		} else {
 
- 			dec.Verified = VerifyPlugin(decoder) == nil
 
- 		}
 
- 	} else {
 
- 		dec.Verified = VerifyPlugin(decoder) == nil
 
- 	}
 
- 	if err := dec.ManifestValidate(); err != nil {
 
- 		return plugin_entities.PluginDeclaration{}, err
 
- 	}
 
- 	p.pluginDeclaration = &dec
 
- 	return dec, nil
 
- }
 
- func (p *PluginDecoderHelper) Assets(decoder PluginDecoder) (map[string][]byte, error) {
 
- 	files, err := decoder.ReadDir("_assets")
 
- 	if err != nil {
 
- 		return nil, err
 
- 	}
 
- 	assets := make(map[string][]byte)
 
- 	for _, file := range files {
 
- 		content, err := decoder.ReadFile(file)
 
- 		if err != nil {
 
- 			return nil, err
 
- 		}
 
- 		// trim _assets
 
- 		file, _ = strings.CutPrefix(file, "_assets"+string(filepath.Separator))
 
- 		assets[file] = content
 
- 	}
 
- 	return assets, nil
 
- }
 
- func (p *PluginDecoderHelper) Checksum(decoder_instance PluginDecoder) (string, error) {
 
- 	if p.checksum != "" {
 
- 		return p.checksum, nil
 
- 	}
 
- 	var err error
 
- 	p.checksum, err = CalculateChecksum(decoder_instance)
 
- 	if err != nil {
 
- 		return "", err
 
- 	}
 
- 	return p.checksum, nil
 
- }
 
- func (p *PluginDecoderHelper) UniqueIdentity(decoder PluginDecoder) (plugin_entities.PluginUniqueIdentifier, error) {
 
- 	manifest, err := decoder.Manifest()
 
- 	if err != nil {
 
- 		return plugin_entities.PluginUniqueIdentifier(""), err
 
- 	}
 
- 	identity := manifest.Identity()
 
- 	checksum, err := decoder.Checksum()
 
- 	if err != nil {
 
- 		return plugin_entities.PluginUniqueIdentifier(""), err
 
- 	}
 
- 	return plugin_entities.NewPluginUniqueIdentifier(fmt.Sprintf("%s@%s", identity, checksum))
 
- }
 
- func (p *PluginDecoderHelper) CheckAssetsValid(decoder PluginDecoder) error {
 
- 	declaration, err := decoder.Manifest()
 
- 	if err != nil {
 
- 		return errors.Join(err, fmt.Errorf("failed to get manifest"))
 
- 	}
 
- 	assets, err := decoder.Assets()
 
- 	if err != nil {
 
- 		return errors.Join(err, fmt.Errorf("failed to get assets"))
 
- 	}
 
- 	if declaration.Model != nil {
 
- 		if declaration.Model.IconSmall != nil {
 
- 			if declaration.Model.IconSmall.EnUS != "" {
 
- 				if _, ok := assets[declaration.Model.IconSmall.EnUS]; !ok {
 
- 					return errors.Join(err, fmt.Errorf("model icon small en_US not found"))
 
- 				}
 
- 			}
 
- 			if declaration.Model.IconSmall.ZhHans != "" {
 
- 				if _, ok := assets[declaration.Model.IconSmall.ZhHans]; !ok {
 
- 					return errors.Join(err, fmt.Errorf("model icon small zh_Hans not found"))
 
- 				}
 
- 			}
 
- 			if declaration.Model.IconSmall.JaJp != "" {
 
- 				if _, ok := assets[declaration.Model.IconSmall.JaJp]; !ok {
 
- 					return errors.Join(err, fmt.Errorf("model icon small ja_JP not found"))
 
- 				}
 
- 			}
 
- 			if declaration.Model.IconSmall.PtBr != "" {
 
- 				if _, ok := assets[declaration.Model.IconSmall.PtBr]; !ok {
 
- 					return errors.Join(err, fmt.Errorf("model icon small pt_BR not found"))
 
- 				}
 
- 			}
 
- 		}
 
- 		if declaration.Model.IconLarge != nil {
 
- 			if declaration.Model.IconLarge.EnUS != "" {
 
- 				if _, ok := assets[declaration.Model.IconLarge.EnUS]; !ok {
 
- 					return errors.Join(err, fmt.Errorf("model icon large en_US not found"))
 
- 				}
 
- 			}
 
- 			if declaration.Model.IconLarge.ZhHans != "" {
 
- 				if _, ok := assets[declaration.Model.IconLarge.ZhHans]; !ok {
 
- 					return errors.Join(err, fmt.Errorf("model icon large zh_Hans not found"))
 
- 				}
 
- 			}
 
- 			if declaration.Model.IconLarge.JaJp != "" {
 
- 				if _, ok := assets[declaration.Model.IconLarge.JaJp]; !ok {
 
- 					return errors.Join(err, fmt.Errorf("model icon large ja_JP not found"))
 
- 				}
 
- 			}
 
- 			if declaration.Model.IconLarge.PtBr != "" {
 
- 				if _, ok := assets[declaration.Model.IconLarge.PtBr]; !ok {
 
- 					return errors.Join(err, fmt.Errorf("model icon large pt_BR not found"))
 
- 				}
 
- 			}
 
- 		}
 
- 	}
 
- 	if declaration.Tool != nil {
 
- 		if declaration.Tool.Identity.Icon != "" {
 
- 			if _, ok := assets[declaration.Tool.Identity.Icon]; !ok {
 
- 				return errors.Join(err, fmt.Errorf("tool icon not found"))
 
- 			}
 
- 		}
 
- 	}
 
- 	if declaration.Icon != "" {
 
- 		if _, ok := assets[declaration.Icon]; !ok {
 
- 			return errors.Join(err, fmt.Errorf("plugin icon not found"))
 
- 		}
 
- 	}
 
- 	return nil
 
- }
 
 
  |