diff --git a/docs/subscriptions/observable-events.mdx b/docs/subscriptions/observable-events.mdx index d2791b51..8c8e1fb0 100644 --- a/docs/subscriptions/observable-events.mdx +++ b/docs/subscriptions/observable-events.mdx @@ -3,21 +3,40 @@ title: Observable Events nav_label: Observable Events sidebar_position: 100 --- -You can integrate Subscriptions with external systems like enterprise resource planning, fulfilment and other systems. For example, when a subscriber updates their address, the Customer Relationship Management system is updated with the change. -Events are actions that occur in Subscriptions, such as a subscriber changing their address or a subscription changing from active to inactive. You can create custom functions that perform additional processing outside of Subscriptions, and create integrations so that when an event occurs in your store, the custom function is run. +# Observable Events in Subscriptions -Events are processed concurrently. This means that events may not be delivered in the order they are created. For example, if a subscription is updated multiple times, the events may not be delivered in the same sequence they were updated. Events operate on an "at least once" delivery policy. We aim to deliver messages within 30 minutes or less. Ensure that you design your receiving code accordingly. +## Overview +Observable events allow you to integrate Subscriptions with external systems like ERP, fulfillment, and CRM platforms. When specific actions occur within Subscriptions (such as address updates or status changes), these events can trigger custom functions to perform additional processing or update external systems. -For more information about integrating Subscriptions, see [**Integration Types**](/docs/api/integrations/integrations-introduction#integration-types). +## Event Processing Behavior +- Events are processed concurrently and may not be delivered in chronological order +- Events follow an "at least once" delivery policy +- Target delivery time is within 30 minutes +- Integration code should be designed to handle out-of-order and duplicate events -| Resource | Action | Observable Key | Availability | -| --- | --- | --- | --- | -| Product | | | Store | -| Plan | | | Store | -| Offering | | | Store | -| Subscription | | | Store | -| Job | | | Store | -| Invoices | | | Store | -| Schedule | | | Store | -| Subscriber | | | Store | +For detailed information about integration options, see [**Integration Types**](/docs/api/integrations/integrations-introduction#integration-types). + +## Available Events + +### Subscription Management +| Resource | Actions | Observable Keys | +|----------|---------|-----------------| +| Subscription | • Created
• Create-failed
• Paused
• Canceled
• Pending-cancel
• Pending-pause
• Resumed
• Closed | `subscription.created`
`subscription.create-failed`
`subscription.paused`
`subscription.canceled`
`subscription.pending_cancel`
`subscription.pending_pause`
`subscription.resumed`
`subscription.closed` | +| Subscriber | • Created
• Updated
• Deleted | `subscription-subscriber.created`
`subscription-subscriber.updated`
`subscription-subscriber.deleted` | + +### Product Configuration +| Resource | Actions | Observable Keys | +|----------|---------|-----------------| +| Product | • Created
• Updated
• Deleted | `subscription-product.created`
`subscription-product.updated`
`subscription-product.deleted` | +| Plan | • Created
• Updated
• Deleted | `subscription-plan.created`
`subscription-plan.updated`
`subscription-plan.deleted` | +| Offering | • Created
• Updated
• Deleted | `subscription-offering.created`
`subscription-offering.updated`
`subscription-offering.deleted` | + +### Operations +| Resource | Actions | Observable Keys | +|----------|---------|-----------------| +| Job | • Created
• Updated
• Deleted | `subscription-job.created`
`subscription-job.updated`
`subscription-job.deleted` | +| Invoices | • Created
• Deleted | `subscription-invoice.created`
`subscription-invoice.deleted` | +| Schedule | • Created
• Updated
• Deleted | `subscription-schedule.created`
`subscription-schedule.updated`
`subscription-schedule.deleted` | + +> **Note**: All events are available at the Store level. diff --git a/docusaurus.config.js b/docusaurus.config.js index 50b0f77e..1e535740 100644 --- a/docusaurus.config.js +++ b/docusaurus.config.js @@ -181,9 +181,6 @@ const config = { googleTagManager: { containerId: 'G-NZ3NL8DLLD', }, - googleTagManager: { - containerId: 'G-NZ3NL8DLLD', - }, }), ], ], @@ -224,11 +221,13 @@ const config = { { label: "Guides", to: "guides", + datautmcampaign: "guides", }, { label: "Docs", type: "dropdown", className: "nav-dropdown", + datautmcampaign: "docs", items: [ { type: "html", @@ -252,10 +251,12 @@ const config = { { label: "Changelog", to: "/changelog-landing", + datautmcampaign: "changelog", }, { label: "Support", to: "https://support.elasticpath.com", + datautmcampaign: "support", }, { type: "search", @@ -266,12 +267,14 @@ const config = { href: "https://www.elasticpath.com/get-in-touch", position: "right", className: "navbar-book-demo", + datautmcampaign: "get-in-touch", }, { label: "Free Trial", href: "https://useast.cm.elasticpath.com/free-trial", position: "right", className: "dev-portal-signup dev-portal-link", + datautmcampaign: "free-trial", }, ], }, @@ -313,7 +316,7 @@ const config = { }, { label: "GitHub", - href: "https://github.com/facebook/docusaurus", + href: "https://github.com/elasticpath", }, ], }, @@ -1329,7 +1332,7 @@ const config = { { to: '/docs/studio/Settings/Domain-Management/Connecting-your-Namecheap-Domain', from: '/docs/cx-studio/Settings/Domain-Management/Connecting-your-Namecheap-Domain'}, { to: '/docs/api/pxm/catalog/get-catalog-by-id', from: '/docs/pxm/catalogs/catalog-configuration/get-all-catalogs'}, { to: '/docs/api/carts/transactions', from: '/docs/commerce-cloud/payments/transactions/get-all-transactions'}, - { to: '/docs/api/pxm/catalog/get-by-context-all-nodes', from: '/docs/pxm/catalogs/shopper-catalog/get-a-hierarchys-nodes'}, + { to: '/docs/api/pxm/catalog/get-by-context-all-hierarchies', from: '/docs/pxm/catalogs/shopper-catalog/get-a-hierarchys-nodes'}, { to: '/changelog/Studio-Release-Notes/Release-190-February-14-2024', from: '/docs/cx-studio/Release-Notes/Release-190-February-14-2024'}, { to: '/changelog/Studio-Release-Notes/Release-190-February-14-2024', from: '/docs/cx-studio/Release-Notes/Release-191-February-27-2024'}, { to: '/docs/api/pxm/products/build-child-products', from: '/docs/pxm/products/pxm-product-variations/child-products-api/build-child-products'}, @@ -1633,10 +1636,7 @@ const config = { "@docusaurus/theme-live-codeblock", [ require.resolve("@easyops-cn/docusaurus-search-local"), - /** @type {import("@easyops-cn/docusaurus-search-local").PluginOptions} */ ({ - // ... Your options. - // `hashed` is recommended as long-term-cache of index file is possible. hashed: true, indexDocs: true, indexBlog: true, @@ -1646,13 +1646,15 @@ const config = { highlightSearchTermsOnTargetPage: false, searchContextByPaths: ["docs","guides","changelog"], useAllContextsWithNoSearchContext: true, - // For Docs using Chinese, The `language` is recommended to set to: - // ``` - // language: ["en", "zh"], - // ``` }), ], ], + scripts: [ + { + src: '/js/utm-handler.js', + async: true, + }, + ], }; module.exports = config; diff --git a/src/snippets/resourceDOCS.html b/src/snippets/resourceDOCS.html index f89b2bdc..b2f69432 100644 --- a/src/snippets/resourceDOCS.html +++ b/src/snippets/resourceDOCS.html @@ -40,6 +40,7 @@ onmouseover="document.body.setAttribute('docs-menu', 'cm')" class="flex cursor-pointer !mt-2 items-start justify-start gap-2 border-none bg-transparent text-black dark:text-white" href="/ui" + datautmcampaign="commerce-manager" >
{ + if (urlParams.has(param)) { + utmParams[param] = urlParams.get(param); + } + }); + + return utmParams; +} + +// Store UTM params in sessionStorage +function storeUtmParams(params) { + sessionStorage.setItem('utmParams', JSON.stringify(params)); +} + +// Get stored UTM params +function getStoredUtmParams() { + const stored = sessionStorage.getItem('utmParams'); + return stored ? JSON.parse(stored) : null; +} + +// Add UTM params to a URL +function addUtmToUrl(url, campaignType) { + try { + const urlObj = new URL(url); + + // Don't modify external URLs + if (!urlObj.hostname.includes('elasticpath.dev')) { + return url; + } + + // Get stored or default UTM params + const utmParams = getStoredUtmParams() || {...DEFAULT_UTM}; + + // Override campaign if specified + if (campaignType && CAMPAIGN_MAPPING[campaignType]) { + utmParams.utm_campaign = CAMPAIGN_MAPPING[campaignType]; + } + + // Add UTM params to URL + Object.entries(utmParams).forEach(([key, value]) => { + urlObj.searchParams.set(key, value); + }); + + return urlObj.toString(); + } catch (e) { + // Return original URL if invalid + return url; + } +} + +// Initialize UTM handling +function initUtmHandler() { + // Store UTM params from URL if present + const urlUtmParams = getUrlUtmParams(); + if (Object.keys(urlUtmParams).length > 0) { + storeUtmParams(urlUtmParams); + } + + // Add click event listener to handle links + document.addEventListener('click', (e) => { + const link = e.target.closest('a'); + if (link && link.href) { + // Get the campaign type from the data attribute + const campaignType = link.getAttribute('datautmcampaign'); + const modifiedUrl = addUtmToUrl(link.href, campaignType); + if (modifiedUrl !== link.href) { + e.preventDefault(); + window.location.href = modifiedUrl; + } + } + }); +} + +// Run when DOM is loaded +if (document.readyState === 'loading') { + document.addEventListener('DOMContentLoaded', initUtmHandler); +} else { + initUtmHandler(); +} \ No newline at end of file