@yoryoboy/bi-mcp 1.6.0 → 1.7.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.
- package/README.md +87 -87
- package/bin/bi-mcp.js +9 -9
- package/dist/.tsbuildinfo +1 -1
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/mcp-use.json +2 -2
- package/dist/public/icon.svg +6 -6
- package/dist/scripts/_helpers.js.map +1 -1
- package/dist/scripts/admin-profile-delete.js.map +1 -1
- package/dist/scripts/admin-profile-list.js.map +1 -1
- package/dist/scripts/admin-profile-upsert.js.map +1 -1
- package/dist/scripts/admin-vtex-list.js.map +1 -1
- package/dist/scripts/admin-vtex-upsert.js.map +1 -1
- package/dist/scripts/admin-vtex-validate.js.map +1 -1
- package/dist/scripts/run-migrations.js.map +1 -1
- package/dist/scripts/test-db-connection.js.map +1 -1
- package/dist/src/analytics/ga4-channel-groups.js.map +1 -1
- package/dist/src/analytics/ga4-report-utils.js.map +1 -1
- package/dist/src/config/benchmarks.js.map +1 -1
- package/dist/src/config/google-store.js.map +1 -1
- package/dist/src/config/google.js.map +1 -1
- package/dist/src/config/mercadolibre-profile-store.js.map +1 -1
- package/dist/src/config/mercadolibre.js.map +1 -1
- package/dist/src/config/meta.js.map +1 -1
- package/dist/src/config/profile-store.js.map +1 -1
- package/dist/src/config/vtex-crypto.js.map +1 -1
- package/dist/src/config/vtex-profile-store.js.map +1 -1
- package/dist/src/config/vtex.js.map +1 -1
- package/dist/src/db/client.js.map +1 -1
- package/dist/src/google-ads/report-utils.js +78 -0
- package/dist/src/google-ads/report-utils.js.map +7 -0
- package/dist/src/meta/meta-utils.js.map +1 -1
- package/dist/src/prompts/reporte-ventas.js.map +1 -1
- package/dist/src/search-console/search-console-utils.js +275 -0
- package/dist/src/search-console/search-console-utils.js.map +7 -0
- package/dist/src/services/analytics/ga4-client.js.map +1 -1
- package/dist/src/services/analytics/oauth.js.map +1 -1
- package/dist/src/services/google-ads/google-ads-client.js.map +1 -1
- package/dist/src/services/mercadolibre/mercadolibre-api.js.map +1 -1
- package/dist/src/services/mercadolibre/mercadolibre-items.js.map +1 -1
- package/dist/src/services/mercadolibre/mercadolibre-orders.js.map +1 -1
- package/dist/src/services/mercadolibre/mercadolibre-questions.js.map +1 -1
- package/dist/src/services/meta/meta-ads.js.map +1 -1
- package/dist/src/services/meta/meta-api.js.map +1 -1
- package/dist/src/services/search-console/search-console-client.js.map +1 -1
- package/dist/src/services/search-console/search-console-utils.js.map +1 -1
- package/dist/src/services/vtex/vtex-api.js.map +1 -1
- package/dist/src/services/vtex/vtex-catalog-write.js.map +1 -1
- package/dist/src/services/vtex/vtex-catalog.js.map +1 -1
- package/dist/src/services/vtex/vtex-logistics.js.map +1 -1
- package/dist/src/services/vtex/vtex-orders-write.js.map +1 -1
- package/dist/src/services/vtex/vtex-orders.js.map +1 -1
- package/dist/src/services/vtex/vtex-pricing-write.js.map +1 -1
- package/dist/src/services/vtex/vtex-pricing.js.map +1 -1
- package/dist/src/services/vtex/vtex-write.js.map +1 -1
- package/dist/src/tools/analytics/attribution-gaps.js.map +1 -1
- package/dist/src/tools/analytics/channel-mix.js.map +1 -1
- package/dist/src/tools/analytics/ecommerce-tracking-health.js.map +1 -1
- package/dist/src/tools/analytics/engagement-overview.js.map +1 -1
- package/dist/src/tools/analytics/index.js.map +1 -1
- package/dist/src/tools/analytics/list-accessible-properties.js.map +1 -1
- package/dist/src/tools/analytics/property-info.js.map +1 -1
- package/dist/src/tools/analytics/revenue-by-channel.js.map +1 -1
- package/dist/src/tools/analytics/revenue-overview.js.map +1 -1
- package/dist/src/tools/analytics/revenue-trend.js.map +1 -1
- package/dist/src/tools/analytics/source-medium-breakdown.js.map +1 -1
- package/dist/src/tools/analytics/top-landing-pages.js.map +1 -1
- package/dist/src/tools/config/check-database-connection.js.map +1 -1
- package/dist/src/tools/config/index.js.map +1 -1
- package/dist/src/tools/config/list-profiles.js.map +1 -1
- package/dist/src/tools/google-ads/account-overview.js.map +1 -1
- package/dist/src/tools/google-ads/account-risks.js.map +1 -1
- package/dist/src/tools/google-ads/break-even-analysis.js.map +1 -1
- package/dist/src/tools/google-ads/campaign-performance.js.map +1 -1
- package/dist/src/tools/google-ads/channel-mix.js.map +1 -1
- package/dist/src/tools/google-ads/compare-accounts.js.map +1 -1
- package/dist/src/tools/google-ads/customer-clients.js.map +1 -1
- package/dist/src/tools/google-ads/customer-info.js.map +1 -1
- package/dist/src/tools/google-ads/index.js.map +1 -1
- package/dist/src/tools/google-ads/scaling-health.js.map +1 -1
- package/dist/src/tools/google-ads/search-terms-summary.js.map +1 -1
- package/dist/src/tools/google-ads/time-series.js.map +1 -1
- package/dist/src/tools/index.js.map +1 -1
- package/dist/src/tools/mercadolibre/answer-question.js.map +1 -1
- package/dist/src/tools/mercadolibre/create-item.js.map +1 -1
- package/dist/src/tools/mercadolibre/estimate-listing-fee.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-account-context.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-category-requirements.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-item-details.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-item-visits.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-listing-quality.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-order-details.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-orders-summary.js +366 -28
- package/dist/src/tools/mercadolibre/get-orders-summary.js.map +2 -2
- package/dist/src/tools/mercadolibre/get-sales-by-item.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-sales-trend.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-shipping-summary.js.map +1 -1
- package/dist/src/tools/mercadolibre/get-store-performance.js.map +1 -1
- package/dist/src/tools/mercadolibre/helpers.js.map +1 -1
- package/dist/src/tools/mercadolibre/index.js.map +1 -1
- package/dist/src/tools/mercadolibre/pause-or-reactivate-item.js.map +1 -1
- package/dist/src/tools/mercadolibre/predict-category.js.map +1 -1
- package/dist/src/tools/mercadolibre/profile-resolution.js.map +1 -1
- package/dist/src/tools/mercadolibre/search-items.js.map +1 -1
- package/dist/src/tools/mercadolibre/search-questions.js.map +1 -1
- package/dist/src/tools/mercadolibre/update-item-basic-fields.js.map +1 -1
- package/dist/src/tools/mercadolibre/update-item-description.js.map +1 -1
- package/dist/src/tools/mercadolibre/update-item-pictures.js.map +1 -1
- package/dist/src/tools/mercadolibre/validate-connection.js.map +1 -1
- package/dist/src/tools/mercadolibre/write-helpers.js.map +1 -1
- package/dist/src/tools/meta/account-overview.js.map +1 -1
- package/dist/src/tools/meta/ad-account-info.js.map +1 -1
- package/dist/src/tools/meta/ads-performance.js.map +1 -1
- package/dist/src/tools/meta/campaign-performance.js.map +1 -1
- package/dist/src/tools/meta/index.js.map +1 -1
- package/dist/src/tools/meta/list-accessible-ad-accounts.js.map +1 -1
- package/dist/src/tools/meta/list-accessible-businesses.js.map +1 -1
- package/dist/src/tools/meta/placement-mix.js.map +1 -1
- package/dist/src/tools/meta/time-series.js.map +1 -1
- package/dist/src/tools/search-console/country-breakdown.js.map +1 -1
- package/dist/src/tools/search-console/device-breakdown.js.map +1 -1
- package/dist/src/tools/search-console/high-impression-low-click-queries.js.map +1 -1
- package/dist/src/tools/search-console/index.js.map +1 -1
- package/dist/src/tools/search-console/list-accessible-sites.js.map +1 -1
- package/dist/src/tools/search-console/low-ctr-opportunities.js.map +1 -1
- package/dist/src/tools/search-console/page-performance.js.map +1 -1
- package/dist/src/tools/search-console/product-demand-low-capture-queries.js.map +1 -1
- package/dist/src/tools/search-console/query-page-matrix.js.map +1 -1
- package/dist/src/tools/search-console/query-performance.js.map +1 -1
- package/dist/src/tools/search-console/quick-win-opportunities.js.map +1 -1
- package/dist/src/tools/search-console/rising-non-brand-queries.js.map +1 -1
- package/dist/src/tools/search-console/search-performance.js.map +1 -1
- package/dist/src/tools/search-console/site-context.js.map +1 -1
- package/dist/src/tools/search-console/visibility-declines.js.map +1 -1
- package/dist/src/tools/vtex/activate-sku.js.map +1 -1
- package/dist/src/tools/vtex/add-order-tracking.js.map +1 -1
- package/dist/src/tools/vtex/associate-specification.js.map +1 -1
- package/dist/src/tools/vtex/attach-catalog-image.js.map +1 -1
- package/dist/src/tools/vtex/cancel-order.js.map +1 -1
- package/dist/src/tools/vtex/computed-price.js.map +1 -1
- package/dist/src/tools/vtex/create-brand.js.map +1 -1
- package/dist/src/tools/vtex/create-category.js.map +1 -1
- package/dist/src/tools/vtex/create-product-with-sku.js.map +1 -1
- package/dist/src/tools/vtex/create-product.js.map +1 -1
- package/dist/src/tools/vtex/create-sku.js.map +1 -1
- package/dist/src/tools/vtex/create-specification-value.js.map +1 -1
- package/dist/src/tools/vtex/create-specification.js.map +1 -1
- package/dist/src/tools/vtex/deactivate-sku.js.map +1 -1
- package/dist/src/tools/vtex/delete-fixed-price.js.map +1 -1
- package/dist/src/tools/vtex/index.js.map +1 -1
- package/dist/src/tools/vtex/inventory-check.js.map +1 -1
- package/dist/src/tools/vtex/invoice-order.js.map +1 -1
- package/dist/src/tools/vtex/order-details.js.map +1 -1
- package/dist/src/tools/vtex/orders-summary.js.map +1 -1
- package/dist/src/tools/vtex/product-offers.js.map +1 -1
- package/dist/src/tools/vtex/profile-resolution.js.map +1 -1
- package/dist/src/tools/vtex/sku-offers.js.map +1 -1
- package/dist/src/tools/vtex/sku-price.js.map +1 -1
- package/dist/src/tools/vtex/toggle-unlimited-quantity.js.map +1 -1
- package/dist/src/tools/vtex/update-inventory.js.map +1 -1
- package/dist/src/tools/vtex/update-lead-time.js.map +1 -1
- package/dist/src/tools/vtex/update-product-basic-fields.js.map +1 -1
- package/dist/src/tools/vtex/update-sku-basic-fields.js.map +1 -1
- package/dist/src/tools/vtex/update-sku-price.js.map +1 -1
- package/dist/src/tools/vtex/upsert-fixed-price.js.map +1 -1
- package/dist/src/tools/vtex/warehouse-inventory.js.map +1 -1
- package/dist/src/tools/vtex/write-helpers.js.map +1 -1
- package/dist/src/utils/case-conversion.js.map +1 -1
- package/dist/src/utils/currency.js.map +1 -1
- package/dist/src/utils/format-order-details.js.map +1 -1
- package/dist/src/utils/google-ads.js.map +1 -1
- package/dist/src/utils/money.js.map +1 -1
- package/dist/src/utils/order-status.js.map +1 -1
- package/dist/src/utils/pagination.js.map +1 -1
- package/dist/src/utils/strip-payload.js.map +1 -1
- package/dist/src/utils/type-guards.js.map +1 -1
- package/package.json +1 -1
- package/public/icon.svg +6 -6
|
@@ -0,0 +1,275 @@
|
|
|
1
|
+
import { z } from "zod";
|
|
2
|
+
const searchConsoleDateRegex = /^\d{4}-\d{2}-\d{2}$/;
|
|
3
|
+
const searchConsoleSearchTypes = ["web", "image", "video", "news", "discover"];
|
|
4
|
+
const searchConsoleDimensions = [
|
|
5
|
+
"query",
|
|
6
|
+
"page",
|
|
7
|
+
"country",
|
|
8
|
+
"device",
|
|
9
|
+
"date",
|
|
10
|
+
"searchAppearance"
|
|
11
|
+
];
|
|
12
|
+
const searchConsoleFilterOperators = [
|
|
13
|
+
"contains",
|
|
14
|
+
"equals",
|
|
15
|
+
"notContains",
|
|
16
|
+
"notEquals",
|
|
17
|
+
"includingRegex",
|
|
18
|
+
"excludingRegex"
|
|
19
|
+
];
|
|
20
|
+
const searchConsoleAggregationTypes = ["auto", "byPage", "byProperty"];
|
|
21
|
+
const searchConsoleDimensionFilterSchema = z.object({
|
|
22
|
+
dimension: z.enum(searchConsoleDimensions).describe("Dimension to filter on in Search Console (query, page, country, device, date, or searchAppearance)"),
|
|
23
|
+
operator: z.enum(searchConsoleFilterOperators).describe("Filter operator supported by Search Console."),
|
|
24
|
+
expression: z.string().min(1).describe("Filter value or regex expression.")
|
|
25
|
+
});
|
|
26
|
+
const siteUrlSchemaField = z.string().optional().describe(
|
|
27
|
+
"Search Console site URL or domain property identifier (for example https://www.example.com/ or sc-domain:example.com). If omitted, the tool expects the site to be configured elsewhere."
|
|
28
|
+
);
|
|
29
|
+
const profileIdSchemaField = z.string().optional().describe("Optional business profile identifier for future multi-profile resolution. Defaults to the active profile when supported.");
|
|
30
|
+
const startDateSchemaField = z.string().regex(searchConsoleDateRegex).describe("Start date in YYYY-MM-DD format.");
|
|
31
|
+
const endDateSchemaField = z.string().regex(searchConsoleDateRegex).describe("End date in YYYY-MM-DD format.");
|
|
32
|
+
const currentStartDateSchemaField = z.string().regex(searchConsoleDateRegex).describe("Current comparison period start date in YYYY-MM-DD format.");
|
|
33
|
+
const currentEndDateSchemaField = z.string().regex(searchConsoleDateRegex).describe("Current comparison period end date in YYYY-MM-DD format.");
|
|
34
|
+
const previousStartDateSchemaField = z.string().regex(searchConsoleDateRegex).describe("Previous comparison period start date in YYYY-MM-DD format.");
|
|
35
|
+
const previousEndDateSchemaField = z.string().regex(searchConsoleDateRegex).describe("Previous comparison period end date in YYYY-MM-DD format.");
|
|
36
|
+
const rowLimitSchemaField = z.number().int().min(1).max(25e3).optional().describe("Maximum number of rows to return. Search Console supports up to 25,000 rows per request.");
|
|
37
|
+
const startRowSchemaField = z.number().int().min(0).optional().describe("Zero-based row offset for paginated Search Console requests.");
|
|
38
|
+
const searchTypeSchemaField = z.enum(searchConsoleSearchTypes).optional().describe("Search type to query. Defaults to web.");
|
|
39
|
+
const brandTermsSchemaField = z.array(z.string().min(1)).max(50).optional().describe("Brand tokens used to separate branded from non-branded queries.");
|
|
40
|
+
function resolveSearchConsoleSiteUrl(siteUrl, profileId) {
|
|
41
|
+
const explicitSiteUrl = siteUrl?.trim();
|
|
42
|
+
if (explicitSiteUrl) {
|
|
43
|
+
return explicitSiteUrl;
|
|
44
|
+
}
|
|
45
|
+
if (profileId?.trim()) {
|
|
46
|
+
throw new Error(
|
|
47
|
+
`Missing Search Console site URL. profileId "${profileId}" was provided, but business-data based site resolution is not implemented yet. Pass siteUrl explicitly or configure Search Console site resolution first.`
|
|
48
|
+
);
|
|
49
|
+
}
|
|
50
|
+
throw new Error(
|
|
51
|
+
"Missing Search Console site URL. Pass siteUrl explicitly or first call gsc_list_accessible_sites to discover available properties."
|
|
52
|
+
);
|
|
53
|
+
}
|
|
54
|
+
function inferSearchConsolePropertyType(siteUrl) {
|
|
55
|
+
if (siteUrl.startsWith("sc-domain:")) {
|
|
56
|
+
return "domain";
|
|
57
|
+
}
|
|
58
|
+
if (/^https?:\/\//.test(siteUrl)) {
|
|
59
|
+
return "url_prefix";
|
|
60
|
+
}
|
|
61
|
+
return "unknown";
|
|
62
|
+
}
|
|
63
|
+
function normalizePermissionLevel(permissionLevel) {
|
|
64
|
+
return permissionLevel?.trim().toLowerCase() || void 0;
|
|
65
|
+
}
|
|
66
|
+
function normalizeSearchConsoleSiteEntry(siteEntry) {
|
|
67
|
+
return {
|
|
68
|
+
site_url: siteEntry.siteUrl,
|
|
69
|
+
property_type: inferSearchConsolePropertyType(siteEntry.siteUrl),
|
|
70
|
+
permission_level: normalizePermissionLevel(siteEntry.permissionLevel)
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
function toPercent(numerator, denominator) {
|
|
74
|
+
if (!Number.isFinite(numerator) || !Number.isFinite(denominator) || denominator <= 0) {
|
|
75
|
+
return 0;
|
|
76
|
+
}
|
|
77
|
+
return numerator / denominator * 100;
|
|
78
|
+
}
|
|
79
|
+
function round(value, decimals = 2) {
|
|
80
|
+
if (!Number.isFinite(value)) {
|
|
81
|
+
return 0;
|
|
82
|
+
}
|
|
83
|
+
const factor = 10 ** decimals;
|
|
84
|
+
return Math.round(value * factor) / factor;
|
|
85
|
+
}
|
|
86
|
+
function normalizeSearchConsoleRow(row, dimensions = []) {
|
|
87
|
+
const clicks = row.clicks ?? 0;
|
|
88
|
+
const impressions = row.impressions ?? 0;
|
|
89
|
+
const ctr = typeof row.ctr === "number" ? row.ctr * 100 : toPercent(clicks, impressions);
|
|
90
|
+
const keys = row.keys ?? [];
|
|
91
|
+
return {
|
|
92
|
+
keys,
|
|
93
|
+
dimensions: dimensions.reduce((accumulator, dimension, index) => {
|
|
94
|
+
accumulator[dimension] = keys[index] ?? "";
|
|
95
|
+
return accumulator;
|
|
96
|
+
}, {}),
|
|
97
|
+
clicks: round(clicks, 2),
|
|
98
|
+
impressions: round(impressions, 2),
|
|
99
|
+
ctr_percent: round(ctr, 2),
|
|
100
|
+
position: round(row.position ?? 0, 2)
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
function buildSearchConsoleQueryBody(input) {
|
|
104
|
+
const dimensionFilterGroups = input.dimensionFilters && input.dimensionFilters.length > 0 ? [
|
|
105
|
+
{
|
|
106
|
+
groupType: "and",
|
|
107
|
+
filters: input.dimensionFilters
|
|
108
|
+
}
|
|
109
|
+
] : void 0;
|
|
110
|
+
return {
|
|
111
|
+
startDate: input.startDate,
|
|
112
|
+
endDate: input.endDate,
|
|
113
|
+
dimensions: input.dimensions,
|
|
114
|
+
type: input.searchType ?? "web",
|
|
115
|
+
dimensionFilterGroups,
|
|
116
|
+
aggregationType: input.aggregationType,
|
|
117
|
+
rowLimit: input.rowLimit,
|
|
118
|
+
startRow: input.startRow
|
|
119
|
+
};
|
|
120
|
+
}
|
|
121
|
+
function buildPaginationMetadata(input) {
|
|
122
|
+
const startRow = input.startRow ?? 0;
|
|
123
|
+
const rowLimit = input.rowLimit;
|
|
124
|
+
const isPartial = Boolean(rowLimit) && input.returnedRows === rowLimit;
|
|
125
|
+
const nextStartRow = isPartial ? startRow + input.returnedRows : void 0;
|
|
126
|
+
return {
|
|
127
|
+
is_partial: isPartial,
|
|
128
|
+
start_row: startRow,
|
|
129
|
+
row_limit: rowLimit,
|
|
130
|
+
returned_rows: input.returnedRows,
|
|
131
|
+
next_start_row: nextStartRow,
|
|
132
|
+
continuation_instructions: nextStartRow ? "This Search Console response may be truncated at row_limit. If you need additional rows, call the same tool again with startRow set to next_start_row." : void 0
|
|
133
|
+
};
|
|
134
|
+
}
|
|
135
|
+
function parseBrandTerms(brandTerms) {
|
|
136
|
+
return (brandTerms ?? []).map((term) => term.trim().toLowerCase()).filter(Boolean);
|
|
137
|
+
}
|
|
138
|
+
function isBrandedQuery(query, brandTerms) {
|
|
139
|
+
const normalizedQuery = query.trim().toLowerCase();
|
|
140
|
+
const normalizedBrandTerms = parseBrandTerms(brandTerms);
|
|
141
|
+
if (!normalizedQuery || normalizedBrandTerms.length === 0) {
|
|
142
|
+
return false;
|
|
143
|
+
}
|
|
144
|
+
return normalizedBrandTerms.some((brandTerm) => normalizedQuery.includes(brandTerm));
|
|
145
|
+
}
|
|
146
|
+
function classifyBrandLabel(query, brandTerms) {
|
|
147
|
+
return isBrandedQuery(query, brandTerms) ? "brand" : "non_brand";
|
|
148
|
+
}
|
|
149
|
+
function computeChangePercent(currentValue, previousValue) {
|
|
150
|
+
if (!Number.isFinite(currentValue) || !Number.isFinite(previousValue)) {
|
|
151
|
+
return null;
|
|
152
|
+
}
|
|
153
|
+
if (previousValue === 0) {
|
|
154
|
+
if (currentValue === 0) {
|
|
155
|
+
return 0;
|
|
156
|
+
}
|
|
157
|
+
return null;
|
|
158
|
+
}
|
|
159
|
+
return round((currentValue - previousValue) / previousValue * 100, 2);
|
|
160
|
+
}
|
|
161
|
+
function sumSearchConsoleRows(rows) {
|
|
162
|
+
const totals = rows.reduce(
|
|
163
|
+
(accumulator, row) => {
|
|
164
|
+
accumulator.clicks += row.clicks;
|
|
165
|
+
accumulator.impressions += row.impressions;
|
|
166
|
+
accumulator.weightedPosition += row.position * row.impressions;
|
|
167
|
+
return accumulator;
|
|
168
|
+
},
|
|
169
|
+
{
|
|
170
|
+
clicks: 0,
|
|
171
|
+
impressions: 0,
|
|
172
|
+
weightedPosition: 0
|
|
173
|
+
}
|
|
174
|
+
);
|
|
175
|
+
return {
|
|
176
|
+
clicks: round(totals.clicks, 2),
|
|
177
|
+
impressions: round(totals.impressions, 2),
|
|
178
|
+
ctrPercent: round(toPercent(totals.clicks, totals.impressions), 2),
|
|
179
|
+
averagePosition: totals.impressions > 0 ? round(totals.weightedPosition / totals.impressions, 2) : 0
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
function buildComparisonMap(rows, keyBuilder) {
|
|
183
|
+
return new Map(rows.map((row) => [keyBuilder(row), row]));
|
|
184
|
+
}
|
|
185
|
+
function tokenizeQuery(query) {
|
|
186
|
+
return query.toLowerCase().split(/[^a-z0-9áéíóúüñ]+/i).map((token) => token.trim()).filter(Boolean);
|
|
187
|
+
}
|
|
188
|
+
function looksLikeCommercialQuery(query) {
|
|
189
|
+
const tokens = tokenizeQuery(query);
|
|
190
|
+
const commercialTerms = /* @__PURE__ */ new Set([
|
|
191
|
+
"comprar",
|
|
192
|
+
"precio",
|
|
193
|
+
"precios",
|
|
194
|
+
"cuanto",
|
|
195
|
+
"oferta",
|
|
196
|
+
"ofertas",
|
|
197
|
+
"envio",
|
|
198
|
+
"shop",
|
|
199
|
+
"tienda",
|
|
200
|
+
"modelo",
|
|
201
|
+
"medida",
|
|
202
|
+
"ml",
|
|
203
|
+
"cm",
|
|
204
|
+
"kg"
|
|
205
|
+
]);
|
|
206
|
+
return tokens.some((token) => commercialTerms.has(token));
|
|
207
|
+
}
|
|
208
|
+
function looksLikeProductQuery(query, productTerms) {
|
|
209
|
+
const normalizedProductTerms = parseBrandTerms(productTerms);
|
|
210
|
+
const normalizedQuery = query.toLowerCase();
|
|
211
|
+
if (normalizedProductTerms.some((term) => normalizedQuery.includes(term))) {
|
|
212
|
+
return true;
|
|
213
|
+
}
|
|
214
|
+
return looksLikeCommercialQuery(query) || tokenizeQuery(query).length >= 3;
|
|
215
|
+
}
|
|
216
|
+
function classifyQuickWinType(position, ctrPercent) {
|
|
217
|
+
if (position <= 8 && ctrPercent <= 3) {
|
|
218
|
+
return "ctr";
|
|
219
|
+
}
|
|
220
|
+
return "position";
|
|
221
|
+
}
|
|
222
|
+
function scoreHighImpressionLowClickOpportunity(input) {
|
|
223
|
+
const impressionScore = Math.min(input.impressions / 100, 60);
|
|
224
|
+
const ctrPenalty = Math.max(0, 15 - input.ctrPercent) * 2;
|
|
225
|
+
const positionBonus = input.position > 0 && input.position <= 20 ? Math.max(0, 20 - input.position) : 0;
|
|
226
|
+
return round(impressionScore + ctrPenalty + positionBonus, 2);
|
|
227
|
+
}
|
|
228
|
+
function scoreQuickWinOpportunity(input) {
|
|
229
|
+
const impressionWeight = Math.min(input.impressions / 150, 50);
|
|
230
|
+
const ctrWeight = Math.max(0, 10 - input.ctrPercent) * 3;
|
|
231
|
+
const positionWeight = input.position > 0 && input.position <= 20 ? Math.max(0, 18 - input.position) * 1.8 : 0;
|
|
232
|
+
return round(impressionWeight + ctrWeight + positionWeight, 2);
|
|
233
|
+
}
|
|
234
|
+
export {
|
|
235
|
+
brandTermsSchemaField,
|
|
236
|
+
buildComparisonMap,
|
|
237
|
+
buildPaginationMetadata,
|
|
238
|
+
buildSearchConsoleQueryBody,
|
|
239
|
+
classifyBrandLabel,
|
|
240
|
+
classifyQuickWinType,
|
|
241
|
+
computeChangePercent,
|
|
242
|
+
currentEndDateSchemaField,
|
|
243
|
+
currentStartDateSchemaField,
|
|
244
|
+
endDateSchemaField,
|
|
245
|
+
inferSearchConsolePropertyType,
|
|
246
|
+
isBrandedQuery,
|
|
247
|
+
looksLikeCommercialQuery,
|
|
248
|
+
looksLikeProductQuery,
|
|
249
|
+
normalizePermissionLevel,
|
|
250
|
+
normalizeSearchConsoleRow,
|
|
251
|
+
normalizeSearchConsoleSiteEntry,
|
|
252
|
+
parseBrandTerms,
|
|
253
|
+
previousEndDateSchemaField,
|
|
254
|
+
previousStartDateSchemaField,
|
|
255
|
+
profileIdSchemaField,
|
|
256
|
+
resolveSearchConsoleSiteUrl,
|
|
257
|
+
round,
|
|
258
|
+
rowLimitSchemaField,
|
|
259
|
+
scoreHighImpressionLowClickOpportunity,
|
|
260
|
+
scoreQuickWinOpportunity,
|
|
261
|
+
searchConsoleAggregationTypes,
|
|
262
|
+
searchConsoleDateRegex,
|
|
263
|
+
searchConsoleDimensionFilterSchema,
|
|
264
|
+
searchConsoleDimensions,
|
|
265
|
+
searchConsoleFilterOperators,
|
|
266
|
+
searchConsoleSearchTypes,
|
|
267
|
+
searchTypeSchemaField,
|
|
268
|
+
siteUrlSchemaField,
|
|
269
|
+
startDateSchemaField,
|
|
270
|
+
startRowSchemaField,
|
|
271
|
+
sumSearchConsoleRows,
|
|
272
|
+
toPercent,
|
|
273
|
+
tokenizeQuery
|
|
274
|
+
};
|
|
275
|
+
//# sourceMappingURL=search-console-utils.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/search-console/search-console-utils.ts"],
|
|
4
|
+
"sourcesContent": ["import { z } from \"zod\";\n\nimport type {\n SearchConsoleDimensionFilter,\n SearchConsoleDimensionFilterGroup,\n SearchConsoleSearchAnalyticsRequest,\n SearchConsoleSearchAnalyticsRow,\n SearchConsoleSiteEntry,\n} from \"../services/search-console/search-console-client.js\";\n\nexport const searchConsoleDateRegex = /^\\d{4}-\\d{2}-\\d{2}$/;\n\nexport const searchConsoleSearchTypes = [\"web\", \"image\", \"video\", \"news\", \"discover\"] as const;\nexport const searchConsoleDimensions = [\n \"query\",\n \"page\",\n \"country\",\n \"device\",\n \"date\",\n \"searchAppearance\",\n] as const;\nexport const searchConsoleFilterOperators = [\n \"contains\",\n \"equals\",\n \"notContains\",\n \"notEquals\",\n \"includingRegex\",\n \"excludingRegex\",\n] as const;\nexport const searchConsoleAggregationTypes = [\"auto\", \"byPage\", \"byProperty\"] as const;\n\nexport const searchConsoleDimensionFilterSchema = z.object({\n dimension: z\n .enum(searchConsoleDimensions)\n .describe(\"Dimension to filter on in Search Console (query, page, country, device, date, or searchAppearance)\"),\n operator: z\n .enum(searchConsoleFilterOperators)\n .describe(\"Filter operator supported by Search Console.\"),\n expression: z.string().min(1).describe(\"Filter value or regex expression.\"),\n});\n\nexport const siteUrlSchemaField = z\n .string()\n .optional()\n .describe(\n \"Search Console site URL or domain property identifier (for example https://www.example.com/ or sc-domain:example.com). If omitted, the tool expects the site to be configured elsewhere.\"\n );\n\nexport const profileIdSchemaField = z\n .string()\n .optional()\n .describe(\"Optional business profile identifier for future multi-profile resolution. Defaults to the active profile when supported.\");\n\nexport const startDateSchemaField = z\n .string()\n .regex(searchConsoleDateRegex)\n .describe(\"Start date in YYYY-MM-DD format.\");\n\nexport const endDateSchemaField = z\n .string()\n .regex(searchConsoleDateRegex)\n .describe(\"End date in YYYY-MM-DD format.\");\n\nexport const currentStartDateSchemaField = z\n .string()\n .regex(searchConsoleDateRegex)\n .describe(\"Current comparison period start date in YYYY-MM-DD format.\");\n\nexport const currentEndDateSchemaField = z\n .string()\n .regex(searchConsoleDateRegex)\n .describe(\"Current comparison period end date in YYYY-MM-DD format.\");\n\nexport const previousStartDateSchemaField = z\n .string()\n .regex(searchConsoleDateRegex)\n .describe(\"Previous comparison period start date in YYYY-MM-DD format.\");\n\nexport const previousEndDateSchemaField = z\n .string()\n .regex(searchConsoleDateRegex)\n .describe(\"Previous comparison period end date in YYYY-MM-DD format.\");\n\nexport const rowLimitSchemaField = z\n .number()\n .int()\n .min(1)\n .max(25000)\n .optional()\n .describe(\"Maximum number of rows to return. Search Console supports up to 25,000 rows per request.\");\n\nexport const startRowSchemaField = z\n .number()\n .int()\n .min(0)\n .optional()\n .describe(\"Zero-based row offset for paginated Search Console requests.\");\n\nexport const searchTypeSchemaField = z\n .enum(searchConsoleSearchTypes)\n .optional()\n .describe(\"Search type to query. Defaults to web.\");\n\nexport const brandTermsSchemaField = z\n .array(z.string().min(1))\n .max(50)\n .optional()\n .describe(\"Brand tokens used to separate branded from non-branded queries.\");\n\nexport function resolveSearchConsoleSiteUrl(siteUrl?: string, profileId?: string): string {\n const explicitSiteUrl = siteUrl?.trim();\n if (explicitSiteUrl) {\n return explicitSiteUrl;\n }\n\n if (profileId?.trim()) {\n throw new Error(\n `Missing Search Console site URL. profileId \"${profileId}\" was provided, but business-data based site resolution is not implemented yet. Pass siteUrl explicitly or configure Search Console site resolution first.`\n );\n }\n\n throw new Error(\n \"Missing Search Console site URL. Pass siteUrl explicitly or first call gsc_list_accessible_sites to discover available properties.\"\n );\n}\n\nexport function inferSearchConsolePropertyType(\n siteUrl: string\n): \"domain\" | \"url_prefix\" | \"unknown\" {\n if (siteUrl.startsWith(\"sc-domain:\")) {\n return \"domain\";\n }\n\n if (/^https?:\\/\\//.test(siteUrl)) {\n return \"url_prefix\";\n }\n\n return \"unknown\";\n}\n\nexport function normalizePermissionLevel(permissionLevel?: string): string | undefined {\n return permissionLevel?.trim().toLowerCase() || undefined;\n}\n\nexport function normalizeSearchConsoleSiteEntry(siteEntry: SearchConsoleSiteEntry) {\n return {\n site_url: siteEntry.siteUrl,\n property_type: inferSearchConsolePropertyType(siteEntry.siteUrl),\n permission_level: normalizePermissionLevel(siteEntry.permissionLevel),\n };\n}\n\nexport function toPercent(numerator: number, denominator: number): number {\n if (!Number.isFinite(numerator) || !Number.isFinite(denominator) || denominator <= 0) {\n return 0;\n }\n\n return (numerator / denominator) * 100;\n}\n\nexport function round(value: number, decimals = 2): number {\n if (!Number.isFinite(value)) {\n return 0;\n }\n\n const factor = 10 ** decimals;\n return Math.round(value * factor) / factor;\n}\n\nexport function normalizeSearchConsoleRow(\n row: SearchConsoleSearchAnalyticsRow,\n dimensions: readonly string[] = []\n) {\n const clicks = row.clicks ?? 0;\n const impressions = row.impressions ?? 0;\n const ctr = typeof row.ctr === \"number\" ? row.ctr * 100 : toPercent(clicks, impressions);\n const keys = row.keys ?? [];\n\n return {\n keys,\n dimensions: dimensions.reduce<Record<string, string>>((accumulator, dimension, index) => {\n accumulator[dimension] = keys[index] ?? \"\";\n return accumulator;\n }, {}),\n clicks: round(clicks, 2),\n impressions: round(impressions, 2),\n ctr_percent: round(ctr, 2),\n position: round(row.position ?? 0, 2),\n };\n}\n\nexport function buildSearchConsoleQueryBody(input: {\n startDate: string;\n endDate: string;\n dimensions?: string[];\n searchType?: string;\n dimensionFilters?: SearchConsoleDimensionFilter[];\n aggregationType?: string;\n rowLimit?: number;\n startRow?: number;\n}): SearchConsoleSearchAnalyticsRequest {\n const dimensionFilterGroups: SearchConsoleDimensionFilterGroup[] | undefined =\n input.dimensionFilters && input.dimensionFilters.length > 0\n ? [\n {\n groupType: \"and\",\n filters: input.dimensionFilters,\n },\n ]\n : undefined;\n\n return {\n startDate: input.startDate,\n endDate: input.endDate,\n dimensions: input.dimensions,\n type: input.searchType ?? \"web\",\n dimensionFilterGroups,\n aggregationType: input.aggregationType,\n rowLimit: input.rowLimit,\n startRow: input.startRow,\n };\n}\n\nexport function buildPaginationMetadata(input: {\n startRow?: number;\n rowLimit?: number;\n returnedRows: number;\n}) {\n const startRow = input.startRow ?? 0;\n const rowLimit = input.rowLimit;\n const isPartial = Boolean(rowLimit) && input.returnedRows === rowLimit;\n const nextStartRow = isPartial ? startRow + input.returnedRows : undefined;\n\n return {\n is_partial: isPartial,\n start_row: startRow,\n row_limit: rowLimit,\n returned_rows: input.returnedRows,\n next_start_row: nextStartRow,\n continuation_instructions: nextStartRow\n ? \"This Search Console response may be truncated at row_limit. If you need additional rows, call the same tool again with startRow set to next_start_row.\"\n : undefined,\n };\n}\n\nexport function parseBrandTerms(brandTerms?: string[]): string[] {\n return (brandTerms ?? []).map((term) => term.trim().toLowerCase()).filter(Boolean);\n}\n\nexport function isBrandedQuery(query: string, brandTerms?: string[]): boolean {\n const normalizedQuery = query.trim().toLowerCase();\n const normalizedBrandTerms = parseBrandTerms(brandTerms);\n\n if (!normalizedQuery || normalizedBrandTerms.length === 0) {\n return false;\n }\n\n return normalizedBrandTerms.some((brandTerm) => normalizedQuery.includes(brandTerm));\n}\n\nexport function classifyBrandLabel(query: string, brandTerms?: string[]): \"brand\" | \"non_brand\" {\n return isBrandedQuery(query, brandTerms) ? \"brand\" : \"non_brand\";\n}\n\nexport function computeChangePercent(currentValue: number, previousValue: number): number | null {\n if (!Number.isFinite(currentValue) || !Number.isFinite(previousValue)) {\n return null;\n }\n\n if (previousValue === 0) {\n if (currentValue === 0) {\n return 0;\n }\n\n return null;\n }\n\n return round(((currentValue - previousValue) / previousValue) * 100, 2);\n}\n\nexport function sumSearchConsoleRows(\n rows: Array<ReturnType<typeof normalizeSearchConsoleRow>>\n): {\n clicks: number;\n impressions: number;\n ctrPercent: number;\n averagePosition: number;\n} {\n const totals = rows.reduce(\n (accumulator, row) => {\n accumulator.clicks += row.clicks;\n accumulator.impressions += row.impressions;\n accumulator.weightedPosition += row.position * row.impressions;\n return accumulator;\n },\n {\n clicks: 0,\n impressions: 0,\n weightedPosition: 0,\n }\n );\n\n return {\n clicks: round(totals.clicks, 2),\n impressions: round(totals.impressions, 2),\n ctrPercent: round(toPercent(totals.clicks, totals.impressions), 2),\n averagePosition: totals.impressions > 0 ? round(totals.weightedPosition / totals.impressions, 2) : 0,\n };\n}\n\nexport function buildComparisonMap(\n rows: Array<ReturnType<typeof normalizeSearchConsoleRow>>,\n keyBuilder: (row: ReturnType<typeof normalizeSearchConsoleRow>) => string\n) {\n return new Map(rows.map((row) => [keyBuilder(row), row]));\n}\n\nexport function tokenizeQuery(query: string): string[] {\n return query\n .toLowerCase()\n .split(/[^a-z0-9\u00E1\u00E9\u00ED\u00F3\u00FA\u00FC\u00F1]+/i)\n .map((token) => token.trim())\n .filter(Boolean);\n}\n\nexport function looksLikeCommercialQuery(query: string): boolean {\n const tokens = tokenizeQuery(query);\n const commercialTerms = new Set([\n \"comprar\",\n \"precio\",\n \"precios\",\n \"cuanto\",\n \"oferta\",\n \"ofertas\",\n \"envio\",\n \"shop\",\n \"tienda\",\n \"modelo\",\n \"medida\",\n \"ml\",\n \"cm\",\n \"kg\",\n ]);\n\n return tokens.some((token) => commercialTerms.has(token));\n}\n\nexport function looksLikeProductQuery(query: string, productTerms?: string[]): boolean {\n const normalizedProductTerms = parseBrandTerms(productTerms);\n const normalizedQuery = query.toLowerCase();\n\n if (normalizedProductTerms.some((term) => normalizedQuery.includes(term))) {\n return true;\n }\n\n return looksLikeCommercialQuery(query) || tokenizeQuery(query).length >= 3;\n}\n\nexport function classifyQuickWinType(position: number, ctrPercent: number): \"ctr\" | \"position\" {\n if (position <= 8 && ctrPercent <= 3) {\n return \"ctr\";\n }\n\n return \"position\";\n}\n\nexport function scoreHighImpressionLowClickOpportunity(input: {\n impressions: number;\n ctrPercent: number;\n position: number;\n}): number {\n const impressionScore = Math.min(input.impressions / 100, 60);\n const ctrPenalty = Math.max(0, 15 - input.ctrPercent) * 2;\n const positionBonus = input.position > 0 && input.position <= 20 ? Math.max(0, 20 - input.position) : 0;\n\n return round(impressionScore + ctrPenalty + positionBonus, 2);\n}\n\nexport function scoreQuickWinOpportunity(input: {\n impressions: number;\n ctrPercent: number;\n position: number;\n}): number {\n const impressionWeight = Math.min(input.impressions / 150, 50);\n const ctrWeight = Math.max(0, 10 - input.ctrPercent) * 3;\n const positionWeight = input.position > 0 && input.position <= 20 ? Math.max(0, 18 - input.position) * 1.8 : 0;\n\n return round(impressionWeight + ctrWeight + positionWeight, 2);\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/services/analytics/ga4-client.ts"],
|
|
4
|
-
"sourcesContent": ["import { refreshGoogleAccessToken } from \"./oauth.js\";\
|
|
4
|
+
"sourcesContent": ["import { refreshGoogleAccessToken } from \"./oauth.js\";\n\nconst GA4_DATA_BASE_URL = \"https://analyticsdata.googleapis.com/v1beta\";\nconst GA4_ADMIN_BASE_URL = \"https://analyticsadmin.googleapis.com/v1alpha\";\n\nexport interface Ga4PropertySummary {\n property: string;\n displayName: string;\n propertyType?: string;\n parent?: string;\n}\n\nexport interface Ga4AccountSummary {\n name: string;\n account: string;\n displayName: string;\n propertySummaries?: Ga4PropertySummary[];\n}\n\nexport interface Ga4AccountSummariesResponse {\n accountSummaries?: Ga4AccountSummary[];\n nextPageToken?: string;\n}\n\nexport interface ListAccountSummariesResult {\n accountSummaries: Ga4AccountSummary[];\n nextPageToken?: string;\n}\n\nexport interface Ga4MetadataItem {\n apiName: string;\n uiName?: string;\n description?: string;\n category?: string;\n type?: string;\n customDefinition?: boolean;\n deprecatedApiNames?: string[];\n}\n\nexport interface Ga4CompatibilityItem {\n compatibility: string;\n dimensionMetadata?: Ga4MetadataItem;\n metricMetadata?: Ga4MetadataItem;\n}\n\nexport interface Ga4MetadataResponse {\n name: string;\n dimensions?: Ga4MetadataItem[];\n metrics?: Ga4MetadataItem[];\n comparisons?: Ga4MetadataItem[];\n}\n\nexport interface Ga4MetricHeader {\n name: string;\n type?: string;\n}\n\nexport interface Ga4DimensionHeader {\n name: string;\n}\n\nexport interface Ga4DimensionValue {\n value: string;\n}\n\nexport interface Ga4MetricValue {\n value: string;\n}\n\nexport interface Ga4ReportRow {\n dimensionValues?: Ga4DimensionValue[];\n metricValues?: Ga4MetricValue[];\n}\n\nexport interface Ga4PropertyQuotaBucket {\n consumed?: number;\n remaining?: number;\n}\n\nexport interface Ga4PropertyQuota {\n tokensPerDay?: Ga4PropertyQuotaBucket;\n tokensPerHour?: Ga4PropertyQuotaBucket;\n concurrentRequests?: Ga4PropertyQuotaBucket;\n serverErrorsPerProjectPerHour?: Ga4PropertyQuotaBucket;\n potentiallyThresholdedRequestsPerHour?: Ga4PropertyQuotaBucket;\n tokensPerProjectPerHour?: Ga4PropertyQuotaBucket;\n}\n\nexport interface Ga4ReportResponse {\n dimensionHeaders?: Ga4DimensionHeader[];\n metricHeaders?: Ga4MetricHeader[];\n rows?: Ga4ReportRow[];\n rowCount?: number;\n metadata?: {\n currencyCode?: string;\n timeZone?: string;\n [key: string]: unknown;\n };\n propertyQuota?: Ga4PropertyQuota;\n kind?: string;\n}\n\nexport interface Ga4RealtimeReportResponse {\n dimensionHeaders?: Ga4DimensionHeader[];\n metricHeaders?: Ga4MetricHeader[];\n rows?: Ga4ReportRow[];\n rowCount?: number;\n kind?: string;\n}\n\nexport interface Ga4CompatibilityResponse {\n dimensionCompatibilities?: Ga4CompatibilityItem[];\n metricCompatibilities?: Ga4CompatibilityItem[];\n}\n\ninterface GoogleApiErrorPayload {\n error?: {\n code?: number;\n message?: string;\n status?: string;\n };\n}\n\nasync function googleApiFetch<T>(url: string, init?: RequestInit): Promise<T> {\n const accessToken = await refreshGoogleAccessToken();\n\n const response = await fetch(url, {\n ...init,\n headers: {\n Authorization: `Bearer ${accessToken.accessToken}`,\n \"Content-Type\": \"application/json\",\n ...(init?.headers ?? {}),\n },\n });\n\n if (!response.ok) {\n const payload = (await response.json().catch(() => ({}))) as GoogleApiErrorPayload;\n const message =\n payload.error?.message ??\n payload.error?.status ??\n `Google API request failed with status ${response.status}`;\n\n throw new Error(message);\n }\n\n return (await response.json()) as T;\n}\n\nexport async function listAccountSummaries(): Promise<ListAccountSummariesResult> {\n const payload = await googleApiFetch<Ga4AccountSummariesResponse>(\n `${GA4_ADMIN_BASE_URL}/accountSummaries`\n );\n\n return {\n accountSummaries: payload.accountSummaries ?? [],\n nextPageToken: payload.nextPageToken,\n };\n}\n\nexport async function getMetadata(propertyId: string): Promise<Ga4MetadataResponse> {\n return googleApiFetch<Ga4MetadataResponse>(\n `${GA4_DATA_BASE_URL}/properties/${propertyId}/metadata`\n );\n}\n\nexport async function runReport(\n propertyId: string,\n body: Record<string, unknown>\n): Promise<Ga4ReportResponse> {\n return googleApiFetch<Ga4ReportResponse>(\n `${GA4_DATA_BASE_URL}/properties/${propertyId}:runReport`,\n {\n method: \"POST\",\n body: JSON.stringify(body),\n }\n );\n}\n\nexport async function runRealtimeReport(\n propertyId: string,\n body: Record<string, unknown>\n): Promise<Ga4RealtimeReportResponse> {\n return googleApiFetch<Ga4RealtimeReportResponse>(\n `${GA4_DATA_BASE_URL}/properties/${propertyId}:runRealtimeReport`,\n {\n method: \"POST\",\n body: JSON.stringify(body),\n }\n );\n}\n\nexport async function checkCompatibility(\n propertyId: string,\n body: Record<string, unknown>\n): Promise<Ga4CompatibilityResponse> {\n return googleApiFetch<Ga4CompatibilityResponse>(\n `${GA4_DATA_BASE_URL}/properties/${propertyId}:checkCompatibility`,\n {\n method: \"POST\",\n body: JSON.stringify(body),\n }\n );\n}\n"],
|
|
5
5
|
"mappings": "AAAA,SAAS,gCAAgC;AAEzC,MAAM,oBAAoB;AAC1B,MAAM,qBAAqB;AAwH3B,eAAe,eAAkB,KAAa,MAAgC;AAC5E,QAAM,cAAc,MAAM,yBAAyB;AAEnD,QAAM,WAAW,MAAM,MAAM,KAAK;AAAA,IAChC,GAAG;AAAA,IACH,SAAS;AAAA,MACP,eAAe,UAAU,YAAY,WAAW;AAAA,MAChD,gBAAgB;AAAA,MAChB,GAAI,MAAM,WAAW,CAAC;AAAA,IACxB;AAAA,EACF,CAAC;AAED,MAAI,CAAC,SAAS,IAAI;AAChB,UAAM,UAAW,MAAM,SAAS,KAAK,EAAE,MAAM,OAAO,CAAC,EAAE;AACvD,UAAM,UACJ,QAAQ,OAAO,WACf,QAAQ,OAAO,UACf,yCAAyC,SAAS,MAAM;AAE1D,UAAM,IAAI,MAAM,OAAO;AAAA,EACzB;AAEA,SAAQ,MAAM,SAAS,KAAK;AAC9B;AAEA,eAAsB,uBAA4D;AAChF,QAAM,UAAU,MAAM;AAAA,IACpB,GAAG,kBAAkB;AAAA,EACvB;AAEA,SAAO;AAAA,IACL,kBAAkB,QAAQ,oBAAoB,CAAC;AAAA,IAC/C,eAAe,QAAQ;AAAA,EACzB;AACF;AAEA,eAAsB,YAAY,YAAkD;AAClF,SAAO;AAAA,IACL,GAAG,iBAAiB,eAAe,UAAU;AAAA,EAC/C;AACF;AAEA,eAAsB,UACpB,YACA,MAC4B;AAC5B,SAAO;AAAA,IACL,GAAG,iBAAiB,eAAe,UAAU;AAAA,IAC7C;AAAA,MACE,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,IAAI;AAAA,IAC3B;AAAA,EACF;AACF;AAEA,eAAsB,kBACpB,YACA,MACoC;AACpC,SAAO;AAAA,IACL,GAAG,iBAAiB,eAAe,UAAU;AAAA,IAC7C;AAAA,MACE,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,IAAI;AAAA,IAC3B;AAAA,EACF;AACF;AAEA,eAAsB,mBACpB,YACA,MACmC;AACnC,SAAO;AAAA,IACL,GAAG,iBAAiB,eAAe,UAAU;AAAA,IAC7C;AAAA,MACE,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,IAAI;AAAA,IAC3B;AAAA,EACF;AACF;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/services/analytics/oauth.ts"],
|
|
4
|
-
"sourcesContent": ["import { googleOAuthConfig, type GoogleOAuthClientConfig } from '../../config/google.js';\
|
|
4
|
+
"sourcesContent": ["import { googleOAuthConfig, type GoogleOAuthClientConfig } from '../../config/google.js';\nimport { getActiveGoogleRefreshTokenFromStore } from '../../config/google-store.js';\n\nexport interface GoogleAccessTokenResponse {\n accessToken: string;\n expiresIn: number;\n tokenType: string;\n scope?: string;\n}\n\nexport interface GoogleOAuthRefreshCredentials extends GoogleOAuthClientConfig {\n refreshToken: string;\n}\n\ninterface GoogleTokenApiResponse {\n access_token?: string;\n expires_in?: number;\n token_type?: string;\n scope?: string;\n error?: string;\n error_description?: string;\n}\n\nasync function resolveGoogleOAuthCredentials(): Promise<GoogleOAuthRefreshCredentials> {\n const refreshToken = await getActiveGoogleRefreshTokenFromStore();\n if (!refreshToken) {\n throw new Error(\n 'Missing Google OAuth refresh token. Connect Google in the dashboard and ensure the stored connection is active.'\n );\n }\n\n return {\n clientId: googleOAuthConfig.clientId,\n clientSecret: googleOAuthConfig.clientSecret,\n refreshToken,\n };\n}\n\nexport async function refreshGoogleAccessToken(\n credentials?: GoogleOAuthRefreshCredentials\n): Promise<GoogleAccessTokenResponse> {\n const resolved = credentials ?? (await resolveGoogleOAuthCredentials());\n const response = await fetch('https://oauth2.googleapis.com/token', {\n method: 'POST',\n headers: { 'Content-Type': 'application/x-www-form-urlencoded' },\n body: new URLSearchParams({\n client_id: resolved.clientId,\n client_secret: resolved.clientSecret,\n refresh_token: resolved.refreshToken,\n grant_type: 'refresh_token',\n }),\n });\n\n const payload = (await response.json()) as GoogleTokenApiResponse;\n\n if (!response.ok || !payload.access_token || !payload.expires_in || !payload.token_type) {\n const errorMessage = payload.error_description ?? payload.error ?? `Unexpected Google OAuth response (${response.status})`;\n throw new Error(`Failed to refresh Google access token: ${errorMessage}`);\n }\n\n return {\n accessToken: payload.access_token,\n expiresIn: payload.expires_in,\n tokenType: payload.token_type,\n scope: payload.scope,\n };\n}\n"],
|
|
5
5
|
"mappings": "AAAA,SAAS,yBAAuD;AAChE,SAAS,4CAA4C;AAsBrD,eAAe,gCAAwE;AACrF,QAAM,eAAe,MAAM,qCAAqC;AAChE,MAAI,CAAC,cAAc;AACjB,UAAM,IAAI;AAAA,MACR;AAAA,IACF;AAAA,EACF;AAEA,SAAO;AAAA,IACL,UAAU,kBAAkB;AAAA,IAC5B,cAAc,kBAAkB;AAAA,IAChC;AAAA,EACF;AACF;AAEA,eAAsB,yBACpB,aACoC;AACpC,QAAM,WAAW,eAAgB,MAAM,8BAA8B;AACrE,QAAM,WAAW,MAAM,MAAM,uCAAuC;AAAA,IAClE,QAAQ;AAAA,IACR,SAAS,EAAE,gBAAgB,oCAAoC;AAAA,IAC/D,MAAM,IAAI,gBAAgB;AAAA,MACxB,WAAW,SAAS;AAAA,MACpB,eAAe,SAAS;AAAA,MACxB,eAAe,SAAS;AAAA,MACxB,YAAY;AAAA,IACd,CAAC;AAAA,EACH,CAAC;AAED,QAAM,UAAW,MAAM,SAAS,KAAK;AAErC,MAAI,CAAC,SAAS,MAAM,CAAC,QAAQ,gBAAgB,CAAC,QAAQ,cAAc,CAAC,QAAQ,YAAY;AACvF,UAAM,eAAe,QAAQ,qBAAqB,QAAQ,SAAS,qCAAqC,SAAS,MAAM;AACvH,UAAM,IAAI,MAAM,0CAA0C,YAAY,EAAE;AAAA,EAC1E;AAEA,SAAO;AAAA,IACL,aAAa,QAAQ;AAAA,IACrB,WAAW,QAAQ;AAAA,IACnB,WAAW,QAAQ;AAAA,IACnB,OAAO,QAAQ;AAAA,EACjB;AACF;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/services/google-ads/google-ads-client.ts"],
|
|
4
|
-
"sourcesContent": ["import { refreshGoogleAccessToken } from \"../analytics/oauth.js\";\
|
|
4
|
+
"sourcesContent": ["import { refreshGoogleAccessToken } from \"../analytics/oauth.js\";\n\nexport interface GoogleAdsCustomer {\n id?: string;\n descriptiveName?: string;\n currencyCode?: string;\n timeZone?: string;\n manager?: boolean;\n testAccount?: boolean;\n}\n\nexport interface GoogleAdsCustomerClient {\n clientCustomer?: string;\n descriptiveName?: string;\n currencyCode?: string;\n timeZone?: string;\n manager?: boolean;\n testAccount?: boolean;\n status?: string;\n}\n\nexport interface GoogleAdsCampaign {\n id?: string;\n name?: string;\n status?: string;\n advertisingChannelType?: string;\n}\n\nexport interface GoogleAdsAdGroup {\n id?: string;\n name?: string;\n}\n\nexport interface GoogleAdsSearchTermView {\n searchTerm?: string;\n}\n\nexport interface GoogleAdsSegments {\n date?: string;\n week?: string;\n}\n\nexport interface GoogleAdsMetrics {\n impressions?: string;\n clicks?: string;\n costMicros?: string;\n conversions?: string;\n conversionsValue?: string;\n ctr?: number | string;\n averageCpc?: string;\n}\n\nexport interface GoogleAdsSearchRow {\n customer?: GoogleAdsCustomer;\n customerClient?: GoogleAdsCustomerClient;\n campaign?: GoogleAdsCampaign;\n adGroup?: GoogleAdsAdGroup;\n searchTermView?: GoogleAdsSearchTermView;\n segments?: GoogleAdsSegments;\n metrics?: GoogleAdsMetrics;\n}\n\nexport type GoogleAdsMetricValue = string | number | undefined;\n\ninterface GoogleAdsListAccessibleCustomersResponse {\n resourceNames?: string[];\n}\n\ninterface GoogleAdsSearchStreamBatch {\n results?: GoogleAdsSearchRow[];\n summaryRow?: GoogleAdsSearchRow;\n fieldMask?: string;\n requestId?: string;\n}\n\ninterface GoogleAdsErrorPayload {\n error?: {\n code?: number;\n message?: string;\n status?: string;\n details?: Array<{\n errors?: Array<{\n errorCode?: Record<string, string>;\n message?: string;\n }>;\n }>;\n };\n}\n\nexport interface GoogleAdsSearchResult {\n rows: GoogleAdsSearchRow[];\n summaryRow?: GoogleAdsSearchRow;\n requestId?: string;\n fieldMask?: string;\n}\n\ninterface GoogleAdsRequestOptions {\n customerId?: string;\n loginCustomerId?: string;\n body?: Record<string, unknown>;\n method?: \"GET\" | \"POST\";\n}\n\nconst GOOGLE_ADS_BASE_URL = \"https://googleads.googleapis.com/v22\";\n\nasync function googleAdsFetch<T>(\n path: string,\n developerToken: string,\n options: GoogleAdsRequestOptions = {}\n): Promise<T> {\n const accessToken = await refreshGoogleAccessToken();\n const response = await fetch(`${GOOGLE_ADS_BASE_URL}${path}`, {\n method: options.method ?? \"POST\",\n headers: {\n Authorization: `Bearer ${accessToken.accessToken}`,\n \"Content-Type\": \"application/json\",\n \"developer-token\": developerToken,\n ...(options.loginCustomerId\n ? { \"login-customer-id\": options.loginCustomerId }\n : {}),\n },\n body: options.body ? JSON.stringify(options.body) : undefined,\n });\n\n if (!response.ok) {\n const payload = (await response.json().catch(() => ({}))) as GoogleAdsErrorPayload;\n const detailedMessage = payload.error?.details\n ?.flatMap((detail) => detail.errors ?? [])\n .map((error) => error.message)\n .filter(Boolean)\n .join(\"; \");\n const message =\n detailedMessage ||\n payload.error?.message ||\n payload.error?.status ||\n `Google Ads API request failed with status ${response.status}`;\n\n throw new Error(message);\n }\n\n return (await response.json()) as T;\n}\n\nexport async function listAccessibleCustomers(\n developerToken: string\n): Promise<string[]> {\n const payload = await googleAdsFetch<GoogleAdsListAccessibleCustomersResponse>(\n \"/customers:listAccessibleCustomers\",\n developerToken,\n { method: \"GET\" }\n );\n\n return payload.resourceNames?.map((value) => value.replace(\"customers/\", \"\")) ?? [];\n}\n\nexport async function searchGoogleAds(\n customerId: string,\n query: string,\n developerToken: string,\n loginCustomerId?: string\n): Promise<GoogleAdsSearchResult> {\n const batches = await googleAdsFetch<GoogleAdsSearchStreamBatch[]>(\n `/customers/${customerId}/googleAds:searchStream`,\n developerToken,\n {\n customerId,\n loginCustomerId,\n body: { query },\n }\n );\n\n const rows = batches.flatMap((batch) => batch.results ?? []);\n const lastBatch = batches.at(-1);\n\n return {\n rows,\n summaryRow: lastBatch?.summaryRow,\n requestId: lastBatch?.requestId,\n fieldMask: lastBatch?.fieldMask,\n };\n}\n"],
|
|
5
5
|
"mappings": "AAAA,SAAS,gCAAgC;AAuGzC,MAAM,sBAAsB;AAE5B,eAAe,eACb,MACA,gBACA,UAAmC,CAAC,GACxB;AACZ,QAAM,cAAc,MAAM,yBAAyB;AACnD,QAAM,WAAW,MAAM,MAAM,GAAG,mBAAmB,GAAG,IAAI,IAAI;AAAA,IAC5D,QAAQ,QAAQ,UAAU;AAAA,IAC1B,SAAS;AAAA,MACP,eAAe,UAAU,YAAY,WAAW;AAAA,MAChD,gBAAgB;AAAA,MAChB,mBAAmB;AAAA,MACnB,GAAI,QAAQ,kBACR,EAAE,qBAAqB,QAAQ,gBAAgB,IAC/C,CAAC;AAAA,IACP;AAAA,IACA,MAAM,QAAQ,OAAO,KAAK,UAAU,QAAQ,IAAI,IAAI;AAAA,EACtD,CAAC;AAED,MAAI,CAAC,SAAS,IAAI;AAChB,UAAM,UAAW,MAAM,SAAS,KAAK,EAAE,MAAM,OAAO,CAAC,EAAE;AACvD,UAAM,kBAAkB,QAAQ,OAAO,SACnC,QAAQ,CAAC,WAAW,OAAO,UAAU,CAAC,CAAC,EACxC,IAAI,CAAC,UAAU,MAAM,OAAO,EAC5B,OAAO,OAAO,EACd,KAAK,IAAI;AACZ,UAAM,UACJ,mBACA,QAAQ,OAAO,WACf,QAAQ,OAAO,UACf,6CAA6C,SAAS,MAAM;AAE9D,UAAM,IAAI,MAAM,OAAO;AAAA,EACzB;AAEA,SAAQ,MAAM,SAAS,KAAK;AAC9B;AAEA,eAAsB,wBACpB,gBACmB;AACnB,QAAM,UAAU,MAAM;AAAA,IACpB;AAAA,IACA;AAAA,IACA,EAAE,QAAQ,MAAM;AAAA,EAClB;AAEA,SAAO,QAAQ,eAAe,IAAI,CAAC,UAAU,MAAM,QAAQ,cAAc,EAAE,CAAC,KAAK,CAAC;AACpF;AAEA,eAAsB,gBACpB,YACA,OACA,gBACA,iBACgC;AAChC,QAAM,UAAU,MAAM;AAAA,IACpB,cAAc,UAAU;AAAA,IACxB;AAAA,IACA;AAAA,MACE;AAAA,MACA;AAAA,MACA,MAAM,EAAE,MAAM;AAAA,IAChB;AAAA,EACF;AAEA,QAAM,OAAO,QAAQ,QAAQ,CAAC,UAAU,MAAM,WAAW,CAAC,CAAC;AAC3D,QAAM,YAAY,QAAQ,GAAG,EAAE;AAE/B,SAAO;AAAA,IACL;AAAA,IACA,YAAY,WAAW;AAAA,IACvB,WAAW,WAAW;AAAA,IACtB,WAAW,WAAW;AAAA,EACxB;AACF;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/services/mercadolibre/mercadolibre-api.ts"],
|
|
4
|
-
"sourcesContent": ["import axios from \"axios\";\r\nimport type { AxiosError, AxiosInstance, AxiosResponse } from \"axios\";\r\n\r\nimport { getMercadoLibreAccessForProfile } from \"../../config/mercadolibre.js\";\r\n\r\nconst BASE_URL = \"https://api.mercadolibre.com\";\r\nconst RETRYABLE_STATUS_CODES = new Set([408, 429, 500, 502, 503, 504]);\r\n\r\nexport class MercadoLibreApiError extends Error {\r\n constructor(\r\n message: string,\r\n public readonly status?: number,\r\n public readonly code?: string,\r\n public readonly retryAfterMs?: number\r\n ) {\r\n super(message);\r\n this.name = \"MercadoLibreApiError\";\r\n }\r\n}\r\n\r\nexport interface MercadoLibreBatchFetchOptions {\r\n maxConcurrency?: number;\r\n maxRetries?: number;\r\n baseRetryDelayMs?: number;\r\n}\r\n\r\nexport interface MercadoLibreDocumentSuccess<T> {\r\n id: string;\r\n document: T;\r\n}\r\n\r\nexport interface MercadoLibreDocumentFailure {\r\n id: string;\r\n message: string;\r\n statusCode?: number;\r\n attempts: number;\r\n retryable: boolean;\r\n}\r\n\r\nexport interface MercadoLibreDocumentsBatchResult<T> {\r\n successful: Array<MercadoLibreDocumentSuccess<T>>;\r\n failed: MercadoLibreDocumentFailure[];\r\n}\r\n\r\nexport function sanitizeMercadoLibreText(value: unknown): string {\r\n const raw = typeof value === \"string\" ? value : String(value ?? \"MercadoLibre request failed\");\r\n return raw\r\n .replace(/access_token[=:][^\\s&]+/gi, \"access_token=[redacted]\")\r\n .replace(/refresh_token[=:][^\\s&]+/gi, \"refresh_token=[redacted]\")\r\n .replace(/client_secret[=:][^\\s&]+/gi, \"client_secret=[redacted]\")\r\n .replace(/code[=:][^\\s&]+/gi, \"code=[redacted]\")\r\n .replace(/Bearer\\s+[A-Za-z0-9._-]+/gi, \"Bearer [redacted]\")\r\n .slice(0, 220);\r\n}\r\n\r\nfunction extractRetryAfterMs(error: AxiosError): number | undefined {\r\n const rawHeader = error.response?.headers?.[\"retry-after\"];\r\n const candidate = Array.isArray(rawHeader) ? rawHeader[0] : rawHeader;\r\n if (!candidate) {\r\n return undefined;\r\n }\r\n\r\n const seconds = Number(candidate);\r\n if (Number.isFinite(seconds) && seconds > 0) {\r\n return seconds * 1000;\r\n }\r\n\r\n const dateMs = new Date(candidate).getTime();\r\n if (Number.isFinite(dateMs)) {\r\n return Math.max(0, dateMs - Date.now());\r\n }\r\n\r\n return undefined;\r\n}\r\n\r\nfunction buildApiError(error: AxiosError): MercadoLibreApiError {\r\n const status = error.response?.status;\r\n const responseData = error.response?.data;\r\n let code: string | undefined;\r\n let message: string | undefined;\r\n\r\n if (responseData && typeof responseData === \"object\") {\r\n const candidate = responseData as {\r\n error?: unknown;\r\n message?: unknown;\r\n error_description?: unknown;\r\n cause?: Array<{ code?: unknown; message?: unknown }>;\r\n };\r\n code = typeof candidate.error === \"string\" ? candidate.error : undefined;\r\n message =\r\n (typeof candidate.message === \"string\" ? candidate.message : undefined) ??\r\n (typeof candidate.error_description === \"string\" ? candidate.error_description : undefined) ??\r\n (Array.isArray(candidate.cause)\r\n ? candidate.cause\r\n .map((entry) => (typeof entry.message === \"string\" ? entry.message : null))\r\n .filter(Boolean)\r\n .join(\"; \")\r\n : undefined);\r\n } else if (typeof responseData === \"string\") {\r\n message = responseData;\r\n }\r\n\r\n const safeMessage = sanitizeMercadoLibreText(message ?? error.message ?? \"MercadoLibre request failed\");\r\n return new MercadoLibreApiError(\r\n `MercadoLibre API request failed${status ? ` (${status})` : \"\"}: ${safeMessage}`,\r\n status,\r\n code,\r\n extractRetryAfterMs(error)\r\n );\r\n}\r\n\r\nexport function formatMercadoLibreError(\r\n error: unknown,\r\n fallback = \"Unexpected MercadoLibre API error\"\r\n): string {\r\n if (error instanceof MercadoLibreApiError) {\r\n return error.message;\r\n }\r\n\r\n if (axios.isAxiosError(error)) {\r\n return buildApiError(error).message;\r\n }\r\n\r\n if (error instanceof Error) {\r\n return sanitizeMercadoLibreText(error.message);\r\n }\r\n\r\n return sanitizeMercadoLibreText(fallback);\r\n}\r\n\r\nexport function createMercadoLibreClient(accessToken: string): AxiosInstance {\r\n const client = axios.create({\r\n baseURL: BASE_URL,\r\n timeout: 30000,\r\n headers: {\r\n Accept: \"application/json\",\r\n \"Content-Type\": \"application/json\",\r\n Authorization: `Bearer ${accessToken}`,\r\n },\r\n });\r\n\r\n client.interceptors.response.use(\r\n (response) => response,\r\n (error: AxiosError) => {\r\n const method = error.config?.method?.toUpperCase() ?? \"UNKNOWN\";\r\n const url = error.config?.url ?? \"UNKNOWN_URL\";\r\n const status = error.response?.status ?? \"NO_STATUS\";\r\n console.error(`[MERCADOLIBRE API ERROR] ${method} ${url} -> ${status}`);\r\n return Promise.reject(buildApiError(error));\r\n }\r\n );\r\n\r\n return client;\r\n}\r\n\r\nexport async function getMercadoLibreApiContext(profileId: string) {\r\n const access = await getMercadoLibreAccessForProfile(profileId);\r\n\r\n return {\r\n ...access,\r\n api: createMercadoLibreClient(access.accessToken),\r\n };\r\n}\r\n\r\nexport async function getMercadoLibreUsersMe(profileId: string): Promise<Record<string, unknown>> {\r\n const { api } = await getMercadoLibreApiContext(profileId);\r\n const response = await api.get<Record<string, unknown>>(\"/users/me\");\r\n return response.data;\r\n}\r\n\r\nexport function isRetryableMercadoLibreError(error: unknown): boolean {\r\n if (error instanceof MercadoLibreApiError) {\r\n return error.status ? RETRYABLE_STATUS_CODES.has(error.status) : true;\r\n }\r\n\r\n return false;\r\n}\r\n\r\nfunction classifyMercadoLibreBatchFailure(\r\n id: string,\r\n error: unknown,\r\n attempts: number\r\n): MercadoLibreDocumentFailure {\r\n const statusCode = error instanceof MercadoLibreApiError ? error.status : undefined;\r\n return {\r\n id,\r\n message: formatMercadoLibreError(error, `Failed to fetch MercadoLibre resource ${id}`),\r\n statusCode,\r\n attempts,\r\n retryable: isRetryableMercadoLibreError(error),\r\n };\r\n}\r\n\r\nfunction sleep(ms: number): Promise<void> {\r\n return new Promise((resolve) => setTimeout(resolve, ms));\r\n}\r\n\r\nasync function withMercadoLibreRetry<T>(\r\n id: string,\r\n fetcher: (id: string) => Promise<T>,\r\n options: MercadoLibreBatchFetchOptions = {}\r\n): Promise<MercadoLibreDocumentSuccess<T>> {\r\n const maxRetries = Math.max(0, Math.floor(options.maxRetries ?? 2));\r\n const baseRetryDelayMs = Math.max(50, Math.floor(options.baseRetryDelayMs ?? 250));\r\n let attempts = 0;\r\n\r\n while (true) {\r\n attempts += 1;\r\n\r\n try {\r\n const document = await fetcher(id);\r\n return { id, document };\r\n } catch (error) {\r\n const retryable = isRetryableMercadoLibreError(error);\r\n if (!retryable || attempts > maxRetries + 1) {\r\n throw classifyMercadoLibreBatchFailure(id, error, attempts);\r\n }\r\n\r\n const retryAfterMs = error instanceof MercadoLibreApiError ? error.retryAfterMs : undefined;\r\n const jitterMs = Math.floor(Math.random() * 100);\r\n const delayMs = retryAfterMs ?? baseRetryDelayMs * 2 ** (attempts - 1) + jitterMs;\r\n await sleep(delayMs);\r\n }\r\n }\r\n}\r\n\r\nexport async function runMercadoLibreBatch<T>(\r\n ids: string[],\r\n fetcher: (id: string) => Promise<T>,\r\n options: MercadoLibreBatchFetchOptions = {}\r\n): Promise<MercadoLibreDocumentsBatchResult<T>> {\r\n const maxConcurrency = Math.max(1, Math.floor(options.maxConcurrency ?? 10));\r\n const successful: Array<MercadoLibreDocumentSuccess<T>> = [];\r\n const failed: MercadoLibreDocumentFailure[] = [];\r\n\r\n for (let start = 0; start < ids.length; start += maxConcurrency) {\r\n const chunk = ids.slice(start, start + maxConcurrency);\r\n const settled = await Promise.allSettled(\r\n chunk.map((id) => withMercadoLibreRetry(id, fetcher, options))\r\n );\r\n\r\n settled.forEach((result, index) => {\r\n const id = chunk[index];\r\n if (result.status === \"fulfilled\") {\r\n successful.push(result.value);\r\n return;\r\n }\r\n\r\n const reason = result.reason;\r\n if (\r\n reason &&\r\n typeof reason === \"object\" &&\r\n \"id\" in reason &&\r\n \"attempts\" in reason &&\r\n \"message\" in reason\r\n ) {\r\n failed.push(reason as MercadoLibreDocumentFailure);\r\n return;\r\n }\r\n\r\n failed.push(classifyMercadoLibreBatchFailure(id, reason, 1));\r\n });\r\n }\r\n\r\n return { successful, failed };\r\n}\r\n\r\nexport function normalizeMercadoLibrePaging(input: Record<string, unknown> | null | undefined) {\r\n const paging = input ?? {};\r\n const total = Number((paging.total as number | undefined) ?? 0);\r\n const limit = Number((paging.limit as number | undefined) ?? 0);\r\n const offset = Number((paging.offset as number | undefined) ?? 0);\r\n\r\n return {\r\n total: Number.isFinite(total) ? Math.max(0, total) : 0,\r\n limit: Number.isFinite(limit) ? Math.max(0, limit) : 0,\r\n offset: Number.isFinite(offset) ? Math.max(0, offset) : 0,\r\n };\r\n}\r\n\r\nexport function buildMercadoLibrePaginationMetadata(params: {\r\n total: number;\r\n limit: number;\r\n offset: number;\r\n returned: number;\r\n nextField: string;\r\n}) {\r\n const effectiveLimit = params.limit > 0 ? params.limit : params.returned;\r\n const hasMore = params.offset + params.returned < params.total;\r\n\r\n return {\r\n total: params.total,\r\n limit: effectiveLimit,\r\n offset: params.offset,\r\n returned: params.returned,\r\n has_more: hasMore,\r\n continuation:\r\n hasMore && effectiveLimit > 0\r\n ? `Reenviar la misma tool con ${params.nextField}=${params.offset + effectiveLimit} para continuar.`\r\n : \"No more pages for this query.\",\r\n };\r\n}\r\n\r\nexport async function parseMercadoLibreMultiget<T>(\r\n response: AxiosResponse<Array<{ code?: number; body?: T }>>\r\n): Promise<T[]> {\r\n return response.data\r\n .filter((entry) => Number(entry.code ?? 0) >= 200 && Number(entry.code ?? 0) < 300 && entry.body)\r\n .map((entry) => entry.body as T);\r\n}\r\n"],
|
|
4
|
+
"sourcesContent": ["import axios from \"axios\";\nimport type { AxiosError, AxiosInstance, AxiosResponse } from \"axios\";\n\nimport { getMercadoLibreAccessForProfile } from \"../../config/mercadolibre.js\";\n\nconst BASE_URL = \"https://api.mercadolibre.com\";\nconst RETRYABLE_STATUS_CODES = new Set([408, 429, 500, 502, 503, 504]);\n\nexport class MercadoLibreApiError extends Error {\n constructor(\n message: string,\n public readonly status?: number,\n public readonly code?: string,\n public readonly retryAfterMs?: number\n ) {\n super(message);\n this.name = \"MercadoLibreApiError\";\n }\n}\n\nexport interface MercadoLibreBatchFetchOptions {\n maxConcurrency?: number;\n maxRetries?: number;\n baseRetryDelayMs?: number;\n}\n\nexport interface MercadoLibreDocumentSuccess<T> {\n id: string;\n document: T;\n}\n\nexport interface MercadoLibreDocumentFailure {\n id: string;\n message: string;\n statusCode?: number;\n attempts: number;\n retryable: boolean;\n}\n\nexport interface MercadoLibreDocumentsBatchResult<T> {\n successful: Array<MercadoLibreDocumentSuccess<T>>;\n failed: MercadoLibreDocumentFailure[];\n}\n\nexport function sanitizeMercadoLibreText(value: unknown): string {\n const raw = typeof value === \"string\" ? value : String(value ?? \"MercadoLibre request failed\");\n return raw\n .replace(/access_token[=:][^\\s&]+/gi, \"access_token=[redacted]\")\n .replace(/refresh_token[=:][^\\s&]+/gi, \"refresh_token=[redacted]\")\n .replace(/client_secret[=:][^\\s&]+/gi, \"client_secret=[redacted]\")\n .replace(/code[=:][^\\s&]+/gi, \"code=[redacted]\")\n .replace(/Bearer\\s+[A-Za-z0-9._-]+/gi, \"Bearer [redacted]\")\n .slice(0, 220);\n}\n\nfunction extractRetryAfterMs(error: AxiosError): number | undefined {\n const rawHeader = error.response?.headers?.[\"retry-after\"];\n const candidate = Array.isArray(rawHeader) ? rawHeader[0] : rawHeader;\n if (!candidate) {\n return undefined;\n }\n\n const seconds = Number(candidate);\n if (Number.isFinite(seconds) && seconds > 0) {\n return seconds * 1000;\n }\n\n const dateMs = new Date(candidate).getTime();\n if (Number.isFinite(dateMs)) {\n return Math.max(0, dateMs - Date.now());\n }\n\n return undefined;\n}\n\nfunction buildApiError(error: AxiosError): MercadoLibreApiError {\n const status = error.response?.status;\n const responseData = error.response?.data;\n let code: string | undefined;\n let message: string | undefined;\n\n if (responseData && typeof responseData === \"object\") {\n const candidate = responseData as {\n error?: unknown;\n message?: unknown;\n error_description?: unknown;\n cause?: Array<{ code?: unknown; message?: unknown }>;\n };\n code = typeof candidate.error === \"string\" ? candidate.error : undefined;\n message =\n (typeof candidate.message === \"string\" ? candidate.message : undefined) ??\n (typeof candidate.error_description === \"string\" ? candidate.error_description : undefined) ??\n (Array.isArray(candidate.cause)\n ? candidate.cause\n .map((entry) => (typeof entry.message === \"string\" ? entry.message : null))\n .filter(Boolean)\n .join(\"; \")\n : undefined);\n } else if (typeof responseData === \"string\") {\n message = responseData;\n }\n\n const safeMessage = sanitizeMercadoLibreText(message ?? error.message ?? \"MercadoLibre request failed\");\n return new MercadoLibreApiError(\n `MercadoLibre API request failed${status ? ` (${status})` : \"\"}: ${safeMessage}`,\n status,\n code,\n extractRetryAfterMs(error)\n );\n}\n\nexport function formatMercadoLibreError(\n error: unknown,\n fallback = \"Unexpected MercadoLibre API error\"\n): string {\n if (error instanceof MercadoLibreApiError) {\n return error.message;\n }\n\n if (axios.isAxiosError(error)) {\n return buildApiError(error).message;\n }\n\n if (error instanceof Error) {\n return sanitizeMercadoLibreText(error.message);\n }\n\n return sanitizeMercadoLibreText(fallback);\n}\n\nexport function createMercadoLibreClient(accessToken: string): AxiosInstance {\n const client = axios.create({\n baseURL: BASE_URL,\n timeout: 30000,\n headers: {\n Accept: \"application/json\",\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${accessToken}`,\n },\n });\n\n client.interceptors.response.use(\n (response) => response,\n (error: AxiosError) => {\n const method = error.config?.method?.toUpperCase() ?? \"UNKNOWN\";\n const url = error.config?.url ?? \"UNKNOWN_URL\";\n const status = error.response?.status ?? \"NO_STATUS\";\n console.error(`[MERCADOLIBRE API ERROR] ${method} ${url} -> ${status}`);\n return Promise.reject(buildApiError(error));\n }\n );\n\n return client;\n}\n\nexport async function getMercadoLibreApiContext(profileId: string) {\n const access = await getMercadoLibreAccessForProfile(profileId);\n\n return {\n ...access,\n api: createMercadoLibreClient(access.accessToken),\n };\n}\n\nexport async function getMercadoLibreUsersMe(profileId: string): Promise<Record<string, unknown>> {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(\"/users/me\");\n return response.data;\n}\n\nexport function isRetryableMercadoLibreError(error: unknown): boolean {\n if (error instanceof MercadoLibreApiError) {\n return error.status ? RETRYABLE_STATUS_CODES.has(error.status) : true;\n }\n\n return false;\n}\n\nfunction classifyMercadoLibreBatchFailure(\n id: string,\n error: unknown,\n attempts: number\n): MercadoLibreDocumentFailure {\n const statusCode = error instanceof MercadoLibreApiError ? error.status : undefined;\n return {\n id,\n message: formatMercadoLibreError(error, `Failed to fetch MercadoLibre resource ${id}`),\n statusCode,\n attempts,\n retryable: isRetryableMercadoLibreError(error),\n };\n}\n\nfunction sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\nasync function withMercadoLibreRetry<T>(\n id: string,\n fetcher: (id: string) => Promise<T>,\n options: MercadoLibreBatchFetchOptions = {}\n): Promise<MercadoLibreDocumentSuccess<T>> {\n const maxRetries = Math.max(0, Math.floor(options.maxRetries ?? 2));\n const baseRetryDelayMs = Math.max(50, Math.floor(options.baseRetryDelayMs ?? 250));\n let attempts = 0;\n\n while (true) {\n attempts += 1;\n\n try {\n const document = await fetcher(id);\n return { id, document };\n } catch (error) {\n const retryable = isRetryableMercadoLibreError(error);\n if (!retryable || attempts > maxRetries + 1) {\n throw classifyMercadoLibreBatchFailure(id, error, attempts);\n }\n\n const retryAfterMs = error instanceof MercadoLibreApiError ? error.retryAfterMs : undefined;\n const jitterMs = Math.floor(Math.random() * 100);\n const delayMs = retryAfterMs ?? baseRetryDelayMs * 2 ** (attempts - 1) + jitterMs;\n await sleep(delayMs);\n }\n }\n}\n\nexport async function runMercadoLibreBatch<T>(\n ids: string[],\n fetcher: (id: string) => Promise<T>,\n options: MercadoLibreBatchFetchOptions = {}\n): Promise<MercadoLibreDocumentsBatchResult<T>> {\n const maxConcurrency = Math.max(1, Math.floor(options.maxConcurrency ?? 10));\n const successful: Array<MercadoLibreDocumentSuccess<T>> = [];\n const failed: MercadoLibreDocumentFailure[] = [];\n\n for (let start = 0; start < ids.length; start += maxConcurrency) {\n const chunk = ids.slice(start, start + maxConcurrency);\n const settled = await Promise.allSettled(\n chunk.map((id) => withMercadoLibreRetry(id, fetcher, options))\n );\n\n settled.forEach((result, index) => {\n const id = chunk[index];\n if (result.status === \"fulfilled\") {\n successful.push(result.value);\n return;\n }\n\n const reason = result.reason;\n if (\n reason &&\n typeof reason === \"object\" &&\n \"id\" in reason &&\n \"attempts\" in reason &&\n \"message\" in reason\n ) {\n failed.push(reason as MercadoLibreDocumentFailure);\n return;\n }\n\n failed.push(classifyMercadoLibreBatchFailure(id, reason, 1));\n });\n }\n\n return { successful, failed };\n}\n\nexport function normalizeMercadoLibrePaging(input: Record<string, unknown> | null | undefined) {\n const paging = input ?? {};\n const total = Number((paging.total as number | undefined) ?? 0);\n const limit = Number((paging.limit as number | undefined) ?? 0);\n const offset = Number((paging.offset as number | undefined) ?? 0);\n\n return {\n total: Number.isFinite(total) ? Math.max(0, total) : 0,\n limit: Number.isFinite(limit) ? Math.max(0, limit) : 0,\n offset: Number.isFinite(offset) ? Math.max(0, offset) : 0,\n };\n}\n\nexport function buildMercadoLibrePaginationMetadata(params: {\n total: number;\n limit: number;\n offset: number;\n returned: number;\n nextField: string;\n}) {\n const effectiveLimit = params.limit > 0 ? params.limit : params.returned;\n const hasMore = params.offset + params.returned < params.total;\n\n return {\n total: params.total,\n limit: effectiveLimit,\n offset: params.offset,\n returned: params.returned,\n has_more: hasMore,\n continuation:\n hasMore && effectiveLimit > 0\n ? `Reenviar la misma tool con ${params.nextField}=${params.offset + effectiveLimit} para continuar.`\n : \"No more pages for this query.\",\n };\n}\n\nexport async function parseMercadoLibreMultiget<T>(\n response: AxiosResponse<Array<{ code?: number; body?: T }>>\n): Promise<T[]> {\n return response.data\n .filter((entry) => Number(entry.code ?? 0) >= 200 && Number(entry.code ?? 0) < 300 && entry.body)\n .map((entry) => entry.body as T);\n}\n"],
|
|
5
5
|
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/services/mercadolibre/mercadolibre-items.ts"],
|
|
4
|
-
"sourcesContent": ["import type { AxiosInstance } from \"axios\";\
|
|
4
|
+
"sourcesContent": ["import type { AxiosInstance } from \"axios\";\n\nimport {\n getMercadoLibreApiContext,\n parseMercadoLibreMultiget,\n runMercadoLibreBatch,\n type MercadoLibreBatchFetchOptions,\n} from \"./mercadolibre-api.js\";\n\nexport async function searchMercadoLibreItems(\n profileId: string,\n params: {\n sellerId: string;\n status?: string;\n category?: string;\n listingTypeId?: string;\n orders?: string;\n limit?: number;\n offset?: number;\n }\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(\n `/users/${params.sellerId}/items/search`,\n {\n params: {\n status: params.status,\n category: params.category,\n listing_type_id: params.listingTypeId,\n orders: params.orders,\n limit: params.limit ?? 20,\n offset: params.offset ?? 0,\n },\n }\n );\n return response.data;\n}\n\nexport async function getMercadoLibreItem(profileId: string, itemId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/items/${itemId}`);\n return response.data;\n}\n\nexport async function getMercadoLibreItemDetailsBatch(\n profileId: string,\n itemIds: string[],\n attributes?: string[]\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Array<{ code?: number; body?: Record<string, unknown> }>>(\"/items\", {\n params: {\n ids: itemIds.join(\",\"),\n attributes: attributes?.join(\",\"),\n },\n });\n return parseMercadoLibreMultiget(response);\n}\n\nexport async function getMercadoLibreItemDetailsWithFallbackBatch(\n profileId: string,\n itemIds: string[],\n options: MercadoLibreBatchFetchOptions = {}\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n return runMercadoLibreBatch(itemIds, (itemId) => getItem(api, itemId), options);\n}\n\nasync function getItem(api: AxiosInstance, itemId: string): Promise<Record<string, unknown>> {\n const response = await api.get<Record<string, unknown>>(`/items/${itemId}`);\n return response.data;\n}\n\nexport async function createMercadoLibreItem(profileId: string, payload: Record<string, unknown>) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.post<Record<string, unknown>>(\"/items\", payload);\n return response.data;\n}\n\nexport async function updateMercadoLibreItem(\n profileId: string,\n itemId: string,\n payload: Record<string, unknown>\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.put<Record<string, unknown>>(`/items/${itemId}`, payload);\n return response.data;\n}\n\nexport async function getMercadoLibreItemDescription(profileId: string, itemId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/items/${itemId}/description`);\n return response.data;\n}\n\nexport async function createMercadoLibreItemDescription(\n profileId: string,\n itemId: string,\n plainText: string\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.post<Record<string, unknown>>(`/items/${itemId}/description`, {\n plain_text: plainText,\n });\n return response.data;\n}\n\nexport async function replaceMercadoLibreItemDescription(\n profileId: string,\n itemId: string,\n plainText: string\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.put<Record<string, unknown>>(\n `/items/${itemId}/description?api_version=2`,\n {\n plain_text: plainText,\n }\n );\n return response.data;\n}\n\nexport async function addMercadoLibreItemPicture(\n profileId: string,\n itemId: string,\n source: string\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.post<Record<string, unknown>>(`/items/${itemId}/pictures`, {\n source,\n });\n return response.data;\n}\n\nexport async function searchMercadoLibreDomainDiscovery(\n profileId: string,\n query: string,\n limit: number,\n siteId = \"MLA\"\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Array<Record<string, unknown>>>(`/sites/${siteId}/domain_discovery/search`, {\n params: { q: query, limit },\n });\n return response.data;\n}\n\nexport async function getMercadoLibreCategoryAttributes(profileId: string, categoryId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Array<Record<string, unknown>>>(`/categories/${categoryId}/attributes`);\n return response.data;\n}\n\nexport async function getMercadoLibreCategoryShippingPreferences(\n profileId: string,\n categoryId: string\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(\n `/categories/${categoryId}/shipping_preferences`\n );\n return response.data;\n}\n\nexport async function estimateMercadoLibreListingPrice(\n profileId: string,\n params: Record<string, string | number | boolean | undefined>\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Array<Record<string, unknown>>>(\"/sites/MLA/listing_prices\", {\n params,\n });\n return response.data;\n}\n\nexport async function getMercadoLibreUserItemsVisits(\n profileId: string,\n sellerId: string,\n dateFrom: string,\n dateTo: string\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/users/${sellerId}/items_visits`, {\n params: {\n date_from: dateFrom,\n date_to: dateTo,\n },\n });\n return response.data;\n}\n\nexport async function getMercadoLibreItemVisitsWindow(\n profileId: string,\n itemId: string,\n last: number,\n unit = \"day\",\n ending?: string\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/items/${itemId}/visits/time_window`, {\n params: {\n last,\n unit,\n ending,\n },\n });\n return response.data;\n}\n\nexport async function getMercadoLibreItemPerformance(profileId: string, itemId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/item/${itemId}/performance`);\n return response.data;\n}\n"],
|
|
5
5
|
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/services/mercadolibre/mercadolibre-orders.ts"],
|
|
4
|
-
"sourcesContent": ["import type { AxiosInstance } from \"axios\";\
|
|
4
|
+
"sourcesContent": ["import type { AxiosInstance } from \"axios\";\n\nimport {\n getMercadoLibreApiContext,\n runMercadoLibreBatch,\n type MercadoLibreDocumentsBatchResult,\n type MercadoLibreBatchFetchOptions,\n} from \"./mercadolibre-api.js\";\n\nexport interface MercadoLibreOrdersSearchParams {\n seller: string;\n from: string;\n to: string;\n status?: string;\n q?: string;\n sort?: string;\n offset?: number;\n limit?: number;\n}\n\nfunction orderSearchParams(params: MercadoLibreOrdersSearchParams) {\n return {\n seller: params.seller,\n \"order.date_created.from\": `${params.from}T00:00:00.000-00:00`,\n \"order.date_created.to\": `${params.to}T23:59:59.999-00:00`,\n \"order.status\": params.status,\n q: params.q,\n sort: params.sort,\n offset: params.offset ?? 0,\n limit: params.limit ?? 50,\n };\n}\n\nasync function fetchOrder(api: AxiosInstance, orderId: string): Promise<Record<string, unknown>> {\n const response = await api.get<Record<string, unknown>>(`/orders/${orderId}`);\n return response.data;\n}\n\nasync function fetchOrdersPage(\n api: AxiosInstance,\n params: MercadoLibreOrdersSearchParams\n): Promise<Record<string, unknown>> {\n const response = await api.get<Record<string, unknown>>(\"/orders/search\", {\n params: orderSearchParams(params),\n });\n return response.data;\n}\n\nexport async function searchMercadoLibreOrders(\n profileId: string,\n params: MercadoLibreOrdersSearchParams\n) {\n const { api, sellerId } = await getMercadoLibreApiContext(profileId);\n return fetchOrdersPage(api, { ...params, seller: params.seller || sellerId });\n}\n\nexport async function searchMercadoLibreOrdersBatch(\n profileId: string,\n params: MercadoLibreOrdersSearchParams,\n offsets: number[],\n options: MercadoLibreBatchFetchOptions = {}\n): Promise<MercadoLibreDocumentsBatchResult<Record<string, unknown>>> {\n const { api, sellerId } = await getMercadoLibreApiContext(profileId);\n const seller = params.seller || sellerId;\n const offsetIds = offsets.map((offset) => String(offset));\n\n return runMercadoLibreBatch(\n offsetIds,\n (offsetId) => fetchOrdersPage(api, { ...params, seller, offset: Number(offsetId) }),\n options\n );\n}\n\nexport async function getMercadoLibreOrder(profileId: string, orderId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n return fetchOrder(api, orderId);\n}\n\nexport async function getMercadoLibreOrderDetailsBatch(\n profileId: string,\n orderIds: string[],\n options: MercadoLibreBatchFetchOptions = {}\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n return runMercadoLibreBatch(orderIds, (orderId) => fetchOrder(api, orderId), options);\n}\n\nexport async function getMercadoLibreOrderShipments(profileId: string, orderId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/orders/${orderId}/shipments`);\n return response.data;\n}\n\nexport async function getMercadoLibreShipment(profileId: string, shipmentId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/shipments/${shipmentId}`);\n return response.data;\n}\n\nexport async function getMercadoLibreShipmentItems(profileId: string, shipmentId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/shipments/${shipmentId}/items`);\n return response.data;\n}\n\nexport async function getMercadoLibreShipmentCosts(profileId: string, shipmentId: string) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(`/shipments/${shipmentId}/costs`);\n return response.data;\n}\n"],
|
|
5
5
|
"mappings": "AAEA;AAAA,EACE;AAAA,EACA;AAAA,OAGK;AAaP,SAAS,kBAAkB,QAAwC;AACjE,SAAO;AAAA,IACL,QAAQ,OAAO;AAAA,IACf,2BAA2B,GAAG,OAAO,IAAI;AAAA,IACzC,yBAAyB,GAAG,OAAO,EAAE;AAAA,IACrC,gBAAgB,OAAO;AAAA,IACvB,GAAG,OAAO;AAAA,IACV,MAAM,OAAO;AAAA,IACb,QAAQ,OAAO,UAAU;AAAA,IACzB,OAAO,OAAO,SAAS;AAAA,EACzB;AACF;AAEA,eAAe,WAAW,KAAoB,SAAmD;AAC/F,QAAM,WAAW,MAAM,IAAI,IAA6B,WAAW,OAAO,EAAE;AAC5E,SAAO,SAAS;AAClB;AAEA,eAAe,gBACb,KACA,QACkC;AAClC,QAAM,WAAW,MAAM,IAAI,IAA6B,kBAAkB;AAAA,IACxE,QAAQ,kBAAkB,MAAM;AAAA,EAClC,CAAC;AACD,SAAO,SAAS;AAClB;AAEA,eAAsB,yBACpB,WACA,QACA;AACA,QAAM,EAAE,KAAK,SAAS,IAAI,MAAM,0BAA0B,SAAS;AACnE,SAAO,gBAAgB,KAAK,EAAE,GAAG,QAAQ,QAAQ,OAAO,UAAU,SAAS,CAAC;AAC9E;AAEA,eAAsB,8BACpB,WACA,QACA,SACA,UAAyC,CAAC,GAC0B;AACpE,QAAM,EAAE,KAAK,SAAS,IAAI,MAAM,0BAA0B,SAAS;AACnE,QAAM,SAAS,OAAO,UAAU;AAChC,QAAM,YAAY,QAAQ,IAAI,CAAC,WAAW,OAAO,MAAM,CAAC;AAExD,SAAO;AAAA,IACL;AAAA,IACA,CAAC,aAAa,gBAAgB,KAAK,EAAE,GAAG,QAAQ,QAAQ,QAAQ,OAAO,QAAQ,EAAE,CAAC;AAAA,IAClF;AAAA,EACF;AACF;AAEA,eAAsB,qBAAqB,WAAmB,SAAiB;AAC7E,QAAM,EAAE,IAAI,IAAI,MAAM,0BAA0B,SAAS;AACzD,SAAO,WAAW,KAAK,OAAO;AAChC;AAEA,eAAsB,iCACpB,WACA,UACA,UAAyC,CAAC,GAC1C;AACA,QAAM,EAAE,IAAI,IAAI,MAAM,0BAA0B,SAAS;AACzD,SAAO,qBAAqB,UAAU,CAAC,YAAY,WAAW,KAAK,OAAO,GAAG,OAAO;AACtF;AAEA,eAAsB,8BAA8B,WAAmB,SAAiB;AACtF,QAAM,EAAE,IAAI,IAAI,MAAM,0BAA0B,SAAS;AACzD,QAAM,WAAW,MAAM,IAAI,IAA6B,WAAW,OAAO,YAAY;AACtF,SAAO,SAAS;AAClB;AAEA,eAAsB,wBAAwB,WAAmB,YAAoB;AACnF,QAAM,EAAE,IAAI,IAAI,MAAM,0BAA0B,SAAS;AACzD,QAAM,WAAW,MAAM,IAAI,IAA6B,cAAc,UAAU,EAAE;AAClF,SAAO,SAAS;AAClB;AAEA,eAAsB,6BAA6B,WAAmB,YAAoB;AACxF,QAAM,EAAE,IAAI,IAAI,MAAM,0BAA0B,SAAS;AACzD,QAAM,WAAW,MAAM,IAAI,IAA6B,cAAc,UAAU,QAAQ;AACxF,SAAO,SAAS;AAClB;AAEA,eAAsB,6BAA6B,WAAmB,YAAoB;AACxF,QAAM,EAAE,IAAI,IAAI,MAAM,0BAA0B,SAAS;AACzD,QAAM,WAAW,MAAM,IAAI,IAA6B,cAAc,UAAU,QAAQ;AACxF,SAAO,SAAS;AAClB;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/services/mercadolibre/mercadolibre-questions.ts"],
|
|
4
|
-
"sourcesContent": ["import { getMercadoLibreApiContext } from \"./mercadolibre-api.js\";\
|
|
4
|
+
"sourcesContent": ["import { getMercadoLibreApiContext } from \"./mercadolibre-api.js\";\n\nexport async function searchMercadoLibreQuestions(\n profileId: string,\n params: {\n sellerId: string;\n itemId?: string;\n status?: string;\n limit?: number;\n offset?: number;\n }\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.get<Record<string, unknown>>(\"/questions/search\", {\n params: {\n seller_id: params.sellerId,\n api_version: 4,\n item_id: params.itemId,\n status: params.status,\n limit: params.limit ?? 20,\n offset: params.offset ?? 0,\n },\n });\n\n return response.data;\n}\n\nexport async function answerMercadoLibreQuestion(\n profileId: string,\n questionId: number,\n text: string\n) {\n const { api } = await getMercadoLibreApiContext(profileId);\n const response = await api.post<Record<string, unknown>>(\"/answers\", {\n question_id: questionId,\n text,\n });\n\n return response.data;\n}\n"],
|
|
5
5
|
"mappings": "AAAA,SAAS,iCAAiC;AAE1C,eAAsB,4BACpB,WACA,QAOA;AACA,QAAM,EAAE,IAAI,IAAI,MAAM,0BAA0B,SAAS;AACzD,QAAM,WAAW,MAAM,IAAI,IAA6B,qBAAqB;AAAA,IAC3E,QAAQ;AAAA,MACN,WAAW,OAAO;AAAA,MAClB,aAAa;AAAA,MACb,SAAS,OAAO;AAAA,MAChB,QAAQ,OAAO;AAAA,MACf,OAAO,OAAO,SAAS;AAAA,MACvB,QAAQ,OAAO,UAAU;AAAA,IAC3B;AAAA,EACF,CAAC;AAED,SAAO,SAAS;AAClB;AAEA,eAAsB,2BACpB,WACA,YACA,MACA;AACA,QAAM,EAAE,IAAI,IAAI,MAAM,0BAA0B,SAAS;AACzD,QAAM,WAAW,MAAM,IAAI,KAA8B,YAAY;AAAA,IACnE,aAAa;AAAA,IACb;AAAA,EACF,CAAC;AAED,SAAO,SAAS;AAClB;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|