@yoryoboy/bi-mcp 1.11.0 → 1.13.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (237) hide show
  1. package/README.md +4 -5
  2. package/dist/.tsbuildinfo +1 -1
  3. package/dist/index.js +398 -8
  4. package/dist/index.js.map +2 -2
  5. package/dist/mcp-use.json +2 -2
  6. package/dist/src/config/__tests__/meta.test.js +35 -0
  7. package/dist/src/config/__tests__/meta.test.js.map +7 -0
  8. package/dist/src/config/meta-store.js +109 -0
  9. package/dist/src/config/meta-store.js.map +7 -0
  10. package/dist/src/config/meta.js +0 -2
  11. package/dist/src/config/meta.js.map +2 -2
  12. package/dist/src/meta/__tests__/meta-utils.test.js +155 -0
  13. package/dist/src/meta/__tests__/meta-utils.test.js.map +7 -0
  14. package/dist/src/services/mercadolibre/__tests__/mercadolibre-api.test.js +265 -0
  15. package/dist/src/services/mercadolibre/__tests__/mercadolibre-api.test.js.map +7 -0
  16. package/dist/src/services/mercadolibre/__tests__/mercadolibre-items.test.js +311 -0
  17. package/dist/src/services/mercadolibre/__tests__/mercadolibre-items.test.js.map +7 -0
  18. package/dist/src/services/mercadolibre/__tests__/mercadolibre-orders.test.js +220 -0
  19. package/dist/src/services/mercadolibre/__tests__/mercadolibre-orders.test.js.map +7 -0
  20. package/dist/src/services/meta/__tests__/meta-ads.test.js +126 -0
  21. package/dist/src/services/meta/__tests__/meta-ads.test.js.map +7 -0
  22. package/dist/src/services/meta/__tests__/meta-api.test.js +70 -0
  23. package/dist/src/services/meta/__tests__/meta-api.test.js.map +7 -0
  24. package/dist/src/services/meta/meta-ads.js +94 -40
  25. package/dist/src/services/meta/meta-ads.js.map +2 -2
  26. package/dist/src/services/meta/meta-api.js +8 -6
  27. package/dist/src/services/meta/meta-api.js.map +2 -2
  28. package/dist/src/services/vtex/__tests__/vtex-catalog-write-batch.test.js +165 -0
  29. package/dist/src/services/vtex/__tests__/vtex-catalog-write-batch.test.js.map +7 -0
  30. package/dist/src/services/vtex/__tests__/vtex-catalog-write-eans.test.js +92 -0
  31. package/dist/src/services/vtex/__tests__/vtex-catalog-write-eans.test.js.map +7 -0
  32. package/dist/src/services/vtex/__tests__/vtex-catalog-write-products.test.js +41 -0
  33. package/dist/src/services/vtex/__tests__/vtex-catalog-write-products.test.js.map +7 -0
  34. package/dist/src/services/vtex/__tests__/vtex-catalog.test.js +80 -0
  35. package/dist/src/services/vtex/__tests__/vtex-catalog.test.js.map +7 -0
  36. package/dist/src/services/vtex/__tests__/vtex-orders-write-http.test.js +85 -0
  37. package/dist/src/services/vtex/__tests__/vtex-orders-write-http.test.js.map +7 -0
  38. package/dist/src/services/vtex/__tests__/vtex-orders-write-state-validation.test.js +251 -0
  39. package/dist/src/services/vtex/__tests__/vtex-orders-write-state-validation.test.js.map +7 -0
  40. package/dist/src/services/vtex/__tests__/vtex-write.test.js +111 -0
  41. package/dist/src/services/vtex/__tests__/vtex-write.test.js.map +7 -0
  42. package/dist/src/services/vtex/vtex-catalog-write.js +194 -1
  43. package/dist/src/services/vtex/vtex-catalog-write.js.map +2 -2
  44. package/dist/src/services/vtex/vtex-catalog.js +118 -1
  45. package/dist/src/services/vtex/vtex-catalog.js.map +2 -2
  46. package/dist/src/services/vtex/vtex-pricing-write.js +5 -0
  47. package/dist/src/services/vtex/vtex-pricing-write.js.map +2 -2
  48. package/dist/src/services/vtex/vtex-write.js +67 -1
  49. package/dist/src/services/vtex/vtex-write.js.map +2 -2
  50. package/dist/src/tools/config/list-profiles.js +37 -6
  51. package/dist/src/tools/config/list-profiles.js.map +2 -2
  52. package/dist/src/tools/index.js.map +2 -2
  53. package/dist/src/tools/mercadolibre/__tests__/helpers.test.js +37 -0
  54. package/dist/src/tools/mercadolibre/__tests__/helpers.test.js.map +7 -0
  55. package/dist/src/tools/mercadolibre/__tests__/profile-resolution.test.js +30 -0
  56. package/dist/src/tools/mercadolibre/__tests__/profile-resolution.test.js.map +7 -0
  57. package/dist/src/tools/mercadolibre/profile-resolution.js +4 -50
  58. package/dist/src/tools/mercadolibre/profile-resolution.js.map +2 -2
  59. package/dist/src/tools/meta/__tests__/pagination.test.js +133 -0
  60. package/dist/src/tools/meta/__tests__/pagination.test.js.map +7 -0
  61. package/dist/src/tools/meta/__tests__/profile-access.test.js +262 -0
  62. package/dist/src/tools/meta/__tests__/profile-access.test.js.map +7 -0
  63. package/dist/src/tools/meta/__tests__/read-tools.test.js +722 -0
  64. package/dist/src/tools/meta/__tests__/read-tools.test.js.map +7 -0
  65. package/dist/src/tools/meta/__tests__/schemas.test.js +103 -0
  66. package/dist/src/tools/meta/__tests__/schemas.test.js.map +7 -0
  67. package/dist/src/tools/meta/account-overview.js +37 -19
  68. package/dist/src/tools/meta/account-overview.js.map +2 -2
  69. package/dist/src/tools/meta/ad-account-info.js +31 -6
  70. package/dist/src/tools/meta/ad-account-info.js.map +2 -2
  71. package/dist/src/tools/meta/ads-performance.js +35 -21
  72. package/dist/src/tools/meta/ads-performance.js.map +2 -2
  73. package/dist/src/tools/meta/campaign-performance.js +35 -18
  74. package/dist/src/tools/meta/campaign-performance.js.map +2 -2
  75. package/dist/src/tools/meta/list-accessible-ad-accounts.js +39 -10
  76. package/dist/src/tools/meta/list-accessible-ad-accounts.js.map +2 -2
  77. package/dist/src/tools/meta/list-accessible-businesses.js +37 -13
  78. package/dist/src/tools/meta/list-accessible-businesses.js.map +2 -2
  79. package/dist/src/tools/meta/placement-mix.js +37 -22
  80. package/dist/src/tools/meta/placement-mix.js.map +2 -2
  81. package/dist/src/tools/meta/profile-access.js +215 -0
  82. package/dist/src/tools/meta/profile-access.js.map +7 -0
  83. package/dist/src/tools/meta/schema-helpers.js +19 -0
  84. package/dist/src/tools/meta/schema-helpers.js.map +7 -0
  85. package/dist/src/tools/meta/time-series.js +40 -23
  86. package/dist/src/tools/meta/time-series.js.map +2 -2
  87. package/dist/src/tools/vtex/__tests__/catalog-admin-batch.test.js +233 -0
  88. package/dist/src/tools/vtex/__tests__/catalog-admin-batch.test.js.map +7 -0
  89. package/dist/src/tools/vtex/__tests__/catalog-admin-categories.test.js +649 -0
  90. package/dist/src/tools/vtex/__tests__/catalog-admin-categories.test.js.map +7 -0
  91. package/dist/src/tools/vtex/__tests__/catalog-admin-product-specifications.test.js +339 -0
  92. package/dist/src/tools/vtex/__tests__/catalog-admin-product-specifications.test.js.map +7 -0
  93. package/dist/src/tools/vtex/__tests__/catalog-admin-products-skus.test.js +235 -0
  94. package/dist/src/tools/vtex/__tests__/catalog-admin-products-skus.test.js.map +7 -0
  95. package/dist/src/tools/vtex/__tests__/catalog-navigation-reads.test.js +372 -0
  96. package/dist/src/tools/vtex/__tests__/catalog-navigation-reads.test.js.map +7 -0
  97. package/dist/src/tools/vtex/__tests__/catalog-navigation-surface.test.js +57 -0
  98. package/dist/src/tools/vtex/__tests__/catalog-navigation-surface.test.js.map +7 -0
  99. package/dist/src/tools/vtex/__tests__/write-helpers.test.js +97 -0
  100. package/dist/src/tools/vtex/__tests__/write-helpers.test.js.map +7 -0
  101. package/dist/src/tools/vtex/activate-sku.js +1 -48
  102. package/dist/src/tools/vtex/activate-sku.js.map +2 -2
  103. package/dist/src/tools/vtex/associate-specification.js +3 -54
  104. package/dist/src/tools/vtex/associate-specification.js.map +2 -2
  105. package/dist/src/tools/vtex/attach-catalog-image.js +3 -57
  106. package/dist/src/tools/vtex/attach-catalog-image.js.map +2 -2
  107. package/dist/src/tools/vtex/catalog-admin-batch.js +298 -0
  108. package/dist/src/tools/vtex/catalog-admin-batch.js.map +7 -0
  109. package/dist/src/tools/vtex/catalog-admin-categories.js +542 -0
  110. package/dist/src/tools/vtex/catalog-admin-categories.js.map +7 -0
  111. package/dist/src/tools/vtex/catalog-admin-product-specifications.js +275 -0
  112. package/dist/src/tools/vtex/catalog-admin-product-specifications.js.map +7 -0
  113. package/dist/src/tools/vtex/catalog-admin-products-skus.js +475 -0
  114. package/dist/src/tools/vtex/catalog-admin-products-skus.js.map +7 -0
  115. package/dist/src/tools/vtex/catalog-navigation-reads.js +430 -0
  116. package/dist/src/tools/vtex/catalog-navigation-reads.js.map +7 -0
  117. package/dist/src/tools/vtex/create-brand.js +1 -64
  118. package/dist/src/tools/vtex/create-brand.js.map +2 -2
  119. package/dist/src/tools/vtex/create-category.js +1 -76
  120. package/dist/src/tools/vtex/create-category.js.map +2 -2
  121. package/dist/src/tools/vtex/create-product-with-sku.js +3 -114
  122. package/dist/src/tools/vtex/create-product-with-sku.js.map +2 -2
  123. package/dist/src/tools/vtex/create-specification-value.js +3 -47
  124. package/dist/src/tools/vtex/create-specification-value.js.map +2 -2
  125. package/dist/src/tools/vtex/create-specification.js +1 -80
  126. package/dist/src/tools/vtex/create-specification.js.map +2 -2
  127. package/dist/src/tools/vtex/deactivate-sku.js +1 -48
  128. package/dist/src/tools/vtex/deactivate-sku.js.map +2 -2
  129. package/dist/src/tools/vtex/delete-all-product-specifications.js +9 -0
  130. package/dist/src/tools/vtex/delete-all-product-specifications.js.map +7 -0
  131. package/dist/src/tools/vtex/delete-all-sku-specifications.js +9 -0
  132. package/dist/src/tools/vtex/delete-all-sku-specifications.js.map +7 -0
  133. package/dist/src/tools/vtex/delete-brand.js +9 -0
  134. package/dist/src/tools/vtex/delete-brand.js.map +7 -0
  135. package/dist/src/tools/vtex/delete-catalog-image.js +9 -0
  136. package/dist/src/tools/vtex/delete-catalog-image.js.map +7 -0
  137. package/dist/src/tools/vtex/delete-product-specification.js +9 -0
  138. package/dist/src/tools/vtex/delete-product-specification.js.map +7 -0
  139. package/dist/src/tools/vtex/delete-sku-price.js +55 -0
  140. package/dist/src/tools/vtex/delete-sku-price.js.map +7 -0
  141. package/dist/src/tools/vtex/delete-sku-specification.js +9 -0
  142. package/dist/src/tools/vtex/delete-sku-specification.js.map +7 -0
  143. package/dist/src/tools/vtex/get-brand.js +6 -0
  144. package/dist/src/tools/vtex/get-brand.js.map +7 -0
  145. package/dist/src/tools/vtex/get-category-tree.js +6 -0
  146. package/dist/src/tools/vtex/get-category-tree.js.map +7 -0
  147. package/dist/src/tools/vtex/get-category.js +6 -0
  148. package/dist/src/tools/vtex/get-category.js.map +7 -0
  149. package/dist/src/tools/vtex/get-product-specifications.js +9 -0
  150. package/dist/src/tools/vtex/get-product-specifications.js.map +7 -0
  151. package/dist/src/tools/vtex/get-product.js +6 -0
  152. package/dist/src/tools/vtex/get-product.js.map +7 -0
  153. package/dist/src/tools/vtex/get-sku.js +6 -0
  154. package/dist/src/tools/vtex/get-sku.js.map +7 -0
  155. package/dist/src/tools/vtex/index.js +23 -1
  156. package/dist/src/tools/vtex/index.js.map +2 -2
  157. package/dist/src/tools/vtex/list-brands.js +6 -0
  158. package/dist/src/tools/vtex/list-brands.js.map +7 -0
  159. package/dist/src/tools/vtex/list-categories.js +6 -0
  160. package/dist/src/tools/vtex/list-categories.js.map +7 -0
  161. package/dist/src/tools/vtex/list-products-by-category.js +6 -0
  162. package/dist/src/tools/vtex/list-products-by-category.js.map +7 -0
  163. package/dist/src/tools/vtex/list-products.js +6 -0
  164. package/dist/src/tools/vtex/list-products.js.map +7 -0
  165. package/dist/src/tools/vtex/list-skus-by-product.js +6 -0
  166. package/dist/src/tools/vtex/list-skus-by-product.js.map +7 -0
  167. package/dist/src/tools/vtex/list-specification-groups.js +9 -0
  168. package/dist/src/tools/vtex/list-specification-groups.js.map +7 -0
  169. package/dist/src/tools/vtex/move-category.js +6 -0
  170. package/dist/src/tools/vtex/move-category.js.map +7 -0
  171. package/dist/src/tools/vtex/profile-resolution.js +4 -51
  172. package/dist/src/tools/vtex/profile-resolution.js.map +2 -2
  173. package/dist/src/tools/vtex/update-brand.js +6 -0
  174. package/dist/src/tools/vtex/update-brand.js.map +7 -0
  175. package/dist/src/tools/vtex/update-category.js +6 -0
  176. package/dist/src/tools/vtex/update-category.js.map +7 -0
  177. package/dist/src/tools/vtex/update-product-basic-fields.js +3 -65
  178. package/dist/src/tools/vtex/update-product-basic-fields.js.map +2 -2
  179. package/dist/src/tools/vtex/update-sku-basic-fields.js +1 -87
  180. package/dist/src/tools/vtex/update-sku-basic-fields.js.map +2 -2
  181. package/dist/src/tools/vtex/update-sku-price.js +21 -1
  182. package/dist/src/tools/vtex/update-sku-price.js.map +2 -2
  183. package/dist/src/tools/vtex/write-helpers.js +104 -14
  184. package/dist/src/tools/vtex/write-helpers.js.map +2 -2
  185. package/dist/src/utils/provider-profile-selection.js +117 -0
  186. package/dist/src/utils/provider-profile-selection.js.map +7 -0
  187. package/dist/tests/meli/mercadolibre-tool-handlers-medium-batch4.test.js +678 -0
  188. package/dist/tests/meli/mercadolibre-tool-handlers-medium-batch4.test.js.map +7 -0
  189. package/dist/tests/meli/mercadolibre-tool-handlers-monster-batch5.test.js +564 -0
  190. package/dist/tests/meli/mercadolibre-tool-handlers-monster-batch5.test.js.map +7 -0
  191. package/dist/tests/meli/mercadolibre-tool-handlers-monster-batch6.test.js +387 -0
  192. package/dist/tests/meli/mercadolibre-tool-handlers-monster-batch6.test.js.map +7 -0
  193. package/dist/tests/meli/mercadolibre-tool-handlers-monster-batch7.test.js +368 -0
  194. package/dist/tests/meli/mercadolibre-tool-handlers-monster-batch7.test.js.map +7 -0
  195. package/dist/tests/meli/mercadolibre-tool-handlers-small.test.js +626 -0
  196. package/dist/tests/meli/mercadolibre-tool-handlers-small.test.js.map +7 -0
  197. package/dist/tests/meli/mercadolibre-tool-handlers-write-batch8.test.js +480 -0
  198. package/dist/tests/meli/mercadolibre-tool-handlers-write-batch8.test.js.map +7 -0
  199. package/dist/tests/setup.js +2 -0
  200. package/dist/tests/setup.js.map +7 -0
  201. package/dist/tests/smoke/test-harness.test.js +7 -0
  202. package/dist/tests/smoke/test-harness.test.js.map +7 -0
  203. package/dist/tests/vtex/read-only-utils.test.js +161 -0
  204. package/dist/tests/vtex/read-only-utils.test.js.map +7 -0
  205. package/dist/tests/vtex/vtex-catalog-admin-docs.test.js +24 -0
  206. package/dist/tests/vtex/vtex-catalog-admin-docs.test.js.map +7 -0
  207. package/dist/tests/vtex/vtex-catalog-admin-surface.test.js +30 -0
  208. package/dist/tests/vtex/vtex-catalog-admin-surface.test.js.map +7 -0
  209. package/dist/tests/vtex/vtex-catalog-write-tools.test.js +712 -0
  210. package/dist/tests/vtex/vtex-catalog-write-tools.test.js.map +7 -0
  211. package/dist/tests/vtex/vtex-catalog.service.test.js +51 -0
  212. package/dist/tests/vtex/vtex-catalog.service.test.js.map +7 -0
  213. package/dist/tests/vtex/vtex-inventory-tools.test.js +201 -0
  214. package/dist/tests/vtex/vtex-inventory-tools.test.js.map +7 -0
  215. package/dist/tests/vtex/vtex-logistics.service.test.js +134 -0
  216. package/dist/tests/vtex/vtex-logistics.service.test.js.map +7 -0
  217. package/dist/tests/vtex/vtex-order-details.tool.test.js +141 -0
  218. package/dist/tests/vtex/vtex-order-details.tool.test.js.map +7 -0
  219. package/dist/tests/vtex/vtex-order-write-tools.test.js +483 -0
  220. package/dist/tests/vtex/vtex-order-write-tools.test.js.map +7 -0
  221. package/dist/tests/vtex/vtex-orders-summary.tool.test.js +185 -0
  222. package/dist/tests/vtex/vtex-orders-summary.tool.test.js.map +7 -0
  223. package/dist/tests/vtex/vtex-orders.service.test.js +120 -0
  224. package/dist/tests/vtex/vtex-orders.service.test.js.map +7 -0
  225. package/dist/tests/vtex/vtex-pricing-write-tools.test.js +202 -0
  226. package/dist/tests/vtex/vtex-pricing-write-tools.test.js.map +7 -0
  227. package/dist/tests/vtex/vtex-pricing-write.service.test.js +106 -0
  228. package/dist/tests/vtex/vtex-pricing-write.service.test.js.map +7 -0
  229. package/dist/tests/vtex/vtex-pricing.service.test.js +88 -0
  230. package/dist/tests/vtex/vtex-pricing.service.test.js.map +7 -0
  231. package/dist/tests/vtex/vtex-small-tools.test.js +190 -0
  232. package/dist/tests/vtex/vtex-small-tools.test.js.map +7 -0
  233. package/dist/tests/vtex/vtex-write-simple-tools.test.js +647 -0
  234. package/dist/tests/vtex/vtex-write-simple-tools.test.js.map +7 -0
  235. package/dist/vitest.config.js +15 -0
  236. package/dist/vitest.config.js.map +7 -0
  237. package/package.json +6 -2
