manage_plugin.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464
  1. package service
  2. import (
  3. "errors"
  4. "time"
  5. "github.com/langgenius/dify-plugin-daemon/internal/db"
  6. "github.com/langgenius/dify-plugin-daemon/internal/types/exception"
  7. "github.com/langgenius/dify-plugin-daemon/internal/types/models"
  8. "github.com/langgenius/dify-plugin-daemon/internal/utils/cache/helper"
  9. "github.com/langgenius/dify-plugin-daemon/internal/utils/strings"
  10. "github.com/langgenius/dify-plugin-daemon/pkg/entities"
  11. "github.com/langgenius/dify-plugin-daemon/pkg/entities/manifest_entities"
  12. "github.com/langgenius/dify-plugin-daemon/pkg/entities/plugin_entities"
  13. )
  14. func ListPlugins(tenant_id string, page int, page_size int) *entities.Response {
  15. type installation struct {
  16. ID string `json:"id"`
  17. Name string `json:"name"`
  18. PluginID string `json:"plugin_id"`
  19. TenantID string `json:"tenant_id"`
  20. PluginUniqueIdentifier string `json:"plugin_unique_identifier"`
  21. EndpointsActive int `json:"endpoints_active"`
  22. EndpointsSetups int `json:"endpoints_setups"`
  23. InstallationID string `json:"installation_id"`
  24. Declaration *plugin_entities.PluginDeclaration `json:"declaration"`
  25. RuntimeType plugin_entities.PluginRuntimeType `json:"runtime_type"`
  26. Version manifest_entities.Version `json:"version"`
  27. CreatedAt time.Time `json:"created_at"`
  28. UpdatedAt time.Time `json:"updated_at"`
  29. Source string `json:"source"`
  30. Checksum string `json:"checksum"`
  31. Meta map[string]any `json:"meta"`
  32. }
  33. pluginInstallations, err := db.GetAll[models.PluginInstallation](
  34. db.Equal("tenant_id", tenant_id),
  35. db.Page(page, page_size),
  36. )
  37. if err != nil {
  38. return exception.InternalServerError(err).ToResponse()
  39. }
  40. data := make([]installation, 0, len(pluginInstallations))
  41. for _, plugin_installation := range pluginInstallations {
  42. pluginUniqueIdentifier, err := plugin_entities.NewPluginUniqueIdentifier(
  43. plugin_installation.PluginUniqueIdentifier,
  44. )
  45. if err != nil {
  46. return exception.UniqueIdentifierError(err).ToResponse()
  47. }
  48. pluginDeclaration, err := helper.CombinedGetPluginDeclaration(
  49. pluginUniqueIdentifier,
  50. plugin_entities.PluginRuntimeType(plugin_installation.RuntimeType),
  51. )
  52. if err != nil {
  53. return exception.InternalServerError(err).ToResponse()
  54. }
  55. data = append(data, installation{
  56. ID: plugin_installation.ID,
  57. Name: pluginDeclaration.Name,
  58. TenantID: plugin_installation.TenantID,
  59. PluginID: pluginUniqueIdentifier.PluginID(),
  60. PluginUniqueIdentifier: pluginUniqueIdentifier.String(),
  61. InstallationID: plugin_installation.ID,
  62. Declaration: pluginDeclaration,
  63. EndpointsSetups: plugin_installation.EndpointsSetups,
  64. EndpointsActive: plugin_installation.EndpointsActive,
  65. RuntimeType: plugin_entities.PluginRuntimeType(plugin_installation.RuntimeType),
  66. Version: pluginDeclaration.Version,
  67. CreatedAt: plugin_installation.CreatedAt,
  68. UpdatedAt: plugin_installation.UpdatedAt,
  69. Source: plugin_installation.Source,
  70. Meta: plugin_installation.Meta,
  71. Checksum: pluginUniqueIdentifier.Checksum(),
  72. })
  73. }
  74. return entities.NewSuccessResponse(data)
  75. }
  76. // Using plugin_ids to fetch plugin installations
  77. func BatchFetchPluginInstallationByIDs(tenant_id string, plugin_ids []string) *entities.Response {
  78. type installation struct {
  79. models.PluginInstallation
  80. Version manifest_entities.Version `json:"version"`
  81. Checksum string `json:"checksum"`
  82. Declaration *plugin_entities.PluginDeclaration `json:"declaration"`
  83. }
  84. if len(plugin_ids) == 0 {
  85. return entities.NewSuccessResponse([]installation{})
  86. }
  87. pluginInstallations, err := db.GetAll[models.PluginInstallation](
  88. db.Equal("tenant_id", tenant_id),
  89. db.InArray("plugin_id", strings.Map(plugin_ids, func(id string) any { return id })),
  90. db.Page(1, 256), // TODO: pagination
  91. )
  92. if err != nil {
  93. return exception.InternalServerError(err).ToResponse()
  94. }
  95. data := make([]installation, 0, len(pluginInstallations))
  96. for _, plugin_installation := range pluginInstallations {
  97. pluginUniqueIdentifier, err := plugin_entities.NewPluginUniqueIdentifier(
  98. plugin_installation.PluginUniqueIdentifier,
  99. )
  100. if err != nil {
  101. return exception.InternalServerError(errors.Join(errors.New("invalid plugin unique identifier found"), err)).ToResponse()
  102. }
  103. pluginDeclaration, err := helper.CombinedGetPluginDeclaration(
  104. pluginUniqueIdentifier,
  105. plugin_entities.PluginRuntimeType(plugin_installation.RuntimeType),
  106. )
  107. if err != nil {
  108. return exception.InternalServerError(errors.Join(errors.New("failed to get plugin declaration"), err)).ToResponse()
  109. }
  110. data = append(data, installation{
  111. PluginInstallation: plugin_installation,
  112. Version: pluginUniqueIdentifier.Version(),
  113. Checksum: pluginUniqueIdentifier.Checksum(),
  114. Declaration: pluginDeclaration,
  115. })
  116. }
  117. return entities.NewSuccessResponse(data)
  118. }
  119. // check which plugin is missing
  120. func FetchMissingPluginInstallations(tenant_id string, plugin_unique_identifiers []plugin_entities.PluginUniqueIdentifier) *entities.Response {
  121. type MissingPluginDependency struct {
  122. PluginUniqueIdentifier string `json:"plugin_unique_identifier"`
  123. CurrentIdentifier string `json:"current_identifier"`
  124. }
  125. result := make([]MissingPluginDependency, 0, len(plugin_unique_identifiers))
  126. if len(plugin_unique_identifiers) == 0 {
  127. return entities.NewSuccessResponse(result)
  128. }
  129. installed, err := db.GetAll[models.PluginInstallation](
  130. db.Equal("tenant_id", tenant_id),
  131. db.InArray(
  132. "plugin_id",
  133. strings.Map(
  134. plugin_unique_identifiers,
  135. func(id plugin_entities.PluginUniqueIdentifier) any {
  136. return id.PluginID()
  137. },
  138. ),
  139. ),
  140. db.Page(1, 256), // TODO: pagination
  141. )
  142. if err != nil {
  143. return exception.InternalServerError(err).ToResponse()
  144. }
  145. // check which plugin is missing
  146. for _, pluginUniqueIdentifier := range plugin_unique_identifiers {
  147. found := false
  148. for _, installedPlugin := range installed {
  149. if installedPlugin.PluginID == pluginUniqueIdentifier.PluginID() {
  150. found = true
  151. if installedPlugin.PluginUniqueIdentifier != pluginUniqueIdentifier.String() {
  152. // version mismatched
  153. result = append(result, MissingPluginDependency{
  154. PluginUniqueIdentifier: pluginUniqueIdentifier.String(),
  155. CurrentIdentifier: installedPlugin.PluginUniqueIdentifier,
  156. })
  157. }
  158. break
  159. }
  160. }
  161. if !found {
  162. result = append(result, MissingPluginDependency{
  163. PluginUniqueIdentifier: pluginUniqueIdentifier.String(),
  164. })
  165. }
  166. }
  167. return entities.NewSuccessResponse(result)
  168. }
  169. func ListTools(tenant_id string, page int, page_size int) *entities.Response {
  170. type Tool struct {
  171. models.ToolInstallation // pointer to avoid deep copy
  172. Declaration *plugin_entities.ToolProviderDeclaration `json:"declaration"`
  173. }
  174. providers, err := db.GetAll[models.ToolInstallation](
  175. db.Equal("tenant_id", tenant_id),
  176. db.Page(page, page_size),
  177. )
  178. if err != nil {
  179. return exception.InternalServerError(err).ToResponse()
  180. }
  181. data := make([]Tool, 0, len(providers))
  182. for _, provider := range providers {
  183. // check if plugin id starts with uuid
  184. // split by uuid length
  185. uniqueIdentifier := plugin_entities.PluginUniqueIdentifier(provider.PluginUniqueIdentifier)
  186. var runtimeType plugin_entities.PluginRuntimeType
  187. if uniqueIdentifier.RemoteLike() {
  188. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_REMOTE
  189. } else {
  190. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_LOCAL
  191. }
  192. declaration, err := helper.CombinedGetPluginDeclaration(
  193. uniqueIdentifier,
  194. runtimeType,
  195. )
  196. if err != nil {
  197. return exception.InternalServerError(err).ToResponse()
  198. }
  199. data = append(data, Tool{
  200. ToolInstallation: provider,
  201. Declaration: declaration.Tool,
  202. })
  203. }
  204. return entities.NewSuccessResponse(data)
  205. }
  206. func ListModels(tenant_id string, page int, page_size int) *entities.Response {
  207. type AIModel struct {
  208. models.AIModelInstallation // pointer to avoid deep copy
  209. Declaration *plugin_entities.ModelProviderDeclaration `json:"declaration"`
  210. }
  211. providers, err := db.GetAll[models.AIModelInstallation](
  212. db.Equal("tenant_id", tenant_id),
  213. db.Page(page, page_size),
  214. )
  215. if err != nil {
  216. return exception.InternalServerError(err).ToResponse()
  217. }
  218. data := make([]AIModel, 0, len(providers))
  219. for _, provider := range providers {
  220. uniqueIdentifier := plugin_entities.PluginUniqueIdentifier(provider.PluginUniqueIdentifier)
  221. var runtimeType plugin_entities.PluginRuntimeType
  222. if uniqueIdentifier.RemoteLike() {
  223. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_REMOTE
  224. } else {
  225. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_LOCAL
  226. }
  227. declaration, err := helper.CombinedGetPluginDeclaration(
  228. uniqueIdentifier,
  229. runtimeType,
  230. )
  231. if err != nil {
  232. return exception.InternalServerError(err).ToResponse()
  233. }
  234. data = append(data, AIModel{
  235. AIModelInstallation: provider,
  236. Declaration: declaration.Model,
  237. })
  238. }
  239. return entities.NewSuccessResponse(data)
  240. }
  241. func GetTool(tenant_id string, plugin_id string, provider string) *entities.Response {
  242. type Tool struct {
  243. models.ToolInstallation // pointer to avoid deep copy
  244. Declaration *plugin_entities.ToolProviderDeclaration `json:"declaration"`
  245. }
  246. // try get tool
  247. tool, err := db.GetOne[models.ToolInstallation](
  248. db.Equal("tenant_id", tenant_id),
  249. db.Equal("plugin_id", plugin_id),
  250. )
  251. if err != nil {
  252. if err == db.ErrDatabaseNotFound {
  253. return exception.ErrPluginNotFound().ToResponse()
  254. }
  255. return exception.InternalServerError(err).ToResponse()
  256. }
  257. if tool.Provider != provider {
  258. return exception.ErrPluginNotFound().ToResponse()
  259. }
  260. uniqueIdentifier := plugin_entities.PluginUniqueIdentifier(tool.PluginUniqueIdentifier)
  261. var runtimeType plugin_entities.PluginRuntimeType
  262. if uniqueIdentifier.RemoteLike() {
  263. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_REMOTE
  264. } else {
  265. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_LOCAL
  266. }
  267. declaration, err := helper.CombinedGetPluginDeclaration(
  268. uniqueIdentifier,
  269. runtimeType,
  270. )
  271. if err != nil {
  272. return exception.InternalServerError(err).ToResponse()
  273. }
  274. return entities.NewSuccessResponse(Tool{
  275. ToolInstallation: tool,
  276. Declaration: declaration.Tool,
  277. })
  278. }
  279. type RequestCheckToolExistence struct {
  280. PluginID string `json:"plugin_id" validate:"required"`
  281. ProviderName string `json:"provider_name" validate:"required"`
  282. }
  283. func CheckToolExistence(tenantId string, providerIds []RequestCheckToolExistence) *entities.Response {
  284. existence := make([]bool, 0, len(providerIds))
  285. // get all providers
  286. providers, err := db.GetAll[models.ToolInstallation](
  287. db.Equal("tenant_id", tenantId),
  288. db.InArray("plugin_id", strings.Map(providerIds, func(id RequestCheckToolExistence) any { return id.PluginID })),
  289. db.Page(1, 256), // TODO: pagination
  290. )
  291. if err != nil {
  292. return exception.InternalServerError(err).ToResponse()
  293. }
  294. // check provider id
  295. for _, providerId := range providerIds {
  296. found := false
  297. for _, provider := range providers {
  298. if provider.PluginID == providerId.PluginID && provider.Provider == providerId.ProviderName {
  299. found = true
  300. break
  301. }
  302. }
  303. existence = append(existence, found)
  304. }
  305. return entities.NewSuccessResponse(existence)
  306. }
  307. func ListAgentStrategies(tenant_id string, page int, page_size int) *entities.Response {
  308. type AgentStrategy struct {
  309. models.AgentStrategyInstallation // pointer to avoid deep copy
  310. Declaration *plugin_entities.AgentStrategyProviderDeclaration `json:"declaration"`
  311. }
  312. providers, err := db.GetAll[models.AgentStrategyInstallation](
  313. db.Equal("tenant_id", tenant_id),
  314. db.Page(page, page_size),
  315. )
  316. if err != nil {
  317. return exception.InternalServerError(err).ToResponse()
  318. }
  319. data := make([]AgentStrategy, 0, len(providers))
  320. for _, provider := range providers {
  321. uniqueIdentifier := plugin_entities.PluginUniqueIdentifier(provider.PluginUniqueIdentifier)
  322. var runtimeType plugin_entities.PluginRuntimeType
  323. if uniqueIdentifier.RemoteLike() {
  324. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_REMOTE
  325. } else {
  326. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_LOCAL
  327. }
  328. declaration, err := helper.CombinedGetPluginDeclaration(
  329. uniqueIdentifier,
  330. runtimeType,
  331. )
  332. if err != nil {
  333. return exception.InternalServerError(err).ToResponse()
  334. }
  335. data = append(data, AgentStrategy{
  336. AgentStrategyInstallation: provider,
  337. Declaration: declaration.AgentStrategy,
  338. })
  339. }
  340. return entities.NewSuccessResponse(data)
  341. }
  342. func GetAgentStrategy(tenant_id string, plugin_id string, provider string) *entities.Response {
  343. type AgentStrategy struct {
  344. models.AgentStrategyInstallation // pointer to avoid deep copy
  345. Declaration *plugin_entities.AgentStrategyProviderDeclaration `json:"declaration"`
  346. }
  347. agent_strategy, err := db.GetOne[models.AgentStrategyInstallation](
  348. db.Equal("tenant_id", tenant_id),
  349. db.Equal("plugin_id", plugin_id),
  350. )
  351. if err != nil {
  352. if err == db.ErrDatabaseNotFound {
  353. return exception.ErrPluginNotFound().ToResponse()
  354. }
  355. return exception.InternalServerError(err).ToResponse()
  356. }
  357. if agent_strategy.Provider != provider {
  358. return exception.ErrPluginNotFound().ToResponse()
  359. }
  360. uniqueIdentifier := plugin_entities.PluginUniqueIdentifier(agent_strategy.PluginUniqueIdentifier)
  361. var runtimeType plugin_entities.PluginRuntimeType
  362. if uniqueIdentifier.RemoteLike() {
  363. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_REMOTE
  364. } else {
  365. runtimeType = plugin_entities.PLUGIN_RUNTIME_TYPE_LOCAL
  366. }
  367. declaration, err := helper.CombinedGetPluginDeclaration(
  368. uniqueIdentifier,
  369. runtimeType,
  370. )
  371. if err != nil {
  372. return exception.InternalServerError(err).ToResponse()
  373. }
  374. return entities.NewSuccessResponse(AgentStrategy{
  375. AgentStrategyInstallation: agent_strategy,
  376. Declaration: declaration.AgentStrategy,
  377. })
  378. }