@@ -0,0 +1,626 @@
1
+ import { beforeEach, describe, expect, it, vi } from "vitest";
2
+ import { meliGetAccountContextHandler } from "../../src/tools/mercadolibre/get-account-context.js";
3
+ import { meliValidateConnectionHandler } from "../../src/tools/mercadolibre/validate-connection.js";
4
+ import { meliPredictCategoryHandler } from "../../src/tools/mercadolibre/predict-category.js";
5
+ import { meliGetCategoryRequirementsHandler } from "../../src/tools/mercadolibre/get-category-requirements.js";
6
+ import { meliEstimateListingFeeHandler } from "../../src/tools/mercadolibre/estimate-listing-fee.js";
7
+ import { meliGetSiteCategoriesAndListingTypesHandler } from "../../src/tools/mercadolibre/get-site-categories-and-listing-types.js";
8
+ import { meliGetItemDetailsHandler } from "../../src/tools/mercadolibre/get-item-details.js";
9
+ import { meliSearchQuestionsHandler } from "../../src/tools/mercadolibre/search-questions.js";
10
+ import { meliAnswerQuestionHandler } from "../../src/tools/mercadolibre/answer-question.js";
11
+ import { meliGetListingQualityHandler } from "../../src/tools/mercadolibre/get-listing-quality.js";
12
+ import { meliGetItemVisitsHandler } from "../../src/tools/mercadolibre/get-item-visits.js";
13
+ import { resolveMercadoLibreProfileOrSelection } from "../../src/tools/mercadolibre/profile-resolution.js";
14
+ import {
15
+ buildMercadoLibreWriteSuccessResponse,
16
+ handleMercadoLibreWriteError,
17
+ resolveMercadoLibreWriteProfile
18
+ } from "../../src/tools/mercadolibre/write-helpers.js";
19
+ import { getMercadoLibreAccessForProfile } from "../../src/config/mercadolibre.js";
20
+ import { getMercadoLibreConnectionRow } from "../../src/config/mercadolibre-profile-store.js";
21
+ import {
22
+ buildMercadoLibrePaginationMetadata,
23
+ formatMercadoLibreError,
24
+ getMercadoLibreUsersMe,
25
+ normalizeMercadoLibrePaging
26
+ } from "../../src/services/mercadolibre/mercadolibre-api.js";
27
+ import {
28
+ getMercadoLibreCategoryAttributes,
29
+ getMercadoLibreCategoryShippingPreferences,
30
+ getMercadoLibreItemDetailsBatch,
31
+ getMercadoLibreItemDetailsWithFallbackBatch,
32
+ getMercadoLibreItemPerformance,
33
+ getMercadoLibreItemVisitsWindow,
34
+ getMercadoLibreSiteCategoriesAndListingTypes,
35
+ getMercadoLibreUserItemsVisits,
36
+ searchMercadoLibreDomainDiscovery
37
+ } from "../../src/services/mercadolibre/mercadolibre-items.js";
38
+ import {
39
+ answerMercadoLibreQuestion,
40
+ searchMercadoLibreQuestions
41
+ } from "../../src/services/mercadolibre/mercadolibre-questions.js";
42
+ import { buildMercadoLibreListingFeeEstimate } from "../../src/tools/mercadolibre/listing-fee-helpers.js";
43
+ vi.mock("mcp-use/server", () => ({
44
+ object: vi.fn((payload) => ({ kind: "object", payload })),
45
+ error: vi.fn((message) => ({ kind: "error", message }))
46
+ }));
47
+ vi.mock("../../src/tools/mercadolibre/profile-resolution.js", () => ({
48
+ resolveMercadoLibreProfileOrSelection: vi.fn()
49
+ }));
50
+ vi.mock("../../src/tools/mercadolibre/write-helpers.js", async () => {
51
+ const { z } = await import("zod");
52
+ return {
53
+ mercadolibreProfileIdSchemaField: z.string().optional(),
54
+ resolveMercadoLibreWriteProfile: vi.fn(),
55
+ buildMercadoLibreWriteSuccessResponse: vi.fn((payload) => ({ kind: "success", payload })),
56
+ handleMercadoLibreWriteError: vi.fn(
57
+ (err, fallback) => ({
58
+ kind: "error",
59
+ message: `write:${fallback}:${err instanceof Error ? err.message : String(err)}`
60
+ })
61
+ )
62
+ };
63
+ });
64
+ vi.mock("../../src/config/mercadolibre.js", () => ({
65
+ getMercadoLibreAccessForProfile: vi.fn()
66
+ }));
67
+ vi.mock("../../src/config/mercadolibre-profile-store.js", () => ({
68
+ getMercadoLibreConnectionRow: vi.fn()
69
+ }));
70
+ vi.mock("../../src/services/mercadolibre/mercadolibre-api.js", () => ({
71
+ getMercadoLibreUsersMe: vi.fn(),
72
+ formatMercadoLibreError: vi.fn(
73
+ (err, fallback) => `formatted:${fallback}:${err instanceof Error ? err.message : String(err)}`
74
+ ),
75
+ normalizeMercadoLibrePaging: vi.fn((paging) => paging),
76
+ buildMercadoLibrePaginationMetadata: vi.fn((metadata) => metadata)
77
+ }));
78
+ vi.mock("../../src/services/mercadolibre/mercadolibre-items.js", () => ({
79
+ searchMercadoLibreDomainDiscovery: vi.fn(),
80
+ getMercadoLibreCategoryAttributes: vi.fn(),
81
+ getMercadoLibreCategoryShippingPreferences: vi.fn(),
82
+ getMercadoLibreSiteCategoriesAndListingTypes: vi.fn(),
83
+ getMercadoLibreItemDetailsBatch: vi.fn(),
84
+ getMercadoLibreItemDetailsWithFallbackBatch: vi.fn(),
85
+ getMercadoLibreItemPerformance: vi.fn(),
86
+ getMercadoLibreItemVisitsWindow: vi.fn(),
87
+ getMercadoLibreUserItemsVisits: vi.fn()
88
+ }));
89
+ vi.mock("../../src/services/mercadolibre/mercadolibre-questions.js", () => ({
90
+ searchMercadoLibreQuestions: vi.fn(),
91
+ answerMercadoLibreQuestion: vi.fn()
92
+ }));
93
+ vi.mock("../../src/tools/mercadolibre/listing-fee-helpers.js", () => ({
94
+ buildMercadoLibreListingFeeEstimate: vi.fn()
95
+ }));
96
+ describe("MercadoLibre small and medium tool handlers", () => {
97
+ beforeEach(() => {
98
+ vi.clearAllMocks();
99
+ vi.mocked(resolveMercadoLibreProfileOrSelection).mockResolvedValue({
100
+ ok: true,
101
+ value: { profileId: "meli-profile-1" }
102
+ });
103
+ vi.mocked(resolveMercadoLibreWriteProfile).mockResolvedValue({
104
+ ok: true,
105
+ value: { profileId: "meli-profile-1" }
106
+ });
107
+ vi.mocked(getMercadoLibreAccessForProfile).mockResolvedValue({
108
+ sellerId: "seller-99",
109
+ scopes: ["read", "write"],
110
+ expiresAt: "2026-07-01T00:00:00.000Z"
111
+ });
112
+ vi.mocked(getMercadoLibreUsersMe).mockResolvedValue({
113
+ id: 12345,
114
+ nickname: "meli-store"
115
+ });
116
+ vi.mocked(buildMercadoLibrePaginationMetadata).mockImplementation((metadata) => metadata);
117
+ vi.mocked(normalizeMercadoLibrePaging).mockImplementation((paging) => paging);
118
+ });
119
+ it("returns MercadoLibre account context with resolved profile access", async () => {
120
+ const result = await meliGetAccountContextHandler({ profileId: "custom-profile" });
121
+ expect(resolveMercadoLibreProfileOrSelection).toHaveBeenCalledWith("custom-profile");
122
+ expect(getMercadoLibreAccessForProfile).toHaveBeenCalledWith("meli-profile-1");
123
+ expect(getMercadoLibreUsersMe).toHaveBeenCalledWith("meli-profile-1");
124
+ expect(result).toEqual({
125
+ kind: "object",
126
+ payload: {
127
+ profile_id: "meli-profile-1",
128
+ seller_id: "seller-99",
129
+ nickname: "meli-store",
130
+ account_id: 12345,
131
+ scopes: ["read", "write"],
132
+ expires_at: "2026-07-01T00:00:00.000Z"
133
+ }
134
+ });
135
+ });
136
+ it("returns profile selection payload unchanged for read handlers", async () => {
137
+ vi.mocked(resolveMercadoLibreProfileOrSelection).mockResolvedValue({
138
+ ok: false,
139
+ response: { selection_required: true }
140
+ });
141
+ const result = await meliGetAccountContextHandler({ profileId: void 0 });
142
+ expect(result).toEqual({ selection_required: true });
143
+ expect(getMercadoLibreAccessForProfile).not.toHaveBeenCalled();
144
+ });
145
+ it("validates connection and reports refresh detection", async () => {
146
+ vi.mocked(getMercadoLibreConnectionRow).mockResolvedValueOnce({
147
+ expiresAt: /* @__PURE__ */ new Date("2026-06-01T00:00:00.000Z"),
148
+ encryptedAccessToken: "old-token"
149
+ }).mockResolvedValueOnce({
150
+ status: "active",
151
+ expiresAt: /* @__PURE__ */ new Date("2026-07-01T00:00:00.000Z"),
152
+ encryptedAccessToken: "new-token"
153
+ });
154
+ const result = await meliValidateConnectionHandler({ profileId: "meli-profile-1" });
155
+ expect(result).toEqual({
156
+ kind: "object",
157
+ payload: {
158
+ profile_id: "meli-profile-1",
159
+ validation_status: "active",
160
+ refreshed: true,
161
+ seller_id: "seller-99",
162
+ account_id: 12345,
163
+ nickname: "meli-store",
164
+ expires_at: "2026-07-01T00:00:00.000Z",
165
+ scopes: ["read", "write"]
166
+ }
167
+ });
168
+ });
169
+ it("predicts categories with normalized candidate payloads and defaults", async () => {
170
+ vi.mocked(searchMercadoLibreDomainDiscovery).mockResolvedValue([
171
+ {
172
+ category_id: "MLA123",
173
+ category_name: "Sneakers",
174
+ domain_id: "MLA-SNEAKERS",
175
+ domain_name: "Sneakers",
176
+ attributes_matched: "4"
177
+ }
178
+ ]);
179
+ const result = await meliPredictCategoryHandler({
180
+ profileId: "meli-profile-1",
181
+ query: "running shoes"
182
+ });
183
+ expect(searchMercadoLibreDomainDiscovery).toHaveBeenCalledWith(
184
+ "meli-profile-1",
185
+ "running shoes",
186
+ 5,
187
+ "MLA"
188
+ );
189
+ expect(result).toEqual({
190
+ kind: "object",
191
+ payload: {
192
+ profile_id: "meli-profile-1",
193
+ query: "running shoes",
194
+ site_id: "MLA",
195
+ candidates: [
196
+ {
197
+ category_id: "MLA123",
198
+ category_name: "Sneakers",
199
+ domain_id: "MLA-SNEAKERS",
200
+ domain_name: "Sneakers",
201
+ attributes_matched: 4
202
+ }
203
+ ]
204
+ }
205
+ });
206
+ });
207
+ it("splits required and recommended category requirements", async () => {
208
+ vi.mocked(getMercadoLibreCategoryAttributes).mockResolvedValue([
209
+ {
210
+ id: "BRAND",
211
+ name: "Brand",
212
+ value_type: "string",
213
+ value_max_length: 255,
214
+ tags: { required: true }
215
+ },
216
+ {
217
+ id: "MODEL",
218
+ name: "Model",
219
+ value_type: "string",
220
+ tags: {}
221
+ }
222
+ ]);
223
+ vi.mocked(getMercadoLibreCategoryShippingPreferences).mockResolvedValue({
224
+ mode: "me2",
225
+ logistic_types: ["drop_off"]
226
+ });
227
+ const result = await meliGetCategoryRequirementsHandler({
228
+ profileId: "meli-profile-1",
229
+ categoryId: "MLA999"
230
+ });
231
+ expect(result).toEqual({
232
+ kind: "object",
233
+ payload: {
234
+ profile_id: "meli-profile-1",
235
+ category_id: "MLA999",
236
+ required_attributes: [
237
+ {
238
+ id: "BRAND",
239
+ name: "Brand",
240
+ value_type: "string",
241
+ value_max_length: 255
242
+ }
243
+ ],
244
+ recommended_attributes: [
245
+ {
246
+ id: "MODEL",
247
+ name: "Model",
248
+ value_type: "string"
249
+ }
250
+ ],
251
+ shipping_preferences: {
252
+ mode: "me2",
253
+ logistics: ["drop_off"]
254
+ }
255
+ }
256
+ });
257
+ });
258
+ it("delegates listing fee estimation with the resolved profile", async () => {
259
+ vi.mocked(buildMercadoLibreListingFeeEstimate).mockResolvedValue({
260
+ profile_id: "meli-profile-1",
261
+ fee_amount: 1234,
262
+ site_id: null
263
+ });
264
+ const result = await meliEstimateListingFeeHandler({
265
+ profileId: "meli-profile-1",
266
+ itemId: "MLA1",
267
+ query: "running shoes",
268
+ price: 1e4,
269
+ categoryId: "MLA123",
270
+ listingTypeId: "gold_special",
271
+ shippingMode: "me2",
272
+ logisticType: "drop_off",
273
+ billableWeight: 1200,
274
+ currencyId: "ARS",
275
+ tags: ["test-tag"]
276
+ });
277
+ expect(buildMercadoLibreListingFeeEstimate).toHaveBeenCalledWith({
278
+ profileId: "meli-profile-1",
279
+ itemId: "MLA1",
280
+ query: "running shoes",
281
+ price: 1e4,
282
+ categoryId: "MLA123",
283
+ listingTypeId: "gold_special",
284
+ shippingMode: "me2",
285
+ logisticType: "drop_off",
286
+ billableWeight: 1200,
287
+ currencyId: "ARS",
288
+ tags: ["test-tag"]
289
+ });
290
+ expect(result).toEqual({
291
+ kind: "object",
292
+ payload: {
293
+ profile_id: "meli-profile-1",
294
+ fee_amount: 1234
295
+ }
296
+ });
297
+ });
298
+ it("combines site categories and listing types metadata", async () => {
299
+ vi.mocked(getMercadoLibreSiteCategoriesAndListingTypes).mockResolvedValue({
300
+ siteId: "MLA",
301
+ categories: [{ id: "MLA1", name: "Shoes" }],
302
+ listingTypes: [{ id: "gold_special", name: "Cl\xE1sica" }]
303
+ });
304
+ const result = await meliGetSiteCategoriesAndListingTypesHandler({ profileId: "meli-profile-1" });
305
+ expect(result).toEqual({
306
+ kind: "object",
307
+ payload: {
308
+ profile_id: "meli-profile-1",
309
+ site_id: "MLA",
310
+ categories: [{ id: "MLA1", name: "Shoes" }],
311
+ listing_types: [{ id: "gold_special", name: "Cl\xE1sica" }],
312
+ metadata: {
313
+ site_id: "MLA",
314
+ categories_count: 1,
315
+ listing_types_count: 1,
316
+ pagination: "No aplica. La tool consolida dos endpoints de cat\xE1logo del sitio MLA en una sola respuesta."
317
+ }
318
+ }
319
+ });
320
+ });
321
+ it("returns item details directly from multiget when all requested ids are found", async () => {
322
+ vi.mocked(getMercadoLibreItemDetailsBatch).mockResolvedValue([
323
+ { id: "MLA1", title: "Boots", pictures: [{ id: "1" }], status: "active" },
324
+ { id: "MLA2", title: "Sneakers", pictures: [{ id: "2" }], status: "paused" }
325
+ ]);
326
+ const result = await meliGetItemDetailsHandler({
327
+ profileId: "meli-profile-1",
328
+ itemIds: ["MLA1", "MLA1", "MLA2"]
329
+ });
330
+ expect(getMercadoLibreItemDetailsBatch).toHaveBeenCalledWith("meli-profile-1", ["MLA1", "MLA2"]);
331
+ expect(result).toEqual({
332
+ kind: "object",
333
+ payload: {
334
+ metadata: {
335
+ profile_id: "meli-profile-1",
336
+ requested: 2,
337
+ successful: 2,
338
+ failed: 0,
339
+ source: "items_multiget"
340
+ },
341
+ items: [
342
+ { id: "MLA1", title: "Boots", status: "active" },
343
+ { id: "MLA2", title: "Sneakers", status: "paused" }
344
+ ]
345
+ }
346
+ });
347
+ });
348
+ it("falls back to item-by-item details when multiget coverage is partial", async () => {
349
+ vi.mocked(getMercadoLibreItemDetailsBatch).mockResolvedValue([{ id: "MLA1", title: "Boots" }]);
350
+ vi.mocked(getMercadoLibreItemDetailsWithFallbackBatch).mockResolvedValue({
351
+ successful: [
352
+ {
353
+ id: "MLA1",
354
+ document: { id: "MLA1", title: "Boots" }
355
+ }
356
+ ],
357
+ failed: [
358
+ {
359
+ id: "MLA2",
360
+ message: "not found",
361
+ statusCode: 404,
362
+ attempts: 2,
363
+ retryable: false
364
+ }
365
+ ]
366
+ });
367
+ const result = await meliGetItemDetailsHandler({
368
+ profileId: "meli-profile-1",
369
+ itemIds: ["MLA1", "MLA2"]
370
+ });
371
+ expect(result).toEqual({
372
+ kind: "object",
373
+ payload: {
374
+ metadata: {
375
+ profile_id: "meli-profile-1",
376
+ requested: 2,
377
+ successful: 1,
378
+ failed: 1,
379
+ has_failures: true,
380
+ source: "item_by_item_fallback"
381
+ },
382
+ items: [{ id: "MLA1", title: "Boots" }],
383
+ failures: [
384
+ {
385
+ item_id: "MLA2",
386
+ message: "not found",
387
+ status_code: 404,
388
+ attempts: 2,
389
+ retryable: false
390
+ }
391
+ ]
392
+ }
393
+ });
394
+ });
395
+ it("searches questions with seller context and normalized pagination", async () => {
396
+ vi.mocked(searchMercadoLibreQuestions).mockResolvedValue({
397
+ questions: [
398
+ {
399
+ id: 777,
400
+ item_id: "MLA1",
401
+ item: { title: "Boots" },
402
+ status: "unanswered",
403
+ text: "Do you ship today?",
404
+ date_created: "2026-03-10T14:45:55.000Z",
405
+ from: { id: 555, nickname: "buyer" },
406
+ answer: { status: "not_answered" }
407
+ }
408
+ ],
409
+ paging: { total: 1, limit: 20, offset: 0 }
410
+ });
411
+ const result = await meliSearchQuestionsHandler({ profileId: "meli-profile-1" });
412
+ expect(searchMercadoLibreQuestions).toHaveBeenCalledWith("meli-profile-1", {
413
+ sellerId: "seller-99",
414
+ itemId: void 0,
415
+ status: void 0,
416
+ limit: 20,
417
+ offset: 0
418
+ });
419
+ expect(result).toEqual({
420
+ kind: "object",
421
+ payload: {
422
+ profile_id: "meli-profile-1",
423
+ metadata: {
424
+ total: 1,
425
+ limit: 20,
426
+ offset: 0,
427
+ returned: 1,
428
+ nextField: "offset"
429
+ },
430
+ questions: [
431
+ {
432
+ question_id: 777,
433
+ item_id: "MLA1",
434
+ item_title: "Boots",
435
+ status: "unanswered",
436
+ text: "Do you ship today?",
437
+ date_created: "2026-03-10 14:45",
438
+ from: { id: "", nickname: "buyer" },
439
+ answer_status: "not_answered"
440
+ }
441
+ ]
442
+ }
443
+ });
444
+ });
445
+ it("routes question answers through write helpers", async () => {
446
+ vi.mocked(answerMercadoLibreQuestion).mockResolvedValue({ id: 777, status: "answered" });
447
+ const result = await meliAnswerQuestionHandler({
448
+ profileId: "meli-profile-1",
449
+ questionId: 777,
450
+ text: "Yes, we can ship today."
451
+ });
452
+ expect(resolveMercadoLibreWriteProfile).toHaveBeenCalledWith("meli-profile-1");
453
+ expect(answerMercadoLibreQuestion).toHaveBeenCalledWith(
454
+ "meli-profile-1",
455
+ 777,
456
+ "Yes, we can ship today."
457
+ );
458
+ expect(buildMercadoLibreWriteSuccessResponse).toHaveBeenCalledWith({
459
+ profileId: "meli-profile-1",
460
+ operation: "answer_question",
461
+ resourceId: "777",
462
+ message: "MercadoLibre question answered successfully.",
463
+ after: { id: 777, status: "answered" }
464
+ });
465
+ expect(result).toEqual({
466
+ kind: "success",
467
+ payload: {
468
+ profileId: "meli-profile-1",
469
+ operation: "answer_question",
470
+ resourceId: "777",
471
+ message: "MercadoLibre question answered successfully.",
472
+ after: { id: 777, status: "answered" }
473
+ }
474
+ });
475
+ });
476
+ it("formats listing quality summaries with warning normalization", async () => {
477
+ vi.mocked(getMercadoLibreItemPerformance).mockResolvedValue({
478
+ score: "85",
479
+ status: "healthy",
480
+ health: "good",
481
+ sales: 20,
482
+ visits: 300,
483
+ warnings: [{ id: "photo", text: "Add more photos", severity: "medium" }],
484
+ item: { title: "Boots", status: "active" }
485
+ });
486
+ const result = await meliGetListingQualityHandler({
487
+ profileId: "meli-profile-1",
488
+ itemId: "MLA1"
489
+ });
490
+ expect(result).toEqual({
491
+ kind: "object",
492
+ payload: {
493
+ profile_id: "meli-profile-1",
494
+ item_id: "MLA1",
495
+ quality_score: 85,
496
+ level: "healthy",
497
+ health: "good",
498
+ sales: 20,
499
+ visits: 300,
500
+ warnings: [
501
+ {
502
+ type: "photo",
503
+ message: "Add more photos",
504
+ severity: "medium"
505
+ }
506
+ ],
507
+ raw_summary: {
508
+ title: "Boots",
509
+ status: "active"
510
+ }
511
+ }
512
+ });
513
+ });
514
+ it("returns item-level visits windows when item ids are provided", async () => {
515
+ vi.mocked(getMercadoLibreItemVisitsWindow).mockResolvedValue({
516
+ total_visits: 14,
517
+ results: [{ date: "2026-03-10", visits: 14 }]
518
+ });
519
+ const result = await meliGetItemVisitsHandler({
520
+ profileId: "meli-profile-1",
521
+ itemIds: ["MLA1", "MLA1"],
522
+ last: 7,
523
+ ending: "2026-03-10"
524
+ });
525
+ expect(getMercadoLibreItemVisitsWindow).toHaveBeenCalledTimes(1);
526
+ expect(result).toEqual({
527
+ kind: "object",
528
+ payload: {
529
+ profile_id: "meli-profile-1",
530
+ seller_id: "seller-99",
531
+ total_visits: 14,
532
+ items: [
533
+ {
534
+ item_id: "MLA1",
535
+ total_visits: 14,
536
+ last: 7,
537
+ unit: "day",
538
+ ending: "2026-03-10",
539
+ time_window: [{ date: "2026-03-10", visits: 14 }]
540
+ }
541
+ ]
542
+ }
543
+ });
544
+ });
545
+ it("returns a validation error when visits are requested without item ids or date range", async () => {
546
+ const result = await meliGetItemVisitsHandler({ profileId: "meli-profile-1" });
547
+ expect(result).toEqual({
548
+ kind: "error",
549
+ message: "Provide itemIds, or provide both startDate and endDate for seller-level visits."
550
+ });
551
+ expect(getMercadoLibreUserItemsVisits).not.toHaveBeenCalled();
552
+ });
553
+ it("returns seller-level visits when a date range is provided", async () => {
554
+ vi.mocked(getMercadoLibreUserItemsVisits).mockResolvedValue({
555
+ results: [
556
+ { item_id: "MLA1", total_visits: 10 },
557
+ { id: "MLA2", visits: 4 }
558
+ ]
559
+ });
560
+ const result = await meliGetItemVisitsHandler({
561
+ profileId: "meli-profile-1",
562
+ startDate: "2026-03-01",
563
+ endDate: "2026-03-10"
564
+ });
565
+ expect(getMercadoLibreUserItemsVisits).toHaveBeenCalledWith(
566
+ "meli-profile-1",
567
+ "seller-99",
568
+ "2026-03-01",
569
+ "2026-03-10"
570
+ );
571
+ expect(result).toEqual({
572
+ kind: "object",
573
+ payload: {
574
+ profile_id: "meli-profile-1",
575
+ seller_id: "seller-99",
576
+ total_visits: 14,
577
+ item_visit_rows: [
578
+ { item_id: "MLA1", total_visits: 10 },
579
+ { item_id: "MLA2", total_visits: 4 }
580
+ ]
581
+ }
582
+ });
583
+ });
584
+ it("formats service failures through MercadoLibre error helpers", async () => {
585
+ vi.mocked(getMercadoLibreUsersMe).mockRejectedValue(new Error("boom"));
586
+ const result = await meliValidateConnectionHandler({ profileId: "meli-profile-1" });
587
+ expect(formatMercadoLibreError).toHaveBeenCalledWith(
588
+ expect.any(Error),
589
+ "Failed to validate MercadoLibre connection"
590
+ );
591
+ expect(result).toEqual({
592
+ kind: "error",
593
+ message: "formatted:Failed to validate MercadoLibre connection:boom"
594
+ });
595
+ });
596
+ it("returns write-helper selection payloads unchanged and skips answer calls", async () => {
597
+ vi.mocked(resolveMercadoLibreWriteProfile).mockResolvedValue({
598
+ ok: false,
599
+ response: { selection_required: true, provider: "mercadolibre" }
600
+ });
601
+ const result = await meliAnswerQuestionHandler({
602
+ profileId: void 0,
603
+ questionId: 777,
604
+ text: "Answer"
605
+ });
606
+ expect(result).toEqual({ selection_required: true, provider: "mercadolibre" });
607
+ expect(answerMercadoLibreQuestion).not.toHaveBeenCalled();
608
+ });
609
+ it("routes write failures through the MercadoLibre write error helper", async () => {
610
+ vi.mocked(answerMercadoLibreQuestion).mockRejectedValue(new Error("cannot-answer"));
611
+ const result = await meliAnswerQuestionHandler({
612
+ profileId: "meli-profile-1",
613
+ questionId: 888,
614
+ text: "Answer"
615
+ });
616
+ expect(handleMercadoLibreWriteError).toHaveBeenCalledWith(
617
+ expect.any(Error),
618
+ "Failed to answer MercadoLibre question 888"
619
+ );
620
+ expect(result).toEqual({
621
+ kind: "error",
622
+ message: "write:Failed to answer MercadoLibre question 888:cannot-answer"
623
+ });
624
+ });
625
+ });
626
+ //# sourceMappingURL=mercadolibre-tool-handlers-small.test.js.map