{"templateName":"quickstart-page-template","cssClassNames":"page basicpage summit-page","allowedRenditionsWidth":["320","480","640","768","960","1200","1440","1920"],"language":"en","description":"Migrate SQL Server databases and Spark pipelines to Snowflake using SnowConvert and Snowpark Migration Accelerator tools.","title":"End-to-End Migration to Snowflake in Action: Data and Pipelines","analyticsPageType":"quickstart-page-template","analyticsCategory":"general","analyticsSubCategory":"","excludeFromAnalytics":false,"analyticsDebugMode":false,"analyticsContentTags":["snowflake-site:taxonomy/product/data-engineering","snowflake-site:taxonomy/snowflake-feature/transformation","snowflake-site:taxonomy/solution-center/certification/quickstart"],"analyticsEnabled":true,":type":"snowflake-site/components/structure/page",":mappedPath":"/en/developers/guides/end2endmigration/",":items":{"root":{"columnClassNames":{"markup_editor_1950346551":"aem-GridColumn aem-GridColumn--default--12","experiencefragment-banner":"aem-GridColumn aem-GridColumn--default--12","experiencefragment-header":"aem-GridColumn aem-GridColumn--default--12","responsivegrid":"aem-GridColumn aem-GridColumn--default--12","experiencefragment-footer":"aem-GridColumn aem-GridColumn--default--12","modal_container":"aem-GridColumn aem-GridColumn--default--12","markup_editor":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","columnCount":12,":items":{"experiencefragment-banner":{"id":"experiencefragment-7694b33a22","localizedFragmentVariationPath":"/content/experience-fragments/snowflake-site/language-masters/en/site/pushdown-banner/master/jcr:content","configured":true,":type":"snowflake-site/components/experiencefragment",":items":{"root":{"columnClassNames":{"pushdown_banner_copy":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-e61b10838b",":type":"snowflake-site/components/container",":items":{"pushdown_banner_copy":{"id":"pushdown-banner-9732a04f70","contentHeadline":"Snowflake World Tour hits your city","contentDescription":"See how leading teams deploy agents at scale. Find a stop near you. Register free.","contentJustifyContent":"center","linkStyle":"text-white","linkCTA":{"id":"link-cta","heapButtonClasses":["pushdown_banner"],"showOutboundIcon":false,"buttonLink":{"valid":true,"attributes":{"target":"_blank"},"url":"/en/world-tour/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Register now"},":type":"snowflake-site/components/pushdown-banner","appliedCssClassNames":"snowflake-pushdown-banner-text-white snowflake-pushdown-banner-background-black"}},":itemsOrder":["pushdown_banner_copy"]},"image":{":type":"nt:unstructured"},"cq:metadata":{":type":"nt:unstructured"}},":itemsOrder":["root","image","cq:metadata"],"classNames":"aem-xf"},"experiencefragment-header":{"id":"experiencefragment-dc974350d4","localizedFragmentVariationPath":"/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/jcr:content","configured":true,":type":"snowflake-site/components/experiencefragment",":items":{"root":{"columnClassNames":{"mega_header":"aem-GridColumn aem-GridColumn--default--12","markup_editor":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-78dc5ec313",":type":"snowflake-site/components/container",":items":{"markup_editor":{"id":"markup-editor-c98aede2e2","title":" ","cssContent":".footer-nav__link-group .snowflake-button-container,.subnav__item--button,.snowflake-card-v2-advanced-button .snowflake-button-container{justify-content:flex-start}.mega-nav__sign-in.snowflake-button-container{display:none}@media screen and (min-width:768px){.mega-nav__sign-in.snowflake-button-container{display:inline-block;font-family:'Texta',sans-serif;font-weight:800 !important}}@media screen and (min-width:1024px) and (max-width:1199px){.snowflake-mega-nav-header-buttons-container .snowflake-button-blue .snowflake-button-container{font-size:13px !important}.snowflake-language-navigation .language-icon{width:18px !important;height:18px !important;margin-right:4px !important}}.mega-nav__sign-in svg{display:none}.nav-item__platform-parent-why-sf.snowflake-mega-nav-nav-item\u003Ea:hover,.nav-item__platform-parent.snowflake-mega-nav-nav-item\u003Ea:hover{background-color:transparent !important}.nav-platform-sidebar .snowflake-mega-nav-nav-item:hover.blue-icon .snowflake-mega-nav-nav-item-icon__inner{background-color:var(--ui-01) !important}@media screen and (min-width:1024px){.snowflake-mega-nav-navigation-dropdown{overflow:hidden}.meganav-platform-features{padding-left:64px}.meganav-platform-features::before{content:'';transform:translateX(-64px);display:block;z-index:0;width:100%;height:100%;position:absolute;top:0;background:#f7f9fa}.nav-item--si.snowflake-mega-nav-nav-item\u003Ea:hover{background-color:transparent}.nav-item--si{border-bottom:1px solid #ccc;padding-bottom:16px;margin-bottom:8px}.nav-item__platform-parent{border-bottom:1px solid #ccc;margin-bottom:8px;padding-bottom:16px}.nav-item__platform-parent-why-sf .snowflake-mega-nav-nav-item-description::after{content:'What Snowflake can do for you \u003E';display:block;color:var(--ui-01);margin-top:16px}.nav-item__platform-parent .snowflake-mega-nav-nav-item-description::after{content:'View the platform \u003E';display:block;color:var(--ui-01);margin-top:16px}}@media screen and (min-width:1367px){.snowflake-mega-nav-nav-item-description{font-size:13px !important;line-height:20px !important}.snowflake-mega-nav-nav-item-title-wrapper\u003E.snowflake-mega-nav-nav-item-title{font-size:17px !important}.nav-item__platform-parent-why-sf .snowflake-mega-nav-nav-item-title,.nav-item__platform-parent .snowflake-mega-nav-nav-item-title{font-size:24px !important;line-height:32px !important;margin-bottom:8px !important}.nav-item__platform-parent-why-sf .snowflake-mega-nav-nav-item-description,.nav-item__platform-parent .snowflake-mega-nav-nav-item-description{font-size:14px !important;line-height:20px !important}}html.wf-texta-n9-loading .display-1-v2{font-size:48px!important;line-height:50px!important;letter-spacing:-.5px!important;font-family:sans-serif!important}html.wf-texta-n9-loading .heading-4-v2{font-size:18px!important;line-height:24px!important;font-family:sans-serif!important}@media screen and (min-width:768px){html.wf-texta-n9-loading .display-2-v2{font-size:48px!important;line-height:50px!important;font-family:sans-serif!important}html.wf-texta-n9-loading .display-1-v2{font-size:55.5px!important;line-height:54px!important;letter-spacing:-.5px!important;font-family:sans-serif!important}html.wf-lato-n4-loading .body-2,html.wf-lato-n4-loading .heading-5-v2,html.wf-lato-n4-loading .snowflake-card-v2-advanced-text .snowflake-text p{font-size:15.5px!important;font-family:sans-serif!important}html.wf-texta-n9-loading .heading-2,html.wf-texta-n9-loading .heading-2-v2{font-size:34px!important;line-height:38px!important;letter-spacing:-.75px!important;font-family:sans-serif!important}html.wf-texta-n8-loading .heading-6-v2.snowflake-mega-nav-navigation-title{font-size:13.5px!important;font-family:sans-serif!important}html.wf-texta-n8-loading .heading-4,html.wf-texta-n8-loading .snowflake-button-container,html.wf-texta-n8-loading .snowflake-button-regular .snowflake-button-container{font-size:13px!important;line-height:20px!important;letter-spacing:.25px!important;font-family:sans-serif!important}}@media screen and (min-width:1024px){html.wf-lato-n4-loading .snowflake-mega-nav-nav-item-description{font-size:11.5px!important;font-family:sans-serif!important}html.wf-lato-n4-loading .body-2,html.wf-lato-n4-loading .text-size-regular .snowflake-text li,html.wf-lato-n4-loading .text-size-regular .snowflake-text p,html.wf-lato-n4-loading .text-size-regular .snowflake-text span[data-testid=text-content],html.wf-lato-n4-loading .text-size-regular.cq-Editable-dom li,html.wf-lato-n4-loading .text-size-regular.cq-Editable-dom p,html.wf-lato-n4-loading .text-size-regular.cq-Editable-dom span[data-testid=text-content]{font-size:13.5px!important;font-family:sans-serif!important}html.wf-texta-n8-loading .snowflake-button-compact .snowflake-button-container{font-size:12px!important;letter-spacing:0!important;line-height:18px!important}}@media screen and (min-width:1367px){html.wf-lato-n4-loading .hp-hero__eyebrow a\u003Eb:first-child{font-size:11px!important;font-family:sans-serif!important}html.wf-texta-n8-loading .hp-hero__eyebrow a{font-size:13px!important;font-family:sans-serif!important}html.wf-texta-n9-loading .display-2-v2{font-size:61px!important;line-height:60px!important;font-family:sans-serif!important}html.wf-texta-n9-loading .display-1-v2{font-size:74.5px!important;line-height:74px!important;letter-spacing:-.75px!important;font-family:sans-serif!important}html.wf-texta-n9-loading .heading-2,html.wf-texta-n9-loading .heading-2-v2{font-size:41px!important;letter-spacing:-.75px!important;font-family:sans-serif!important}html.wf-texta-n9-loading .heading-3-v2{font-family:sans-serif!important;letter-spacing:-.75px!important;font-size:33.75px!important}html.wf-texta-n9-loading .heading-4-v2{font-size:19.5px!important;line-height:26px!important;font-family:sans-serif!important}html.wf-texta-n8-loading .heading-6-v2{font-size:12px!important;font-family:sans-serif!important}html.wf-texta-n8-loading .heading-6-v2.snowflake-mega-nav-navigation-title{font-size:14px!important;font-family:sans-serif!important}html.wf-lato-n4-loading .body-1,html.wf-lato-n4-loading .cq-Editable-dom[data-cq-data-path*=text] ol\u003Eli,html.wf-lato-n4-loading .snowflake-text li,html.wf-lato-n4-loading .snowflake-text p,html.wf-lato-n4-loading .text-size-large .snowflake-text li,html.wf-lato-n4-loading .text-size-large .snowflake-text p,html.wf-lato-n4-loading .text-size-large .snowflake-text span[data-testid=text-content],html.wf-lato-n4-loading .text-size-large.cq-Editable-dom li,html.wf-lato-n4-loading .text-size-large.cq-Editable-dom p,html.wf-lato-n4-loading .text-size-large.cq-Editable-dom span[data-testid=text-content],html.wf-lato-n4-loading.cq-Editable-dom[data-cq-data-path*=text]\u003Ep,html.wf-lato-n4-loading.cq-Editable-dom[data-cq-data-path*=text]\u003Eul\u003Eli{font-size:17.5px!important;font-family:sans-serif!important}html.wf-lato-n4-loading .body-2,html.wf-lato-n4-loading .text-size-regular .snowflake-text li,html.wf-lato-n4-loading .text-size-regular .snowflake-text p,html.wf-lato-n4-loading .text-size-regular .snowflake-text span[data-testid=text-content],html.wf-lato-n4-loading .text-size-regular.cq-Editable-dom li,html.wf-lato-n4-loading .text-size-regular.cq-Editable-dom p,html.wf-lato-n4-loading .text-size-regular.cq-Editable-dom span[data-testid=text-content],html.wf-texta-n8-loading .snowflake-button-link .snowflake-button-container,html.wf-texta-n8-loading .snowflake-button-link-back .snowflake-button-container{font-size:15.5px!important;font-family:sans-serif!important}html.wf-lato-n4-loading .body-3,html.wf-lato-n4-loading .text-size-small .snowflake-text li,html.wf-lato-n4-loading .text-size-small .snowflake-text p,html.wf-lato-n4-loading .text-size-small .snowflake-text span[data-testid=text-content],html.wf-lato-n4-loading .text-size-small.cq-Editable-dom li,html.wf-lato-n4-loading .text-size-small.cq-Editable-dom p,html.wf-lato-n4-loading .text-size-small.cq-Editable-dom span[data-testid=text-content]{font-size:13.5px!important;font-family:sans-serif!important}}#industryPlatformSection,.sc-hero{background-position:top left;background-size:20% auto}.bwalignc,.bwalignr{list-style-position:inside}.snowflake-text p sup{font-size:10px}#industryPlatformSection .industry-platform__row .snowflake-flexible-column-container-items,.button-group-pair\u003E.container\u003E.cmp-container\u003E.aem-container,.snowflake-hero-system-content-container{gap:16px}.agenda-item p,.button-group-pair\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv,.partner-details p{margin:0!important}.button-group-pair\u003E.container\u003E.cmp-container\u003E.aem-container::after,.button-group-pair\u003E.container\u003E.cmp-container\u003E.aem-container::before,.hide-logo .snowflake-case-study-card-logo,.partner-page__powered-by-logo,.sc-hero div.code-toolbar\u003E.toolbar,.snowflake-card-v2-advanced.no-link .snowflake-card-v2-advanced-button,.snowflake-partner-hero-card-badge-container{display:none!important}.section--card-mobile-carousel .snowflake-flexible-column-container-items-with-carousel{max-width:100%!important}@media screen and (min-width:768px){.button-group-pair .snowflake-button-container.inline-button--desktop,.button-group-pair\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:auto!important;display:inline-block!important}.button-group-pair\u003E.container\u003E.cmp-container\u003E.aem-container{align-items:center;justify-content:flex-start!important}.button-group-pair.center\u003E.container\u003E.cmp-container\u003E.aem-container{justify-content:center!important}.section--card-mobile-carousel{margin-left:var(--tablet-portrait-margin,48px)!important;margin-right:var(--tablet-portrait-margin,48px);width:calc(100% - 96px)!important;width:calc(100% - var(--tablet-portrait-margin) * 2)!important}}@media screen and (min-width:1024px){.section--card-mobile-carousel{margin-left:var(--tablet-horizontal-margin,48px)!important;margin-right:var(--tablet-horizontal-margin,48px);width:calc(100% - 96px)!important;width:calc(100% - var(--tablet-horizontal-margin) * 2)!important}.snowflake-mega-nav-header-mobile-icon{display:none!important}}@media screen and (min-width:1367px){.section--card-mobile-carousel{margin-left:var(--desktop-margin,6.5%)!important;margin-right:var(--desktop-margin,6.5%);width:87%!important;width:calc(100% - var(--desktop-margin) * 2)!important}.logo-container{min-width:143px}.sc-hero__headline .heading-1-v2{font-size:60px}.snowflake-mega-nav-navigation-title{font-size:17px}.snowflake-mega-nav-dropdown-footer-wrapper .snowflake-title-v2 .snowflake-title-v2-line:first-child{font-size:16px!important;line-height:24px!important}}.hero--home{overflow:hidden;background-color:var(--ui-01);z-index:2}.hp-hero__subheadline{width:90%}.hero--home .snowflake-button-container{transition:.3s}.hero--home .snowflake-button-primary a:hover,.hero--home .snowflake-button-secondary a:hover,.hero--home .snowflake-button-white a:hover{transition:.3s;background-color:var(--ui-02)!important;color:var(--ui-05)!important}.hero--home .snowflake-button-secondary a:hover{border-color:var(--ui-05)!important}.hero--home .snowflake-button-primary a:hover,.hero--home .snowflake-button-white a:hover{border-color:var(--ui-02)!important}.bwalignc,.hp-hero__eyebrow{text-align:center}.hp-hero__eyebrow a{display:inline-flex;flex-direction:column;justify-content:center;cursor:pointer;padding:8px;border-radius:var(--spacing-01);gap:8px;align-items:center;background-color:#45aee3;color:var(--ui-03);font-family:Texta,sans-serif;font-weight:800;font-size:16px;line-height:22px;transition:background-color .3s}.hp-hero__eyebrow a:hover{background-color:#7fc6ea;text-decoration:none;transition:background-color .3s}.hp-hero__eyebrow a\u003Eb:first-child{text-transform:uppercase;white-space:nowrap;display:inline-block;background-color:var(--ui-02);color:var(--ui-05);font-size:12px!important;line-height:16px!important;font-family:Lato,sans-serif;font-weight:500!important;padding:3px 6px;border-radius:2px;letter-spacing:1px}@media screen and (min-width:767px){.hp-hero__eyebrow{text-align:left}.hp-hero__eyebrow a{flex-direction:row;text-align:left}}.hero--home__inner .offset-video,.hero--home__inner .snowflake-experience-fragment,.offset-video__bg-image{max-height:200px;overflow:hidden}.hero--home__inner .offset-video .wistia-responsive-padding{padding-top:100%}.hero--home__inner .snowflake-experience-fragment,.offset-video__bg-image{position:absolute!important;top:0;left:0;width:100%}.offset-video__bg-image{z-index:-1}@media screen and (min-width:768px){.hero--home__inner .snowflake-experience-fragment,.offset-video,.offset-video__bg-image{position:absolute!important;max-height:none;top:0;left:0;width:250%;padding-bottom:250%;transform:translate(0,-50%);height:0}.workloads_7.unistore{max-width:317px}}.promo-banner--homepage{z-index:2}.homepage-banner-offset-container::after{content:\"\";display:block;position:absolute;bottom:0;z-index:1;left:0;width:100%;height:80%;background:#fff}.section--quicklinks .snowflake-button-full-width a{padding-left:24px!important;padding-right:24px!important;transition:box-shadow .25s cubic-bezier(.4,0,.2,1);text-align:left;display:flex;justify-content:center;align-items:center}.section--quicklinks .snowflake-button-full-width a:hover{box-shadow:0 16px 16px 0 rgb(0 0 0 / .16);transition:box-shadow .25s cubic-bezier(.4,0,.2,1)}.section--quicklinks .snowflake-button-container:focus-visible a::before,.section--quicklinks .snowflake-button-full-width a::before{content:\"\";width:23px;height:23px;flex-shrink:0;margin-right:12px;display:inline-block;background-size:cover;background-repeat:no-repeat;background-position:center}#industryPartnerSlider .snowflake-navigation-icon.swiper-button-disabled,#partnerResources .section--resource-hub a svg,.button-tabs span.snowflake-tabs-navigation-item:after,.customer-card--hide-cta .snowflake-case-study-card-button,.dot-tabs span.snowflake-tabs-navigation-item::after,.partner-sidebar__mobile-expand,html:not(.aem-AuthorLayer-initial):not(.aem-AuthorLayer-Edit) .tab-content:not(.is-active){display:none}.section--quicklinks .snowflake-button-full-width a.pricing::before{background-image:url(https://www.snowflake.com/content/dam/snowflake-site/general/icons/decorative-icons/pricing-icon.svg)}.section--quicklinks .snowflake-button-full-width a.snowflake_on_snowflake::before{background-image:url(https://www.snowflake.com/content/dam/snowflake-site/general/icons/navigation/nav-icon_snowflake-bug.svg)}.section--quicklinks .snowflake-button-full-width a.virtual_hands_on_labs::before{background-image:url(https://www.snowflake.com/content/dam/snowflake-site/general/icons/navigation/nav-icon__training.svg)}.section--quicklinks .snowflake-button-full-width a.weekly_demo::before{background-image:url(https://www.snowflake.com/content/dam/snowflake-site/general/icons/navigation/nav-icon__webinars.svg)}@media screen and (min-width:1024px){.hero--home__inner .snowflake-experience-fragment,.offset-video,.offset-video__bg-image{left:-50%}.section--quicklinks .snowflake-flexible-column-container-items{gap:24px}.snowflake-quote-item-inner{padding:32px 24px 24px!important}}#communitiesOuter_overflowBottomGray::after{max-height:100px}#caseStudyOuter_overflowBottomMidBlue::after{max-height:180px}#caseStudyInner .snowflake-case-study-card .snowflake-wistia-video{border-radius:0!important}#caseStudyInner .snowflake-case-study-card{box-shadow:none!important;border-radius:0}#caseStudyInner{max-width:1200px;margin:0 auto;box-shadow:rgb(152 162 179 / .1) 0 10px 20px 0,rgb(152 162 179 / .25) 0 2px 6px 0;border-radius:8px;overflow:hidden;position:relative;z-index:1}.case-study__logo-bar\u003E.snowflake-flexible-column-container-items{background:#f7f9fa;padding:32px 16px 40px}.case-study__logo-bar .cmp-image__image{width:90%;margin:0 auto;max-width:240px}.hp-platform__text-group\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:not(:first-child),.sc-sidebar__group .snowflake-button-link{margin-top:8px}.workloads_7.unistore{margin-left:auto;margin-right:auto}#homepageFootnotesInner .snowflake-simple-stat-disclaimer .snowflake-text p{color:#fff!important}.snowflake-simple-stat-disclaimer .snowflake-text p\u003Ea{border-bottom:1px solid var(--ui-03);color:var(--text-03)}.snowflake-card-v2-advanced{color:inherit}#workloadCardGridOuter .snowflake-card-v2-base-front{gap:0}.video-modal.snowflake-modal-window-open-inner{background-color:#fff0;padding:8px;border:none}.snowflake-container-arrow-dotted-faded .snowflake-container-arrow-dotted-faded-image{width:40%!important;max-width:420px;top:4%!important}.list--blue-bullets ul{margin:0!important;padding:0!important;list-style-type:none}.list--blue-bullets li{margin:0;padding:0 0 0 32px;position:relative}.list--blue-bullets li::before{content:\"\";display:block;border-radius:100%;background:#29b5e8;width:18px;height:18px;position:absolute;top:4px;left:0;border:5px solid #e5f2f7;box-sizing:border-box}.list--blue-bullets li:not(:last-child){margin-bottom:1rem}.logo-tabs .snowflake-navigation-container,.snowflake-simple-stat-content:empty,.summit-speaker-card .snowflake-card-v2-advanced-text{margin-bottom:0}#techResourceInner,#techResourceOuter,div.overflow-bottom--blue,div.overflow-bottom--gray,div.overflow-bottom--mid-blue,div.overflow-bottom--white,div.overflow-top--blue,div.overflow-top--gray,div.overflow-top--mid-blue,div.overflow-top--white,div[id$=overflowBottomGray],div[id$=overflowBottomMidBlue],div[id$=overflowTopBlue],div[id$=overflowTopGray]{position:relative}div.overflow-bottom--blue::after,div.overflow-bottom--gray::after,div.overflow-bottom--mid-blue::after,div.overflow-bottom--white::after,div.overflow-top--blue::after,div.overflow-top--gray::after,div.overflow-top--mid-blue::after,div.overflow-top--white::after,div[id$=overflowBottomGray]::after,div[id$=overflowBottomMidBlue]::after,div[id$=overflowBottomWhite]::after,div[id$=overflowTopBlue]::after,div[id$=overflowTopGray]::after,div[id$=overflowTopWhite]::after{content:\"\";display:block;position:absolute;left:0;width:100%;height:40%}div.overflow-top--blue::after,div.overflow-top--gray::after,div.overflow-top--mid-blue::after,div.overflow-top--white::after,div[id$=overflowTopBlue]::after,div[id$=overflowTopGray]::after,div[id$=overflowTopWhite]::after{top:0}div.overflow-bottom--blue::after,div.overflow-bottom--gray::after,div.overflow-bottom--mid-blue::after,div.overflow-bottom--white::after,div[id$=overflowBottomGray]::after,div[id$=overflowBottomMidBlue]::after,div[id$=overflowBottomWhite]::after{bottom:0}div.overflow-bottom--white::after,div.overflow-top--white::after,div[id$=overflowBottomWhite]::after,div[id$=overflowTopWhite]::after{background:#fff!important}div.overflow-bottom--gray::after,div.overflow-top--gray::after,div[id$=overflowBottomGray]::after,div[id$=overflowTopGray]::after{background:#f6f9fa!important}div.overflow-bottom--mid-blue::after,div.overflow-top--mid-blue::after,div[id$=overflowBottomMidBlue]::after,div[id$=overflowTopMidBlue]::after{background:#11567f!important}div.overflow-bottom--blue::after,div.overflow-top--blue::after,div[id$=overflowBottomBlue]::after,div[id$=overflowTopBlue]::after{background:#259edc!important}.snowflake-premium-content-banner.promo-banner--no-shadow{box-shadow:none!important}#industryPartnerSlider .cmp-image__image,#industryPartnerSlider .section--partner-tabs .snowflake-image-container .cmp-image__image,#partnerSidebar,.has-shadow .cmp-image__image{box-shadow:0 10px 20px 0 rgb(152 162 179 / .1),0 2px 6px 0 rgb(152 162 179 / .25)}.content-chip--has-desc{align-items:flex-start;padding:20px!important}.content-chip--has-desc .snowflake-content-chip-image{max-width:100px}.content-chip--has-desc .snowflake-content-chip-image__image{aspect-ratio:1}.content-chip--has-desc .snowflake-title-v2-line:first-child{font-size:18px!important}.content-chip--has-desc .snowflake-title-v2-line:nth-child(2){color:#000!important;font-weight:500!important;font-size:16px!important;line-height:22px!important;margin-top:2px!important}.content-chip--has-desc .snowflake-content-chip-button{margin-top:6px!important;font-size:18px!important;display:none}.square-image .snowflake-content-chip-image{aspect-ratio:1;max-width:120px}.section--logo-bar.smaller-logos .snowflake-image-container .cmp-image__image{max-width:200px;margin:0 auto}.snowflake-card-v2-advanced-tag,.snowflake-content-chip-tag{padding:3px 6px!important}.sc-overview__webinar-promo-banner .snowflake-content-chip-button,.snowflake-card-v2-advanced-title:first-child,.summit-pricing-block__aside ul{margin-top:0}.dot-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item{width:40px;height:40px;display:flex;justify-content:center;align-items:center;margin:0!important}.dot-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item p{width:12px;height:12px;background:var(--ui-12);border-radius:100%}.dot-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item p,.logo-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item p{font-size:0!important}.dot-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item.active p{background:var(--ui-01)}.button-tabs .snowflake-navigation-container .swiper-wrapper{padding:8px 0}.button-tabs .snowflake-navigation-container .swiper-slide{margin:0 6px}.button-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item{padding:8px 24px;background-color:#f6f9fa;border-radius:48px;margin:0}.button-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item p{text-transform:uppercase;font-family:Texta,sans-serif;font-weight:700}.button-tabs .border-top{border-top:1px solid #ccc}.button-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item.active{background-color:var(--ui-01);box-shadow:0 2px 6px 0 rgb(152 162 179 / .25),0 10px 20px 0 rgb(152 162 179 / .1)}.button-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item.active p{color:#fff}.button-tabs.has-icons .snowflake-navigation-container .snowflake-tabs-navigation-item p::before{content:\"\";display:inline-block;width:20px;height:20px;background-size:contain;background-repeat:no-repeat;background-position:center center;margin-right:12px;vertical-align:middle;margin-top:-3px}.logo-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item{width:220px;padding-bottom:50%;height:0;margin:0 8px!important;background-size:cover;background-repeat:no-repeat;opacity:.5;transition:opacity .3s}.logo-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item:hover{opacity:.75;transition:opacity .3s}.logo-tabs .snowflake-navigation-container .snowflake-tabs-navigation-item.active{opacity:1;transition:opacity .3s}.dot-tabs .aem-container.cmp-tabs,.logo-tabs .aem-container.cmp-tabs{display:flex;flex-direction:column-reverse}.snowflake-icon.is-center{margin:0 auto;display:block}#industryPartnerSlider .snowflake-flexible-column-container-items,#partnerLogoSquare .snowflake-flexible-column-container-items{gap:24px}#techResourceOuter::after{content:\"\";display:block;position:absolute;top:0;left:0;width:100%;height:40%;background:#f6f9fa}#techResourceInner{z-index:1}.partner-tier-tag h6{display:inline-block!important;padding:2px 6px;border-radius:2px;color:#666}.partner-tier-tag.registered h6{background-color:#f6f9fa}.partner-tier-tag.elite h6{background-color:#11567f;color:#fff}.partner-tier-tag.premier h6{background-color:#b14c77;color:#fff}.partner-tier-tag.select h6{background-color:#5094a0;color:#fff}.partner-details\u003Espan{display:flex;gap:24px}.partner-details a{color:inherit!important;font-weight:400!important}.partner-details p::before{content:\"\";display:inline-block;vertical-align:middle;width:16px;height:16px;background-repeat:no-repeat;background-position:center;transform:translateY(-1px);background-size:auto 90%;margin-right:6px}.partner-details__location::before{background-image:url(\"data:image/svg+xml,%3Csvg width='13' height='18' viewBox='0 0 13 18' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M6.25 17.7531C6.4375 17.7531 6.6 17.6844 6.7375 17.5531C6.875 17.4219 6.95 17.2531 6.95 17.0531C6.95 16.8531 7.075 16.4281 7.3 15.7969C7.5875 15.0281 7.925 14.3156 8.30625 13.6406C8.8 12.7781 9.3125 12.1031 9.85 11.6094C10.75 10.7969 11.4125 9.96563 11.85 9.12188C12.2875 8.27813 12.5063 7.40313 12.5063 6.49063C12.5063 5.36563 12.2187 4.31563 11.6437 3.33438C11.0937 2.40313 10.3438 1.65938 9.4 1.10938C8.43125 .534376 7.375 .246876 6.24375 .246876C5.1125 .246876 4.06875 .534376 3.0875 1.10938C2.15625 1.65938 1.4125 2.40313 .862498 3.33438C.287498 4.31563 0 5.36563 0 6.49063C0 7.47188 .262499 8.42813 .787499 9.35938C1.14375 10.0031 1.65625 10.6656 2.3125 11.3344C2.75625 11.8031 3.24375 12.4781 3.78125 13.3656C4.225 14.0969 4.63125 14.8594 5 15.6656C5.35 16.3844 5.53125 16.8531 5.55625 17.0656C5.55625 17.2594 5.625 17.4156 5.7625 17.5531C5.9 17.6844 6.0625 17.7531 6.25 17.7531ZM6.16875 14.9156C5.775 14.0656 5.325 13.2469 4.825 12.4594C4.275 11.5594 3.7625 10.8719 3.28125 10.3969C2.625 9.71563 2.1375 9.05938 1.825 8.43438C1.5125 7.80313 1.35625 7.16563 1.35625 6.50313C1.35625 5.61563 1.575 4.80313 2.0125 4.05313C2.45 3.30313 3.04375 2.71563 3.7875 2.27813C4.5375 1.84063 5.35 1.62188 6.2375 1.62188C7.125 1.62188 7.9375 1.84063 8.6875 2.27813C9.4375 2.71563 10.0312 3.30313 10.475 4.04688C10.9187 4.80313 11.1375 5.62188 11.1375 6.50313C11.1375 7.90313 10.3937 9.26563 8.9125 10.5969C8.35 11.1094 7.8125 11.7906 7.3 12.6406C6.88125 13.3344 6.50625 14.0969 6.16875 14.9219V14.9156ZM6.26875 8.36563C6.65625 8.36563 7.01875 8.26563 7.35625 8.07188C7.69375 7.87813 7.95625 7.60938 8.14375 7.28438C8.3375 6.95313 8.43125 6.59063 8.43125 6.19688C8.43125 5.80313 8.33125 5.43438 8.1375 5.10313C7.9375 4.76563 7.675 4.50313 7.3375 4.31563C7 4.12813 6.6375 4.02813 6.24375 4.02813C5.85 4.02813 5.4875 4.12813 5.15625 4.32188C4.825 4.52188 4.56875 4.78438 4.375 5.12188C4.18125 5.45938 4.0875 5.82188 4.0875 6.20938C4.0875 6.59688 4.1875 6.95938 4.38125 7.29688C4.58125 7.63438 4.84375 7.89688 5.18125 8.08438C5.51875 8.27813 5.88125 8.37188 6.26875 8.37188V8.36563ZM6.24375 7.50313C5.8875 7.50313 5.575 7.37188 5.31875 7.11563C5.0625 6.85938 4.93125 6.55313 4.93125 6.19063C4.93125 5.82813 5.0625 5.52188 5.31875 5.26563C5.575 5.00938 5.88125 4.87813 6.24375 4.87813C6.60625 4.87813 6.9125 5.00938 7.16875 5.26563C7.425 5.52188 7.55625 5.82813 7.55625 6.19063C7.55625 6.55313 7.425 6.85938 7.16875 7.11563C6.9125 7.37188 6.60625 7.50313 6.24375 7.50313Z' fill='%2329B5E8'/%3E%3C/svg%3E%0A\")}.partner-details__website::before{background-image:url(\"data:image/svg+xml,%3Csvg width='18' height='16' viewBox='0 0 18 16' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M2.61587 2.96889C2.61587 2.75109 2.79633 2.57062 3.01413 2.57062C3.23192 2.57062 3.41238 2.75109 3.41238 2.96889C3.41238 3.18669 3.23192 3.36716 3.01413 3.36716C2.79633 3.36716 2.61587 3.18669 2.61587 2.96889ZM4.21512 2.96889C4.21512 2.75109 4.39558 2.57062 4.61338 2.57062C4.83117 2.57062 5.01163 2.75109 5.01163 2.96889C5.01163 3.18669 4.83117 3.36716 4.61338 3.36716C4.39558 3.36716 4.21512 3.18669 4.21512 2.96889ZM5.81438 2.96889C5.81438 2.75109 5.99484 2.57062 6.21264 2.57062C6.43043 2.57062 6.61089 2.75109 6.61089 2.96889C6.61089 3.18669 6.43043 3.36716 6.21264 3.36716C5.99484 3.36716 5.81438 3.18669 5.81438 2.96889ZM17.2518 .697559H1.19085C.811258 .697559 .506348 1.0025 .506348 1.38209V14.6179C.506348 14.9975 .811258 15.3024 1.19085 15.3024H17.2518C17.6314 15.3024 17.9363 14.9975 17.9363 14.6179V1.38209C17.9363 1.0025 17.6314 .697559 17.2518 .697559ZM16.5673 2.06035V3.90853H1.86914V2.06035H16.5673ZM1.86914 13.9334V4.78593H16.5673V13.9334H1.86914Z' fill='%2329B5E8'/%3E%3C/svg%3E%0A\")}#partnerSidebar{border-radius:4px;background-color:#fff;padding:24px 24px 32px;border-bottom:6px solid #29b5e8}#partnerSidebar h5,.newsletter-disclaimer p{font-size:14px!important}#partnerSidebar ul{margin-top:0;list-style-type:none;padding:0;display:flex;flex-wrap:wrap;gap:8px}#partnerSidebar li{border:1px solid;border-radius:2px;padding:0 4px!important;font-size:11px!important;letter-spacing:.25px;text-transform:uppercase}div.snowflake-partner-hero-card{width:100%;margin:0}.partner-details__logo{max-width:380px;margin:0 auto}@media screen and (max-width:767px){.left-alignment .hp-hero__subheadline{margin-left:auto;margin-right:auto}.left-alignment .hp-hero__headline .snowflake-title-v2-line,.left-alignment .hp-hero__subheadline .snowflake-title-v2-line{text-align:center}.hero--home__inner .snowflake-flexible-column-container-items-top-padding-large{padding-top:var(--spacing-02)}.section--logo-bar\u003E.snowflake-flexible-column-container-items{display:flex;flex-wrap:wrap;flex-direction:row;justify-content:center;gap:8px}.section--logo-bar\u003E.snowflake-flexible-column-container-items\u003Ediv{width:calc(33.33% - 8px)}.partner-sidebar__mobile-expand{display:inline-block;color:#249edc;border-color:#249edc!important}#partnerSidebar li:nth-child(n+6),.summit-nav__links .snowflake-button-tertiary{display:none}.sc-body__sidebar{background-color:#f6f9fa;padding:24px}.sc-body__content{padding:0 24px 24px}.summit-speaker-card .snowflake-card-v2-advanced-content{padding:24px}}#partnerResources h6,.snowflake-tabs-navigation-item p.body-1{font-size:16px!important}#partnerResources .section--resource-hub{padding:0 16px}#partnerResources .section--resource-hub a,.bwalignl{text-align:left}@media screen and (max-width:1023px){.hero--workload .snowflake-hero-system-media-container{width:100%}}.section--timely-content .snowflake-content-chip,.snowflake-mega-nav-dropdown-footer-wrapper{align-items:center}.section--timely-content .snowflake-content-chip-image{max-width:94px}.section--timely-content .snowflake-content-chip-image__inner{line-height:0}.section--timely-content .snowflake-content-chip-image__image{aspect-ratio:1;height:auto}.section--workload-overview .workload-overview__headline{max-width:280px;margin:0 auto}#industryPartnerSlider .swiper-slide{margin-top:0!important;padding:0 12px}#industryPartnerSlider .snowflake-tabs-navigation-item{margin-left:0!important;margin-right:0!important}#industryPartnerSlider .snowflake-premium-content-banner-background-grad-white .snowflake-premium-content-banner{box-shadow:none}#industryPartnerSlider .logo-slider__slide .aem-container{display:flex;padding:0 8px!important;flex-wrap:wrap;gap:16px!important;justify-content:center}#industryPartnerSlider .logo-slider__slide .aem-container\u003Ediv{width:48%;max-width:200px}#useCaseTabs{padding-top:24px;padding-bottom:24px;padding-right:24px}#useCaseTabs .tab-content.is-active{display:block}#useCaseTabs .vert-tab{border-bottom:1px solid #a0bbcc;padding-bottom:16px}#useCaseTabs .vert-tab p{display:inline-block}#useCaseTabs .vert-tab p:hover{cursor:pointer}#useCaseTabs .vert-tab p,#useCaseTabs .vert-tab.is-active p.not-active{color:#249edc}#useCaseTabs .vert-tab p.is-active,#useCaseTabs .vert-tab.is-active p{color:#000}#industryPlatformSection{background-image:url(/adobe/dynamicmedia/deliver/dm-aid--db074ad5-7122-4c51-87a3-76c3aa466182/double-arrow-bg%403x.png);background-repeat:no-repeat}.snowflake-text p.featured-quote__source{font-weight:900!important;text-transform:uppercase;font-size:16px!important;margin-top:2rem!important}.snowflake-text p.featured-quote__title{margin-top:0!important;font-size:16px!important}.snowflake-case-study-card-logo img{width:auto!important;height:100px!important;transform:translateX(-15%)}.snowflake-quote-item-quote-text{font-weight:600!important}#customerStoryStatsInner\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;flex-direction:row}#customerStoryStat1,#customerStoryStat2{max-width:240px}#storyHighlights{border-radius:4px;padding:1rem}.sc-overview__webinar-promo-banner .snowflake-content-chip-content .snowflake-title-v2-line,.summit-pricing-block__tile .black-blue-text-color .snowflake-title-v2-line{color:#000!important}.snowflake-youtube-embedded-wrapper{border-radius:var(--small-border-radius)}#arcticNavItem::before,#offset::before,#open-source::before{color:var(--text-05);font-family:Texta,sans-serif!important}#offset,.sc-architecture-caption{margin-top:16px}.hero--press .snowflake-title-v2-line{text-transform:none!important}@media screen and (min-width:768px){.subpage-timely-content__inner\u003E.snowflake-flexible-column-container-items{box-shadow:0 10px 20px 0 rgb(152 162 179 / .1),0 2px 6px 0 rgb(152 162 179 / .25);padding:var(--spacing-04);border-radius:4px;overflow:hidden}#partnerLogoSquare{padding:0 0 0 48px}.hero--workload .snowflake-container{max-width:1440px;margin:0 auto!important;align-items:center}#industryPartnerSlider.snowflake-flexible-column-container-2-column-40-60\u003E.snowflake-flexible-column-container-items{grid-template-columns:minmax(40%,4fr) minmax(0,6fr)}#industryPartnerSlider .swiper-slide{padding:0 24px}.sc-body{padding:48px}.sc-body\u003E.snowflake-flexible-column-container-items{grid-template-columns:7fr 3fr;gap:124px}}.snowflake-button-container.has-icon{display:inline-flex;justify-content:center;align-items:center;text-align:left}.snowflake-button-container.has-icon::before{content:\"\";display:inline-block;width:20px;height:20px;margin-right:12px;background-size:contain;background-repeat:no-repeat;background-position:center}.snowflake-button-container.is-video::before{background-image:url(\"data:image/svg+xml,%3Csvg width='18' height='18' viewBox='0 0 18 18' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M9 1.28663C13.2523 1.28663 16.7134 4.74768 16.7134 9C16.7134 13.2523 13.2523 16.7134 9 16.7134C4.74768 16.7198 1.28663 13.2588 1.28663 9C1.28663 4.74124 4.74768 1.28663 9 1.28663ZM9 0C4.0336 0 0 4.0336 0 9C0 13.9664 4.0336 18 9 18C13.9728 18 18 13.9664 18 9C18 4.0336 13.9728 0 9 0Z' fill='white'/%3E%3Cpath d='M7.75106 6.18211C7.42941 6.16925 7.16565 6.42658 7.16565 6.74823V11.2772C7.16565 11.7082 7.65457 11.9848 8.02126 11.7597L11.7975 9.4952C12.1578 9.27647 12.1578 8.74252 11.7975 8.52379L8.02126 6.25931C7.93763 6.21428 7.84756 6.18211 7.75106 6.18211Z' fill='white'/%3E%3C/svg%3E%0A\")}.snowflake-button-container.is-github::before{background-image:url(\"data:image/svg+xml,%3Csvg width='20' height='21' viewBox='0 0 20 21' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M10 .651794C4.475 .651794 0 5.12679 0 10.6518C0 15.0768 2.8625 18.8143 6.8375 20.1393C7.3375 20.2268 7.525 19.9268 7.525 19.6643C7.525 19.4268 7.5125 18.6393 7.5125 17.8018C5 18.2643 4.35 17.1893 4.15 16.6268C4.0375 16.3393 3.55 15.4518 3.125 15.2143C2.775 15.0268 2.275 14.5643 3.1125 14.5518C3.9 14.5393 4.4625 15.2768 4.65 15.5768C5.55 17.0893 6.9875 16.6643 7.5625 16.4018C7.65 15.7518 7.9125 15.3143 8.2 15.0643C5.975 14.8143 3.65 13.9518 3.65 10.1268C3.65 9.03929 4.0375 8.13929 4.675 7.43929C4.575 7.18929 4.225 6.16429 4.775 4.78929C4.775 4.78929 5.6125 4.52679 7.525 5.81429C8.325 5.58929 9.175 5.47679 10.025 5.47679C10.875 5.47679 11.725 5.58929 12.525 5.81429C14.4375 4.51429 15.275 4.78929 15.275 4.78929C15.825 6.16429 15.475 7.18929 15.375 7.43929C16.0125 8.13929 16.4 9.02679 16.4 10.1268C16.4 13.9643 14.0625 14.8143 11.8375 15.0643C12.2 15.3768 12.5125 15.9768 12.5125 16.9143C12.5125 18.2518 12.5 19.3268 12.5 19.6643C12.5 19.9268 12.6875 20.2393 13.1875 20.1393C17.1375 18.8143 20 15.0643 20 10.6518C20 5.12679 15.525 .651794 10 .651794Z' fill='%23249EDC'/%3E%3C/svg%3E%0A\")}.snowflake-button-container.is-quickstart::before{background-image:url(\"data:image/svg+xml,%3Csvg width='15' height='21' viewBox='0 0 15 21' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M13.8489 2.79368H11.6439V2.38493C11.6439 1.71368 11.1451 .967427 10.4251 .967427H8.94762C8.80887 .359927 8.37387 .299927 7.89762 .299927H7.23012C6.85512 .299927 6.26637 .299927 6.08637 .967427H4.68387C3.94887 .967427 3.35637 1.74368 3.35637 2.38493V2.79368H1.15137C.738867 2.79368 .401367 3.13118 .401367 3.54368V20.2537C.401367 20.6662 .738867 21.0037 1.15137 21.0037H13.8489C14.2614 21.0037 14.5989 20.6662 14.5989 20.2537V3.54368C14.5989 3.13118 14.2614 2.79368 13.8489 2.79368ZM4.29387 2.38493C4.29387 2.18243 4.54137 1.90493 4.68387 1.90493H6.50262C6.76137 1.90493 6.97137 1.69493 6.97137 1.43618C6.97137 1.33868 6.97887 1.27868 6.98637 1.24118C7.05012 1.23368 7.15512 1.23368 7.23387 1.23368H7.90137C7.95012 1.23368 8.00637 1.23368 8.05137 1.23368C8.05512 1.27868 8.05887 1.34243 8.05887 1.43243C8.05887 1.69118 8.26887 1.90118 8.52762 1.90118H10.4289C10.5301 1.90118 10.7101 2.14493 10.7101 2.38118V2.78993H4.29762V2.38118L4.29387 2.38493ZM13.0989 19.4999H1.90137V4.29368H13.0989V19.5037V19.4999Z' fill='%23249EDC'/%3E%3Cpath d='M3.82512 16.0424H11.1751C11.4339 16.0424 11.6439 15.8324 11.6439 15.5736V6.88486C11.6439 6.62611 11.4339 6.41611 11.1751 6.41611H3.82512C3.56637 6.41611 3.35637 6.62611 3.35637 6.88486V15.5736C3.35637 15.8324 3.56637 16.0424 3.82512 16.0424ZM4.29387 15.1049V13.3686H10.7064V15.1049H4.29387ZM10.7101 7.35361V12.4311H4.29762V7.35361H10.7101Z' fill='%23249EDC'/%3E%3Cpath d='M6.16512 9.35989H8.83887C9.09762 9.35989 9.30762 9.14989 9.30762 8.89114C9.30762 8.63239 9.09762 8.42239 8.83887 8.42239H6.16512C5.90637 8.42239 5.69637 8.63239 5.69637 8.89114C5.69637 9.14989 5.90637 9.35989 6.16512 9.35989Z' fill='%23249EDC'/%3E%3Cpath d='M6.16512 11.3624H8.83887C9.09762 11.3624 9.30762 11.1524 9.30762 10.8937C9.30762 10.6349 9.09762 10.4249 8.83887 10.4249H6.16512C5.90637 10.4249 5.69637 10.6349 5.69637 10.8937C5.69637 11.1524 5.90637 11.3624 6.16512 11.3624Z' fill='%23249EDC'/%3E%3C/svg%3E%0A\")}.snowflake-button-container.is-download::before{background-image:url(\"data:image/svg+xml,%3Csvg width='16' height='18' viewBox='0 0 16 18' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M15.2017 17.1637H.798265C.364425 17.1637 0 16.7993 0 16.3655V12.3568C0 11.923 .364425 11.5585 .798265 11.5585C1.2321 11.5585 1.59653 11.923 1.59653 12.3568V15.5498H14.4035V12.3568C14.4035 11.923 14.7679 11.5585 15.2017 11.5585C15.6356 11.5585 16 11.923 16 12.3568V16.3655C16 16.7993 15.6529 17.1637 15.2017 17.1637Z' fill='%23249EDC'/%3E%3Cpath d='M7.94793 12.9642C7.84381 12.9642 7.73969 12.9468 7.63557 12.8947C7.34056 12.7733 7.14967 12.4783 7.14967 12.1485L7.18437 .938127C7.18437 .504287 7.5488 .139862 7.98264 .139862C8.41648 .139862 8.7809 .504287 8.7809 .938127L8.7462 10.257L12.8416 6.33509C13.154 6.02273 13.6746 6.04008 13.9696 6.35244C14.282 6.66481 14.2646 7.18542 13.9523 7.48043L8.50325 12.7386C8.36442 12.8774 8.15617 12.9642 7.94793 12.9642Z' fill='%23249EDC'/%3E%3Cpath d='M7.94793 12.9642C7.73969 12.9642 7.54881 12.8947 7.39262 12.7386L2.03037 7.53249C1.718 7.22012 1.70065 6.71687 2.01301 6.40451C2.32538 6.09214 2.82863 6.07479 3.141 6.38715L8.50325 11.5932C8.81562 11.9056 8.83297 12.4088 8.52061 12.7212C8.36442 12.8774 8.15617 12.9642 7.94793 12.9642Z' fill='%23249EDC'/%3E%3C/svg%3E%0A\")}.snowflake-button-container.is-expand::before{background-image:url(\"data:image/svg+xml,%3Csvg width='18' height='18' viewBox='0 0 18 18' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M6.64375 10.9125C6.9375 11.2062 6.93125 11.6812 6.64375 11.9687L2.57502 16H3.79375C4.20625 16 4.54376 16.3375 4.54376 16.75C4.54376 17.1625 4.20625 17.5 3.79375 17.5H.756264C.556264 17.5 .36876 17.4187 .22501 17.2812C.22501 17.2812 .206248 17.25 .193748 17.2375C.143748 17.1812 .100004 17.1125 .0625038 17.0437C.0375038 16.9687 .0187492 16.8937 .0187492 16.8187C.0187492 16.8 .0062561 16.7813 .0062561 16.7625V13.725C.0187561 13.3125 .356257 12.9875 .768757 12.9937C1.16876 13 1.48752 13.325 1.50002 13.725V14.9688L5.5875 10.9187C5.88125 10.6312 6.35 10.6312 6.64375 10.9187V10.9125ZM17.5063 .743732C17.5063 .543732 17.425 .356235 17.2875 .218735C17.2875 .218735 17.2562 .199998 17.2437 .193748C17.1875 .137498 17.1188 .0937347 17.0438 .0624847C16.9688 .0374847 16.8938 .0187492 16.8188 .0187492C16.8 .0187492 16.7813 .00623703 16.7625 .00623703H13.725C13.3125 .00623703 12.975 .343745 12.975 .756245C12.975 1.16874 13.3125 1.50623 13.725 1.50623H14.9688L11.1312 5.37498C10.8437 5.67498 10.8563 6.14999 11.1563 6.43124C11.45 6.71249 11.9063 6.70624 12.1938 6.43124L16.0125 2.575V3.79375C16.0125 4.20625 16.35 4.54372 16.7625 4.54372C17.175 4.54372 17.5125 4.20625 17.5125 3.79375V.756245L17.5063 .743732ZM16.7562 12.9688C16.3437 12.9688 16.0063 13.3063 16.0063 13.7188V14.8937L12.1938 10.925C11.9063 10.625 11.4375 10.6188 11.1375 10.9063C10.8375 11.1938 10.8313 11.6625 11.1188 11.9625L15 16.0062H13.7188C13.3063 16.0062 12.9688 16.3437 12.9688 16.7562C12.9688 17.1687 13.3063 17.5063 13.7188 17.5063H16.7562C16.85 17.5063 16.95 17.4875 17.0375 17.45C17.0875 17.425 17.1313 17.3937 17.175 17.3625C17.2063 17.3437 17.2438 17.325 17.275 17.3C17.3313 17.2375 17.375 17.1687 17.4125 17.1C17.4188 17.0875 17.4375 17.075 17.4438 17.0562C17.45 17.025 17.4563 16.9938 17.4625 16.9625C17.4813 16.9 17.5 16.8375 17.5 16.7687V13.725C17.5 13.3125 17.1687 12.975 16.7562 12.975V12.9688ZM.750008 4.53125C1.16251 4.53125 1.50002 4.19374 1.50002 3.78124V2.5L5.59376 6.43124C5.89376 6.71874 6.36251 6.70626 6.65001 6.41251C6.93751 6.11876 6.92501 5.64375 6.63126 5.35625L2.61251 1.49998H3.7875C4.2 1.49998 4.53751 1.16249 4.53751 .749989C4.53751 .337489 4.2 0 3.7875 0H.743752C.668752 0 .600004 .0187355 .531254 .0437355C.506254 .0499855 .481263 .0437477 .462513 .0562477C.443763 .0687477 .425015 .0812462 .406265 .0937462C.337515 .124996 .275004 .168741 .218754 .224991H.212498C.212498 .224991 .175 .28125 .15625 .3125C.11875 .3625 .0812477 .4125 .0562477 .46875C.0374977 .525 .0249992 .587499 .0187492 .643749C.0124992 .674999 0 .712482 0 .743732V3.78124C0 4.19374 .337508 4.53125 .750008 4.53125Z' fill='white'/%3E%3C/svg%3E%0A\")}@keyframes slow-scroll{100%{transform:translateY(-50%)}}.sc-hero{overflow:hidden;background-color:#212d35;background-repeat:repeat-y;background-image:url(\"data:image/svg+xml,%3Csvg width='389' height='17' viewBox='0 0 389 17' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M.638672 7.80824L.638672 9.2566C.638672 9.52364 .85538 9.74024 1.12262 9.74024H2.57204C2.83928 9.74024 3.05598 9.52364 3.05598 9.2566V7.80824C3.05598 7.54119 2.83928 7.32472 2.57204 7.32472L1.12262 7.32472C.85538 7.32472 .638672 7.54119 .638672 7.80824Z' fill='url(%23paint0_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M10.9639 7.80824V9.2566C10.9639 9.52364 11.1806 9.74024 11.4478 9.74024L12.8972 9.74024C13.1645 9.74024 13.3812 9.52364 13.3812 9.2566V7.80824C13.3812 7.54119 13.1645 7.32471 12.8972 7.32471L11.4478 7.32471C11.1806 7.32471 10.9639 7.54119 10.9639 7.80824Z' fill='url(%23paint1_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M21.2891 7.80823V9.2566C21.2891 9.52364 21.5058 9.74024 21.773 9.74024L23.2224 9.74024C23.4897 9.74024 23.7064 9.52364 23.7064 9.2566V7.80823C23.7064 7.54119 23.4897 7.32471 23.2224 7.32471L21.773 7.32471C21.5058 7.32471 21.2891 7.54119 21.2891 7.80823Z' fill='url(%23paint2_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M31.6143 7.80823V9.2566C31.6143 9.52364 31.831 9.74024 32.0982 9.74024H33.5476C33.8149 9.74024 34.0316 9.52364 34.0316 9.2566V7.80823C34.0316 7.54119 33.8149 7.32471 33.5476 7.32471L32.0982 7.32471C31.831 7.32471 31.6143 7.54119 31.6143 7.80823Z' fill='url(%23paint3_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M41.9395 7.80823V9.2566C41.9395 9.52364 42.1562 9.74024 42.4234 9.74024H43.8728C44.1401 9.74024 44.3568 9.52364 44.3568 9.2566V7.80823C44.3568 7.54119 44.1401 7.32471 43.8728 7.32471L42.4234 7.32471C42.1562 7.32471 41.9395 7.54119 41.9395 7.80823Z' fill='url(%23paint4_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M52.5076 7.80823V9.2566C52.5076 9.52364 52.7243 9.74024 52.9916 9.74024H54.441C54.7082 9.74024 54.9249 9.52364 54.9249 9.2566V7.80823C54.9249 7.54119 54.7082 7.32471 54.441 7.32471L52.9916 7.32471C52.7243 7.32471 52.5076 7.54119 52.5076 7.80823Z' fill='url(%23paint5_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M62.8331 7.80823V9.2566C62.8331 9.52364 63.0493 9.74024 63.3165 9.74024H64.7664C65.0332 9.74024 65.2504 9.52364 65.2504 9.2566V7.80823C65.2504 7.54119 65.0332 7.32471 64.7664 7.32471L63.3165 7.32471C63.0493 7.32471 62.8331 7.54119 62.8331 7.80823Z' fill='url(%23paint6_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M73.1583 7.80823V9.2566C73.1583 9.52364 73.3745 9.74024 73.6417 9.74024H75.0916C75.3584 9.74024 75.5756 9.52364 75.5756 9.2566V7.80823C75.5756 7.54119 75.3584 7.32471 75.0916 7.32471L73.6417 7.32471C73.3745 7.32471 73.1583 7.54119 73.1583 7.80823Z' fill='url(%23paint7_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M83.4835 7.80823V9.2566C83.4835 9.52364 83.6997 9.74024 83.9669 9.74024H85.4168C85.6836 9.74024 85.9008 9.52364 85.9008 9.2566V7.80823C85.9008 7.54119 85.6836 7.32471 85.4168 7.32471L83.9669 7.32471C83.6997 7.32471 83.4835 7.54119 83.4835 7.80823Z' fill='url(%23paint8_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M93.8087 7.80823V9.2566C93.8087 9.52364 94.0249 9.74024 94.2921 9.74024H95.742C96.0088 9.74024 96.226 9.52364 96.226 9.2566V7.80823C96.226 7.54119 96.0088 7.32471 95.742 7.32471L94.2921 7.32471C94.0249 7.32471 93.8087 7.54119 93.8087 7.80823Z' fill='url(%23paint9_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M104.134 7.80823V9.2566C104.134 9.52364 104.35 9.74024 104.617 9.74024H106.067C106.334 9.74024 106.551 9.52364 106.551 9.2566V7.80823C106.551 7.54119 106.334 7.32471 106.067 7.32471L104.617 7.32471C104.35 7.32471 104.134 7.54119 104.134 7.80823Z' fill='url(%23paint10_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M114.702 7.80823V9.2566C114.702 9.52364 114.918 9.74024 115.185 9.74024L116.635 9.74024C116.902 9.74024 117.119 9.52364 117.119 9.25659V7.80823C117.119 7.54119 116.902 7.32471 116.635 7.32471L115.185 7.32471C114.918 7.32471 114.702 7.54119 114.702 7.80823Z' fill='url(%23paint11_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M125.027 7.80823V9.25659C125.027 9.52364 125.243 9.74024 125.511 9.74024L126.961 9.74024C127.227 9.74024 127.445 9.52364 127.445 9.25659V7.80823C127.445 7.54119 127.227 7.32471 126.961 7.32471L125.511 7.32471C125.243 7.32471 125.027 7.54119 125.027 7.80823Z' fill='url(%23paint12_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M135.352 7.80823V9.25659C135.352 9.52364 135.569 9.74024 135.836 9.74024H137.286C137.553 9.74024 137.77 9.52364 137.77 9.25659V7.80823C137.77 7.54119 137.553 7.32471 137.286 7.32471L135.836 7.32471C135.569 7.32471 135.352 7.54119 135.352 7.80823Z' fill='url(%23paint13_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M145.678 7.80823V9.25659C145.678 9.52364 145.894 9.74024 146.161 9.74024H147.611C147.878 9.74024 148.095 9.52364 148.095 9.25659V7.80823C148.095 7.54119 147.878 7.32471 147.611 7.32471L146.161 7.32471C145.894 7.32471 145.678 7.54119 145.678 7.80823Z' fill='url(%23paint14_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M156.003 7.80823V9.25659C156.003 9.52364 156.219 9.74024 156.486 9.74024H157.936C158.203 9.74024 158.42 9.52364 158.42 9.25659V7.80823C158.42 7.54119 158.203 7.32471 157.936 7.32471L156.486 7.32471C156.219 7.32471 156.003 7.54119 156.003 7.80823Z' fill='url(%23paint15_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M166.328 7.80823V9.25659C166.328 9.52363 166.544 9.74024 166.811 9.74024H168.261C168.528 9.74024 168.745 9.52363 168.745 9.25659V7.80823C168.745 7.54119 168.528 7.32471 168.261 7.32471L166.811 7.32471C166.544 7.32471 166.328 7.54119 166.328 7.80823Z' fill='url(%23paint16_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M176.896 7.80823V9.25659C176.896 9.52363 177.112 9.74023 177.38 9.74023H178.83C179.096 9.74023 179.313 9.52363 179.313 9.25659V7.80823C179.313 7.54119 179.096 7.32471 178.83 7.32471L177.38 7.32471C177.112 7.32471 176.896 7.54119 176.896 7.80823Z' fill='url(%23paint17_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M187.221 7.80823V9.25659C187.221 9.52363 187.438 9.74023 187.705 9.74023H189.155C189.421 9.74023 189.639 9.52363 189.639 9.25659V7.80823C189.639 7.54119 189.421 7.32471 189.155 7.32471L187.705 7.32471C187.438 7.32471 187.221 7.54119 187.221 7.80823Z' fill='url(%23paint18_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M199.639 7.80824V9.2566C199.639 9.52364 199.855 9.74024 200.123 9.74024H201.572C201.839 9.74024 202.056 9.52364 202.056 9.2566V7.80824C202.056 7.54119 201.839 7.32472 201.572 7.32472L200.123 7.32472C199.855 7.32472 199.639 7.54119 199.639 7.80824Z' fill='url(%23paint19_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M209.964 7.80824V9.2566C209.964 9.52364 210.181 9.74024 210.448 9.74024L211.897 9.74024C212.164 9.74024 212.381 9.52364 212.381 9.2566V7.80824C212.381 7.54119 212.164 7.32471 211.897 7.32471L210.448 7.32471C210.181 7.32471 209.964 7.54119 209.964 7.80824Z' fill='url(%23paint20_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M220.289 7.80823V9.2566C220.289 9.52364 220.506 9.74024 220.773 9.74024L222.222 9.74024C222.49 9.74024 222.706 9.52364 222.706 9.2566V7.80823C222.706 7.54119 222.49 7.32471 222.222 7.32471L220.773 7.32471C220.506 7.32471 220.289 7.54119 220.289 7.80823Z' fill='url(%23paint21_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M230.614 7.80823V9.2566C230.614 9.52364 230.831 9.74024 231.098 9.74024H232.548C232.815 9.74024 233.032 9.52364 233.032 9.2566V7.80823C233.032 7.54119 232.815 7.32471 232.548 7.32471L231.098 7.32471C230.831 7.32471 230.614 7.54119 230.614 7.80823Z' fill='url(%23paint22_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M240.939 7.80823V9.2566C240.939 9.52364 241.156 9.74024 241.423 9.74024H242.873C243.14 9.74024 243.357 9.52364 243.357 9.2566V7.80823C243.357 7.54119 243.14 7.32471 242.873 7.32471L241.423 7.32471C241.156 7.32471 240.939 7.54119 240.939 7.80823Z' fill='url(%23paint23_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M251.508 7.80823V9.2566C251.508 9.52364 251.724 9.74024 251.992 9.74024H253.441C253.708 9.74024 253.925 9.52364 253.925 9.2566V7.80823C253.925 7.54119 253.708 7.32471 253.441 7.32471L251.992 7.32471C251.724 7.32471 251.508 7.54119 251.508 7.80823Z' fill='url(%23paint24_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M261.833 7.80823V9.2566C261.833 9.52364 262.049 9.74024 262.317 9.74024H263.766C264.033 9.74024 264.25 9.52364 264.25 9.2566V7.80823C264.25 7.54119 264.033 7.32471 263.766 7.32471L262.317 7.32471C262.049 7.32471 261.833 7.54119 261.833 7.80823Z' fill='url(%23paint25_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M272.158 7.80823V9.2566C272.158 9.52364 272.374 9.74024 272.642 9.74024H274.092C274.358 9.74024 274.576 9.52364 274.576 9.2566L274.576 7.80823C274.576 7.54119 274.358 7.32471 274.092 7.32471L272.642 7.32471C272.374 7.32471 272.158 7.54119 272.158 7.80823Z' fill='url(%23paint26_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M282.483 7.80823V9.2566C282.483 9.52364 282.7 9.74024 282.967 9.74024H284.417C284.684 9.74024 284.901 9.52364 284.901 9.2566V7.80823C284.901 7.54119 284.684 7.32471 284.417 7.32471L282.967 7.32471C282.7 7.32471 282.483 7.54119 282.483 7.80823Z' fill='url(%23paint27_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M292.809 7.80823L292.809 9.2566C292.809 9.52364 293.025 9.74024 293.292 9.74024H294.742C295.009 9.74024 295.226 9.52364 295.226 9.2566V7.80823C295.226 7.54119 295.009 7.32471 294.742 7.32471L293.292 7.32471C293.025 7.32471 292.809 7.54119 292.809 7.80823Z' fill='url(%23paint28_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M303.134 7.80823L303.134 9.2566C303.134 9.52364 303.35 9.74024 303.617 9.74024H305.067C305.334 9.74024 305.551 9.52364 305.551 9.2566V7.80823C305.551 7.54119 305.334 7.32471 305.067 7.32471L303.617 7.32471C303.35 7.32471 303.134 7.54119 303.134 7.80823Z' fill='url(%23paint29_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M313.702 7.80823L313.702 9.2566C313.702 9.52364 313.918 9.74024 314.185 9.74024L315.635 9.74024C315.902 9.74024 316.119 9.52364 316.119 9.25659V7.80823C316.119 7.54119 315.902 7.32471 315.635 7.32471L314.185 7.32471C313.918 7.32471 313.702 7.54119 313.702 7.80823Z' fill='url(%23paint30_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M324.027 7.80823V9.25659C324.027 9.52364 324.243 9.74024 324.511 9.74024L325.961 9.74024C326.227 9.74024 326.445 9.52364 326.445 9.25659V7.80823C326.445 7.54119 326.227 7.32471 325.961 7.32471L324.511 7.32471C324.243 7.32471 324.027 7.54119 324.027 7.80823Z' fill='url(%23paint31_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M334.352 7.80823V9.25659C334.352 9.52364 334.569 9.74024 334.836 9.74024H336.286C336.553 9.74024 336.77 9.52364 336.77 9.25659L336.77 7.80823C336.77 7.54119 336.553 7.32471 336.286 7.32471L334.836 7.32471C334.569 7.32471 334.352 7.54119 334.352 7.80823Z' fill='url(%23paint32_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M344.678 7.80823V9.25659C344.678 9.52364 344.894 9.74024 345.161 9.74024H346.611C346.878 9.74024 347.095 9.52364 347.095 9.25659L347.095 7.80823C347.095 7.54119 346.878 7.32471 346.611 7.32471L345.161 7.32471C344.894 7.32471 344.678 7.54119 344.678 7.80823Z' fill='url(%23paint33_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M355.003 7.80823V9.25659C355.003 9.52364 355.219 9.74024 355.486 9.74024H356.936C357.203 9.74024 357.42 9.52364 357.42 9.25659L357.42 7.80823C357.42 7.54119 357.203 7.32471 356.936 7.32471L355.486 7.32471C355.219 7.32471 355.003 7.54119 355.003 7.80823Z' fill='url(%23paint34_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M365.328 7.80823V9.25659C365.328 9.52363 365.544 9.74024 365.811 9.74024H367.261C367.528 9.74024 367.745 9.52363 367.745 9.25659V7.80823C367.745 7.54119 367.528 7.32471 367.261 7.32471L365.811 7.32471C365.544 7.32471 365.328 7.54119 365.328 7.80823Z' fill='url(%23paint35_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M375.896 7.80823V9.25659C375.896 9.52363 376.112 9.74023 376.38 9.74023H377.83C378.096 9.74023 378.313 9.52363 378.313 9.25659V7.80823C378.313 7.54119 378.096 7.32471 377.829 7.32471L376.38 7.32471C376.112 7.32471 375.896 7.54119 375.896 7.80823Z' fill='url(%23paint36_linear_8295_70635)'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M386.221 7.80823V9.25659C386.221 9.52363 386.438 9.74023 386.705 9.74023H388.155C388.421 9.74023 388.639 9.52363 388.639 9.25659V7.80823C388.639 7.54119 388.421 7.32471 388.155 7.32471L386.705 7.32471C386.438 7.32471 386.221 7.54119 386.221 7.80823Z' fill='url(%23paint37_linear_8295_70635)'/%3E%3Cdefs%3E%3ClinearGradient id='paint0_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint1_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint2_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint3_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint4_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint5_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint6_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint7_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint8_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint9_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint10_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint11_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint12_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint13_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint14_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint15_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint16_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint17_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint18_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint19_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint20_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint21_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint22_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint23_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint24_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint25_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint26_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint27_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint28_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint29_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint30_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint31_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint32_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint33_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint34_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint35_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint36_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3ClinearGradient id='paint37_linear_8295_70635' x1='-47.5' y1='8.99989' x2='332' y2='8.99989' gradientUnits='userSpaceOnUse'%3E%3Cstop stop-color='%2329B5E8' stop-opacity='.8'/%3E%3Cstop offset='1' stop-color='%2329B5E8' stop-opacity='0'/%3E%3C/linearGradient%3E%3C/defs%3E%3C/svg%3E%0A\")}.sc-hero__inner\u003E.snowflake-flexible-column-container-items\u003Ediv:first-child{position:relative;z-index:3}.sc-hero__inner\u003E.snowflake-flexible-column-container-items\u003Ediv:last-child{position:absolute;height:100%;width:100%;top:0;left:-24px}.sc-hero__inner\u003E.snowflake-flexible-column-container-items\u003Ediv:last-child::before{content:\"\";display:block;z-index:1;position:absolute;top:-64px;left:0;width:150%;height:calc(100% + 160px);background-color:rgb(32 44 53 / .9)}.sc-body__content .heading-3-v2,.sc-hero__headline .heading-1-v2{text-transform:none}.sc-body__content span.snowflake-image-caption{display:block!important;font-style:italic}.sc-body__content .snowflake-text p+ul{margin-top:24px!important;padding-left:16px!important}.white-blue-text-color .snowflake-title-v2.solution-center-hero__certification .snowflake-typographyv2\u003Espan.snowflake-title-v2-line{color:#e9eaeb!important;font-size:16px}.white-blue-text-color .snowflake-title-v2.solution-center-hero__certification.is-large .snowflake-typographyv2\u003Espan.snowflake-title-v2-line{color:#fff!important;font-size:18px}.solution-center-hero__certification\u003E.snowflake-title-v2-line\u003Espan:first-child{display:flex;justify-content:flex-start;align-items:center;gap:8px}.solution-center-hero__certification\u003E.snowflake-title-v2-line\u003Espan:first-child::before{content:\"\";display:inline-block;width:16px;height:16px;background-image:url(\"data:image/svg+xml,%3Csvg width='16' height='16' viewBox='0 0 16 16' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M8 0C3.58146 0 0 3.58146 0 8C0 12.4185 3.58146 16 8 16C12.4185 16 16 12.4185 16 8C16 3.58146 12.4185 0 8 0ZM12.7184 5.91984L7.33471 11.3026C7.31293 11.3244 7.31293 11.3454 7.29198 11.3454L7.20653 11.4308C6.94933 11.688 6.54132 11.7525 6.21962 11.6235C6.11238 11.5808 6.00514 11.5163 5.9197 11.4308L5.83425 11.3454C5.83425 11.3454 5.83425 11.3236 5.81246 11.3236L3.28149 8.79347C2.93799 8.44997 2.93799 7.87107 3.28149 7.50664L3.36694 7.42119C3.71044 7.07769 4.28934 7.07769 4.65377 7.42119L6.58401 9.35143L11.3877 4.5477C11.7312 4.2042 12.3101 4.2042 12.6746 4.5477L12.76 4.63315C13.0826 4.99758 13.0828 5.55541 12.7184 5.91984Z' fill='%230E8A16'/%3E%3C/svg%3E%0A\");background-size:contain;background-repeat:no-repeat;background-color:#fff;border-radius:100%}.sc-hero__byline{padding-top:8px}.sc-hero__byline p{color:#e2e2e2;margin-top:0!important}.sc-hero pre[class*=language-]{overflow:visible}.snowflake-code-snippet,.snowflake-code-snippet code,.snowflake-code-snippet pre{font-size:16px}.sc-hero__code-snippet:not(pre)\u003Ecode[class*=language-],.sc-hero__code-snippet pre[class*=language-]{background:0 0}.sc-hero__code-snippet{opacity:.8;background-color:transparent!important;position:absolute;top:0;right:0;width:100%;animation:240s linear 1s forwards slow-scroll}.sc-hero__button-container .snowflake-flexible-column-container-items{padding:0 0 24px;margin-top:-8px;margin-left:24px}.sc-sidebar__partner-logo{width:100%;max-width:140px;margin-top:8px}.sc-sidebar__partner-logo .cmp-image__image{border-radius:0}.sc-tag-cluster.snowflake-text ul{list-style-type:none;padding:0;display:flex;flex-wrap:wrap;gap:8px;margin:0}.sc-tag-cluster.snowflake-text li{color:#373f41;border-radius:4px;display:inline-block;padding:6px;text-transform:uppercase;letter-spacing:1px;font-size:12px!important;line-height:12px!important;margin:0!important;background-color:#f3f3f3}.sc-body .share-icon svg{height:24px;cursor:pointer}.sc-body .share-icon svg:hover path{fill:var(--ui-02)}.sc-overview__webinar-promo-banner{align-items:center;border:1px solid #ccc;padding:var(--spacing-02)}.sc-overview__webinar-promo-banner .snowflake-content-chip-image{max-width:32px;margin-right:var(--spacing-02);line-height:0}.sc-overview__webinar-promo-banner .snowflake-content-chip-image__image,.summit-speaker-card .snowflake-card-v2-advanced-image__image{aspect-ratio:1}.sc-overview__webinar-promo-banner .snowflake-content-chip-content .heading-5-v2{font-size:14px;font-family:Lato,sans-serif}.sc-overview__webinar-promo-banner .snowflake-content-chip-content .snowflake-title-v2-line:not(:first-child){font-weight:400}.sc-overview__webinar-promo-banner .snowflake-content-chip-button .snowflake-button-container{font-size:14px!important}.diagram-group__button{position:absolute;bottom:24px;right:24px;background-color:#212c35!important}.section--mountains-bottom,.summit-hp-hero{position:relative}.sc-cert-banner{background-color:#212d35;border-radius:8px;padding:24px;overflow:hidden}.sc-cert-banner\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;flex-direction:row;align-items:center}:root{--text-secondary:#706f6f;--summit-bg-ltblue:#eaf8fd;--summit-bg-blue:#249edc;--summit-border:#d2d1d4;--summit-border-radius:8px;--summit-card-padding:32px;--summit-card-padding-sm:28px}.section--mountains-bottom::after,.section--mountains-bottom::before{content:\"\";display:block;position:absolute;bottom:-1px;max-width:400px;background-size:100% auto;height:100%;width:30%;line-height:0;background-repeat:no-repeat}.button-group\u003E.container\u003E.cmp-container\u003E.aem-container{justify-content:center;align-items:center}.button-group\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:auto!important;margin:0 8px!important}.button-group .snowflake-button-container{font-family:Texta,sans-serif}.section--summit-bg-ltblue{background-color:var(--summit-bg-ltblue)}.section--summit-bg-blue,.summit-hero-secondary{background-color:var(--summit-bg-blue)}.section--mountains-bottom::before{left:0;background-image:url(\"data:image/svg+xml,%3Csvg width='402' height='309' viewBox='0 0 402 309' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M401.523 308.761H0V0L181.63 182.431L228.479 135.531L401.523 308.761Z' fill='%23249EDC'/%3E%3C/svg%3E%0A\");background-position:bottom left}.section--mountains-bottom::after{right:0;background-image:url(\"data:image/svg+xml,%3Csvg width='402' height='309' viewBox='0 0 402 309' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M0 308.761H401.523V0L219.893 182.431L173.044 135.531L0 308.761Z' fill='%23249EDC'/%3E%3C/svg%3E%0A\");background-position:bottom right}.summit-hp-hero{overflow:hidden}.summit-hero__bg-video{position:absolute;top:50%;left:50%;width:120%;height:100%;opacity:.3;transform:translate(-50%,-50%)}.summit-hero__bg-svg,.summit-prefooter__bg-image,.summit-secondary-hero__bg-image{position:absolute;bottom:0;left:0;width:100%}.summit-hp-promo-banner__headline .heading-4-v2{font-weight:900}.summit-hero-secondary .hero-lottie__left{position:absolute;bottom:0;left:0;width:30%;line-height:0}.summit-timeline__card::after,.summit-timeline__card::before{bottom:0;left:50%;position:absolute;display:block;background-color:var(--ui-01);content:\"\"}.summit-hero-secondary .snowflake-text p{font-size:24px!important;line-height:32px!important;max-width:720px;margin:0 auto}.summit-stat-container\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;justify-content:center}.summit-stat-container\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:auto!important;max-width:25%}.summit-stat-container\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:not(:last-child){border-right:1px solid #fff}.summit-timeline__card{border:1px solid var(--summit-border);border-radius:var(--summit-border-radius);padding:var(--summit-card-padding);position:relative;background-color:#fff}.summit-timeline__card::before{width:20px;height:20px;border-radius:100%;transform:translate(-50%,50%)}.summit-timeline__card::after{width:3px;height:50px;transform:translate(-50%,100%)}.summit-timeline-card__icon{width:48px;height:48px}.summit-timeline-card__headline .heading-3-v2{font-size:32px}.faq-group{border:1px solid var(--ui-12);border-radius:4px;background-color:#fff}.faq-group__question{padding:24px}.faq-group__question:hover{color:var(--ui-01);cursor:pointer}.faq-group__question .heading-4-v2,.faq-group__question .heading-5-v2{position:relative;padding-right:64px}.faq-group__question .heading-4-v2::after,.faq-group__question .heading-5-v2::after{content:\"\";display:block;width:32px;height:32px;background-image:url(\"data:image/svg+xml,%3Csvg width='29' height='16' viewBox='0 0 29 16' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M14.16 14.6807C14.2537 14.7957 14.3719 14.8884 14.506 14.952C14.64 15.0157 14.7866 15.0487 14.935 15.0487C15.0834 15.0487 15.2299 15.0157 15.3639 14.952C15.498 14.8884 15.6162 14.7957 15.71 14.6807V14.6807L28.51 2.00068C29.07 1.43068 29.07 .92068 28.51 .44068C27.95 -.0393204 27.43 -.11932 26.96 .44068L14.94 12.0007L2.99996 .45068C2.90725 .322624 2.7855 .218374 2.6447 .146483C2.50389 .0745926 2.34805 .0371094 2.18996 .0371094C2.03187 .0371094 1.87603 .0745926 1.73522 .146483C1.59442 .218374 1.47267 .322624 1.37996 .45068C.819961 .93068 .819961 1.45068 1.37996 2.01068L14.16 14.6807Z' fill='black'/%3E%3C/svg%3E%0A\");background-size:80% auto;background-repeat:no-repeat;background-position:center;position:absolute;top:-2px;right:0;transition:.3s 150ms}.faq-group__question .heading-5-v2::after{top:-4px}.faq-group__answer{max-height:0;overflow:hidden;width:95%;padding:0 24px;transition:.5s}.faq-group__answer\u003Espan{display:block;padding-bottom:24px}.is-open .faq-group__answer{max-height:600px;transition:1s}.is-open .faq-group__question .heading-4-v2::after,.is-open .faq-group__question .heading-5-v2::after{transform:rotate(180deg);transition:.3s}.summit-agenda{box-shadow:2px 4px 10px 0 rgb(156 156 156 / .52);border-radius:8px;background-color:#fff;max-width:980px;margin-left:auto;margin-right:auto;padding:40px;width:90%}.agenda-item{border-radius:8px;background-color:#d4f0fa;padding:16px;border-left:4px solid var(--ui-01);position:relative}.summit-pricing-block__tile.is-past,.summit-pricing-block__tile.is-upcoming{pointer-events:none;border-color:#d2d1d4}p.agenda-item__time{width:25%;font-family:Texta!important;font-size:32px!important;font-weight:900!important;text-transform:uppercase!important;max-width:140px}@media screen and (max-width:991px){#partnerResources .section--resource-hub .snowflake-button-link .snowflake-button-container{font-size:14px!important;line-height:20px!important;margin-top:4px}#industryPartnerSlider\u003E.snowflake-flexible-column-container-items{display:flex;flex-direction:column}#industryPartnerSlider\u003E.snowflake-flexible-column-container-items\u003Ediv{width:100%}.sc-cert-banner__left{text-align:center}.sc-cert-banner__left .solution-center-hero__certification .snowflake-title-v2-line{justify-content:center}.summit-hero__bg-video{width:200%}.summit-leadership-grid .snowflake-flexible-column-container-items{grid-template-columns:repeat(2,1fr)}.summit-stat-container\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:50%!important;max-width:50%!important}.summit-stat-container\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:not(:last-child){border-right:none!important}.summit-agenda{padding:24px}p.agenda-item__time{font-size:24px!important;width:auto;white-space:nowrap;padding-right:24px}}.agenda-item\u003Espan{display:flex;align-items:center}.summit-add-on-block,.summit-pricing-block{border:1px solid #d2d1d4;border-radius:8px;overflow:hidden;box-shadow:2px 4px 10px 0 rgb(156 156 156 / .52);background-color:#fff}.summit-add-on-block__content,.summit-pricing-block__content{padding:0 20px 20px}.summit-pricing-block__tile{padding:24px 20px;border-radius:4px;background:#fff;border:1px solid var(--ui-01);position:relative;transition:background-color .3s}.summit-pricing-block__tile:hover{background-color:var(--ui-01);transition:background-color .3s}.summit-pricing-block__tile.is-past{background-color:#d4f0fa}.summit-pricing-block__tile:hover .black-blue-text-color .snowflake-title-v2-line{color:#fff!important;transition:color .3s}.partner-card__logo-grid\u003E.container\u003E.cmp-container\u003E.aem-container::after,.partner-card__logo-grid\u003E.container\u003E.cmp-container\u003E.aem-container::before,.summit-add-on-block__content\u003E.container\u003E.cmp-container\u003E.aem-container::after,.summit-add-on-block__content\u003E.container\u003E.cmp-container\u003E.aem-container::before,.summit-pricing-block__tile.is-past .snowflake-content-chip-button,.summit-pricing-block__tile.is-upcoming .snowflake-content-chip-button,.summit-speaker-card .snowflake-card-v2-advanced-tag-indicator{display:none}.summit-pricing-block__tile.is-past .black-blue-text-color .snowflake-title-v2-line{color:#7cc7eb!important}.summit-pricing-block__tile.is-upcoming .black-blue-text-color .snowflake-title-v2-line{color:#8c8c8c!important}.summit-pricing-block__aside{background-color:#d4f0fa;border:1px solid #d2d1d4;border-radius:8px;padding:24px;width:100%}.summit-pricing-block__aside li::marker{color:var(--ui-01)}.summit-pricing-block__aside-headline .heading-5-v2{font-weight:900;margin-bottom:12px}.summit-pricing-block__header{background:#000;padding:24px 40px}.summit-pricing-block__header .heading-4-v2{font-weight:900;letter-spacing:.5px}.bwwidth100,.snowflake-mega-nav-dropdown-footer-content,.summit-pricing-block__tile .black-blue-text-color{width:100%}.summit-pricing-block__tile .heading-5-v2{position:static}.summit-pricing-block__tile .heading-5-v2 span.snowflake-title-v2-line:first-child{text-transform:uppercase;font-weight:900!important;letter-spacing:.25px;font-size:24px!important}.summit-pricing-block__tile .heading-5-v2 span.snowflake-title-v2-line:nth-child(2){margin-top:8px;font-family:Lato,sans-serif;font-size:14px;font-style:normal;font-weight:400;line-height:16px}.summit-pricing-block__tile .heading-5-v2 span.snowflake-title-v2-line:last-child{font-weight:900!important;font-size:40px!important}.snowflake-mega-nav-nav-item\u003Ea:hover .snowflake-mega-nav-nav-item-title-wrapper\u003E.snowflake-mega-nav-nav-item-title,.summit-pricing-block__tile:not(.is-upcoming):not(.is-past) .heading-5-v2 span.snowflake-title-v2-line:last-child{color:var(--ui-01)!important}.summit-pricing-block__tile:hover:not(.is-upcoming):not(.is-past) .heading-5-v2 span.snowflake-title-v2-line:last-child{color:#fff!important}.summit-pricing-block__tile.is-past .heading-5-v2 span.snowflake-title-v2-line:last-child{text-decoration:line-through}.summit-pricing-block__tile .snowflake-content-chip-button{margin-top:0;white-space:nowrap;display:none}.snowflake-card-v2-advanced.no-link{pointer-events:none!important}.snowpro-card{border:1px solid var(--summit-border);border-radius:var(--summit-border-radius);padding:var(--summit-card-padding-sm);display:flex;height:100%}.snowpro-card__headline{margin:24px 0 12px}.snowpro-card__pricing{margin-top:48px}.snowpro-card .snowflake-text .snowpro-card__price{color:var(--ui-01);font-weight:900;font-size:40px!important;font-family:Texta,sans-serif}.summit-stat-container\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;flex-direction:row;flex-wrap:wrap}.summit-stat-container\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:not(:last-child){border-right:1px solid var(--summit-border)}.summit-stat-card{padding:0 40px}.summit-stat .heading-2-v2 .snowflake-title-v2-line:first-child{font-size:64px;line-height:52px;margin-bottom:8px}.summit-stat .heading-2-v2 .snowflake-title-v2-line:last-child{font-size:32px;line-height:30px;margin-bottom:16px}.summit-speaker-card .snowflake-card-v2-advanced-title{margin-bottom:var(--spacing-01)}.summit-add-on-card{padding:24px;border:1px solid #d2d1d4;border-radius:8px}.summit-add-on__subhead{padding-left:40px;padding-right:40px}.partner-card__logo-grid,.partner-card__logo-single{padding:40px}.partner-card__logo-grid .snowflake-image-container .cmp-image__image,.partner-card__logo-single .snowflake-image-container .cmp-image__image{border-radius:0;max-width:240px;margin:0 auto}.partner-card\u003E.container,.partner-card\u003E.container\u003E.aem-container,.partner-card\u003E.container\u003E.cmp-container{height:100%}.summit-add-on-block__content\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;flex-direction:row;gap:24px;align-items:stretch}.partner-card__logo-grid\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;flex-direction:row;flex-wrap:wrap;gap:40px 24px;justify-content:center;align-items:center}.partner-card__logo-grid\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(33.3333% - 24px);margin:0!important}.partner-card{border-radius:8px;border:1px solid #d2d1d4;overflow:hidden;height:100%;background-color:#fff}.partner-card__header{padding:16px 24px;border-bottom:1px solid #d2d1d4}.partner-card__header.is-purple{background-color:#7d44cf}.partner-card__header h4{display:flex;flex-direction:row!important;align-items:center;gap:12px}.partner-card__header h4::before{vertical-align:middle;content:\"\";display:inline-block;width:20px;height:20px;background-size:contain;background-repeat:no-repeat;background-image:url(\"data:image/svg+xml,%3Csvg width='21' height='23' viewBox='0 0 21 23' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M20.0375 12.8374C20.1644 12.439 20.2172 12.0289 20.2077 11.6237C20.193 11.3305 20.1548 11.0373 20.0712 10.7441C19.8196 9.83306 19.223 9.01989 18.3294 8.50724L5.61817 1.2017C3.82388 .173815 1.53618 .784335 .506483 2.56804C-.533615 4.34915 .0797871 6.62351 1.87408 7.65398L8.97715 11.7427L1.87408 15.8201C.0797871 16.8527 -.531016 19.1271 .506483 20.9156C1.53618 22.6941 3.82388 23.302 5.61817 22.2746L18.3294 14.9643C19.1871 14.4728 19.7693 13.7027 20.0375 12.8374Z' fill='black'/%3E%3C/svg%3E%0A\")}.partner-card__header.is-purple h4::before{background-image:url(\"data:image/svg+xml,%3Csvg width='21' height='23' viewBox='0 0 21 23' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M20.0375 12.8374C20.1644 12.439 20.2172 12.0289 20.2077 11.6237C20.193 11.3305 20.1548 11.0373 20.0712 10.7441C19.8196 9.83306 19.223 9.01989 18.3294 8.50724L5.61817 1.2017C3.82388 .173815 1.53618 .784335 .506483 2.56804C-.533615 4.34915 .0797871 6.62351 1.87408 7.65398L8.97715 11.7427L1.87408 15.8201C.0797871 16.8527 -.531016 19.1271 .506483 20.9156C1.53618 22.6941 3.82388 23.302 5.61817 22.2746L18.3294 14.9643C19.1871 14.4728 19.7693 13.7027 20.0375 12.8374Z' fill='white'/%3E%3C/svg%3E%0A\")}.sf-blue-mountains{background-size:90% auto;background-repeat:no-repeat;background-position:center bottom;background-image:url(\"data:image/svg+xml,%3Csvg width='1361' height='410' viewBox='0 0 1361 410' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M1360.25 410L1065.53 114.309L976.256 203.875L773.049 0L364.393 410H1360.25Z' fill='%233AA8DF'/%3E%3Cpath d='M274.778 410L137.467 272.238L.15625 410H274.778Z' fill='%233AA8DF'/%3E%3C/svg%3E%0A\")}.bwalignr,.main-pr-body .bwalignr{text-align:right}.bwblockalignl{margin-left:0;margin-right:auto}.bwcellpmargin{margin-top:0;margin-bottom:0}.bwlistdisc{list-style-type:disc}.bwpadb3{padding-bottom:4px}.bwpadb4{padding-bottom:5px}.bwpadl0{padding-left:0}.bwpadl3{padding-left:15px}.bwpadl6{padding-left:30px}.bwpadl9{padding-left:45px}.bwpadl12{padding-left:60px}.bwpadr0{padding-right:0}.bwtablemarginb{margin-bottom:10px}.bwvertalignb{vertical-align:bottom}.bwvertalignt{vertical-align:top}.bwsinglebottom{border-bottom:1pt solid #000}.bwdoublebottom{border-bottom:2.25pt double #000}.bwwidth1{width:1%}.bwwidth2{width:2%}.bwwidth6{width:6%}.bwwidth7{width:7%}.bwwidth8{width:8%}.bwwidth10{width:10%}.bwwidth12{width:12%}.bwwidth32{width:32%}.bwwidth44{width:44%}.bwwidth72{width:72%}.bwwidth97{width:97%}.main-pr-body{font-size:18px;line-height:26px}.main-pr-body img{display:block;width:100%;height:auto!important;border-radius:var(--small-border-radius)}.main-pr-body table{width:100%;display:block}.main-pr-body tbody{background-color:#f7f7f7}.main-pr-body .bwsinglebottom{border-bottom:1pt solid #000!important}.main-pr-body td.bwwidth44{padding-right:40px}.main-pr-body .bw-release-story{font-family:Lato,sans-serif}.main-pr-body .bw-release-story sup,.snowflake-mega-nav-dropdown-header-content-right a{white-space:nowrap}.main-pr-body .bw-release-story\u003E*,.main-pr-body\u003Espan\u003E*{margin-bottom:2rem!important}.snowflake-text.main-pr-body tbody,.snowflake-text.main-pr-body tbody p{font-size:14px!important;line-height:20px!important;width:100%;display:block}.press-body .snowflake-flexible-column-container-items{gap:var(--spacing-08)}.about-snowflake{border:1px solid #ccc;background-color:var(--ui-background-05);padding:24px;border-radius:8px;margin-top:0}.about-snowflake__logo{max-width:140px;margin-top:16px}.hero--press .snowflake-hero-system-inner{max-width:1408px;margin:0 auto!important}#arcticNavItem{flex-direction:column}#arcticNavItem::before{content:\"Featured Open Source Technologies\";display:block;margin-top:48px;margin-bottom:24px;font-size:16px!important;line-height:16px!important;font-weight:800!important;text-transform:uppercase}@media screen and (min-width:768px){.sc-hero__inner\u003E.snowflake-flexible-column-container-items\u003Ediv:last-child{position:relative;height:100%;top:auto;left:auto;width:auto}.sc-hero__inner\u003E.snowflake-flexible-column-container-items\u003Ediv:last-child::before{background:linear-gradient(180deg,#202c35 -7.5%,#fff0 51.25%,#202c35 107.69%)}.sc-hero__byline\u003Espan{display:flex;flex-wrap:wrap}.sc-hero__byline p:not(:last-child)::after{content:\"|\";margin:0 12px;opacity:.5}.sc-hero__button-container .snowflake-flexible-column-container-items{position:absolute;bottom:0;padding:0;margin:0 24px 0 0}.sc-hero__button-container .hero-watch-the-demo{padding:12px 16px!important;float:right;margin-bottom:48px;background-color:rgb(35 45 54 / .8)}.summit-overview-stat{padding:0 40px}.summit-timeline{border-bottom:3px solid var(--ui-01);margin-bottom:64px}.summit-add-on-block__content,.summit-pricing-block__content{padding:0 40px 40px}#arcticNavItem::before{font-size:12px!important;margin-bottom:8px;margin-top:16px}.snowflake-mega-nav-nav-item-title-wrapper\u003E.snowflake-mega-nav-nav-item-title{line-height:20px!important}.snowflake-card .heading-2.snowflake-title-line{font-size:24px!important;line-height:28px!important}}@media screen and (min-width:992px){.hp-hero__eyebrow a{gap:12px;margin-left:0;margin-right:0}.hp-hero__eyebrow a::after{content:\"\";background-image:url(\"data:image/svg+xml,%3Csvg width='6' height='11' viewBox='0 0 6 11' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M5.49134 5.79438C5.53447 5.75922 5.56923 5.71489 5.5931 5.66463C5.61697 5.61436 5.62935 5.55941 5.62935 5.50376C5.62935 5.44811 5.61697 5.39316 5.5931 5.34289C5.56923 5.29263 5.53447 5.2483 5.49134 5.21314L.736339 .413136C.522589 .203135 .331339 .203135 .151339 .413136C-.0286612 .623135 -.0586612 .818135 .151339 .994386L4.48634 5.50188L.155089 9.97938C.107068 10.0142 .0679743 10.0598 .0410153 10.1126C.0140562 10.1654 0 10.2238 0 10.2831C0 10.3424 .0140562 10.4009 .0410153 10.4537C.0679743 10.5065 .107068 10.5521 .155089 10.5869C.335089 10.7969 .530089 10.7969 .740089 10.5869L5.49134 5.79438Z' fill='black'/%3E%3C/svg%3E%0A\");display:inline-block;width:12px;height:12px;background-repeat:no-repeat;background-size:auto 100%;background-position:left center}.promo-banner--homepage{padding-top:32px}.homepage-banner-offset-container::after{height:50%}#storyHighlights{padding:2rem}.body-display-v2.snowflake-quote-item-quote-text{line-height:28px!important}.snowflake-hero-system-headline .heading-1-v2{line-height:48px;font-size:54px!important}.sc-overview__webinar-promo-banner .snowflake-content-chip-content{flex-direction:row;justify-content:space-between;align-items:center;width:100%}.sc-overview__webinar-promo-banner .snowflake-content-chip-content .heading-5-v2{flex-direction:row}.sc-overview__webinar-promo-banner .snowflake-content-chip-content .snowflake-title-v2-line:not(:first-child)::before{content:\"|\";margin:0 6px}.sc-cert-banner{padding:40px}.sc-cert-banner\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{margin:0!important;width:50%}.sc-cert-banner\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:first-child{flex-grow:1;padding-right:24px}.sc-cert-banner\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:last-child{max-width:240px}.summit-pricing-block__content\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:last-child{width:70%;padding-left:40px}.summit-pricing-block__content\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:first-child{width:30%}.summit-add-on-block__content\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(33.3333% - 24px);margin:0!important;display:flex}.summit-pricing-block__tile .snowflake-content-chip-content{display:flex;flex-direction:row;align-items:center;width:calc(100% - 200px)}.summit-pricing-block__tile .heading-5-v2 span.snowflake-title-v2-line:last-child{position:absolute;top:50%;transform:translate(0,-50%);right:40px}.press-body\u003E.snowflake-flexible-column-container-items\u003Ediv:last-child{position:sticky;top:120px}.snowflake-mega-nav-navigation-title:hover{color:var(--ui-01)}}@media screen and (min-width:1024px){.about-snowflake{padding:28px}.about-snowflake__logo{max-width:none;padding:0 0 0 48px;margin-bottom:0}.hero--press .snowflake-hero-system-layout-70-30 .snowflake-hero-system-content-container{width:85%}.snowflake-hero-system{padding-bottom:var(--spacing-04);padding-top:var(--spacing-07)}.hero--press .display-2-v2{font-size:64px;line-height:56px}.about-snowflake\u003E.container\u003E.cmp-container\u003E.aem-container{flex-direction:row;flex-wrap:nowrap;align-items:center}.about-snowflake\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:last-child{max-width:280px}.about-snowflake\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:first-child{flex-grow:1;margin-bottom:0!important}#polarisNavItem{margin-top:40px}.snowflake-mega-nav-nav-item-description{line-height:18px!important}.snowflake-mega-nav-column-items{gap:var(--spacing-01);grid-gap:var(--spacing-01)}.snowflake-mega-nav-navigation-title{text-transform:none}}div[id*=blueIcon] .snowflake-mega-nav-nav-item-icon__inner{background:var(--ui-01);padding:8px}div[id*=blueIcon]:hover .snowflake-mega-nav-nav-item-icon__inner{background:var(--ui-01)!important}.snowflake-mega-nav-nav-item-icon__inner{border-radius:4px;background:var(--ui-background-05);padding:6px}.snowflake-mega-nav-nav-item:hover .snowflake-mega-nav-nav-item-icon__inner{background:#fff!important}.snowflake-mega-nav-nav-item-icon.snowflake-image-container{height:40px;width:40px}.snowflake-mega-nav-dropdown-footer-links\u003E.snowflake-button-link\u003E.snowflake-button-container{font-size:16px!important;font-family:Texta!important;font-weight:800!important}.snowflake-mega-nav-dropdown-footer-icon.snowflake-image-container{margin-right:8px;width:40px!important;height:40px!important}#viewAllCapabilities a:hover{background:0 0!important}#platformFooter .snowflake-title-v2 .snowflake-title-v2-line:last-child{font-family:Lato;font-size:14px;font-weight:500}#platformFooter .snowflake-mega-nav-dropdown-footer-links{flex-grow:1;justify-content:flex-end;align-items:center}#platformFooter .snowflake-mega-nav-dropdown-footer-content{flex-direction:row}#offset,#open-source{flex-direction:column;border-top:1px solid #ccc}#offset::before,#open-source::before{content:\" \";display:block;width:100%;font-weight:800!important;font-size:12px!important;line-height:14px;text-transform:uppercase;white-space:nowrap;margin-top:16px;margin-bottom:8px}#open-source::before{content:\"Open Source Technologies\"}.snowflake-mega-nav-dropdown-menu-close-button{margin:var(--spacing-04) 0 var(--spacing-03)}.snowflake-mega-nav-column{gap:var(--spacing-02)!important}.snowflake-mega-nav-nav-item\u003Ea{width:100%;margin-left:-8px;padding:8px;border-radius:4px}.snowflake-mega-nav-nav-item\u003Ea:hover{background-color:var(--ui-background-05)}.snowflake-mega-nav-nav-item-description{margin-top:2px;display:block}#promobanner_overflowBottomDarkBlue::before{content:'';display:block;position:absolute;bottom:0;left:0;width:100%;height:50%;background:#212d35}#promobanner_overflowTopDarkBlue::before{content:'';display:block;position:absolute;top:0;left:0;width:100%;height:50%;background:#212d35}.overview-card\u003Ediv{box-shadow:0 0 14px 0 rgba(0,0,0,.10);background-color:#fff;border-radius:16px;overflow:hidden}.overview-card-text{padding:40px}.overview-card-image img{border-radius:0 !important}.overview-card-text h3,.overview-card-text .heading-3-v2{font-size:18px;line-height:1.1;margin-top:0}",":type":"snowflake-site/components/markup-editor","isGSAPEnabled":false},"mega_header":{"additionalClasses":"heap-nav-header","layout":"SIMPLE","id":"container-276668c87b",":type":"snowflake-site/components/mega-header",":items":{"nav_mega":{"activeItem":"item_1719963657751_c_663444255","id":"tabs-8660386c2f",":type":"snowflake-site/components/nav/nav-mega",":items":{"item_1719963657751_c_663444255":{"id":"nav-dropdown-menu-769eb6e409","enableDropdown":true,"nav_column_container":{"layout":"SIMPLE","id":"container-96f89025ee",":type":"snowflake-site/components/nav/nav-column/nav-column-container",":items":{"nav_column":{"additionalClasses":"nav-platform-sidebar","numberOfSubColumns":"one-column","minWidth":"230","maxWidth":"350","layout":"SIMPLE","id":"container-0e71841f8d",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item_copy_copy_2_793631646":{"id":"nav-item-2dd28d0a3e","additionalClasses":"nav-item__platform-parent is-platform","linkDescription":"Develop AI products, apps and more on a fully managed platform that securely connects businesses globally — across any type or scale of data.","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/platform/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"The Snowflake Platform"},":type":"snowflake-site/components/nav/nav-item"},"nav_item":{"id":"nav-item-3fa3fac3da","additionalClasses":"nav-item nav-item--si is-si","linkDescription":"All your knowledge. One trusted enterprise agent.","flag":"NOW GA","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/snowflake-cowork/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Snowflake CoWork"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_copy_2_836345186":{"id":"nav-item-d00d62e0d6","additionalClasses":"blue-icon is-analytics","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/analytics/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Analytics"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_copy_2":{"id":"nav-item-96da57db32","additionalClasses":"blue-icon is-ai","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/ai/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"AI"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_copy_2_1314771042":{"id":"nav-item-098bce1f19","additionalClasses":"blue-icon is-data-eng","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/data-engineering/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Data Engineering"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_144634":{"id":"nav-item-c8d3bb4296","additionalClasses":"blue-icon is-apps-collab","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/applications-and-collaboration/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Applications & Collaboration"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_144634_2013333117":{"id":"nav-item-8a6d6a54a0","additionalClasses":"blue-icon is-transactions","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/transactions/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Transactions"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item_copy_copy_2_793631646","nav_item","nav_item_copy_copy_2_836345186","nav_item_copy_copy_2","nav_item_copy_copy_2_1314771042","nav_item_copy_144634","nav_item_copy_144634_2013333117"]},"nav_column_copy_copy":{"additionalClasses":"meganav-platform-features","navColumnTitle":"Featured Capabilities","numberOfSubColumns":"one-column","layout":"SIMPLE","id":"container-37815df429",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item_copy_212715":{"id":"nav-item-ba353351c6","additionalClasses":"is-cortex-code","linkDescription":"Snowflake-native AI coding agent ","flag":"New","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/snowflake-coco/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Snowflake CoCo"},":type":"snowflake-site/components/nav/nav-item"},"nav_item":{"id":"nav-item-1e50b0c3fc","additionalClasses":"is-cortex-ai","linkDescription":"Instant access to industry-leading LLMs","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/product/features/cortex/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Cortex AI"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_660590635":{"id":"nav-item-dc30cc4bc3","additionalClasses":"is-marketplace","linkDescription":"Third-party data sources connected within minutes","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/product/features/marketplace/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Marketplace"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_660590":{"id":"nav-item-a437c76962","additionalClasses":"is-snowpark","linkDescription":"Libraries and code execution environments that run Python and more","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/product/features/snowpark/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Snowpark"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_660590_983061516":{"id":"nav-item-8bbc35085b","additionalClasses":"is-streamlit","linkDescription":"Framework for transforming Python scripts into web apps","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/product/features/streamlit-in-snowflake/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Streamlit"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item_copy_212715","nav_item","nav_item_copy_660590635","nav_item_copy_660590","nav_item_copy_660590_983061516"]},"nav_column_692142673":{"navColumnTitle":" ","numberOfSubColumns":"one-column","layout":"SIMPLE","id":"container-adeeaab061",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item_copy_660590_1739526127":{"id":"nav-item-6651106d08","additionalClasses":"is-postgres","linkDescription":"Fully compatible open source Postgres running on Snowflake","flag":"Now GA","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/features/postgres/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Postgres"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_185565":{"id":"nav-item-36f029c7ad","additionalClasses":"is-dcr","linkDescription":"Streamlined model development and MLOps from a centralized UI","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/features/end-to-end-ml-workflows/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Snowflake ML"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_212715":{"id":"nav-item-4e111cbf09","additionalClasses":"is-openflow","linkDescription":"Effortless data movement for integrations","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/product/features/openflow/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Openflow"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_660590":{"id":"nav-item-4ab9de5f9c","additionalClasses":"is-notebooks","linkDescription":"Interactive dev environment for data and AI teams","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/product/features/notebooks/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Notebooks"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_258535199":{"id":"nav-item-1abfc75aaf","propertiesId":"workload-nav-1","additionalClasses":"is-native-apps","linkDescription":"End-to-end, Snowflake-native app creation and distribution","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/product/features/native-apps/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Native Apps"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item_copy_660590_1739526127","nav_item_copy_185565","nav_item_copy_212715","nav_item_copy_660590","nav_item_258535199"]},"nav_column_782221091":{"navColumnTitle":" ","numberOfSubColumns":"one-column","layout":"SIMPLE","id":"container-10aa943b36",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item_copy":{"id":"nav-item-626959ece1","additionalClasses":"is-light-gray-icon is-horizon-catalog","linkDescription":"Universal AI catalog","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/features/horizon/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Horizon Catalog"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_660590_1293798742":{"id":"nav-item-6418784392","additionalClasses":"is-snowflake-ml","linkDescription":"Governed context layer that keeps AI, BI and data apps working from one truth","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/features/horizon-context/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Horizon Context"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_511717659_c":{"id":"nav-item-7ce1a86d77","additionalClasses":"is-unistore","linkDescription":"Unify transactional and analytical workloads in Snowflake for enhanced simplicity","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/features/unistore/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Unistore"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_511717659_c_1443811525":{"id":"nav-item-5435a9fa84","additionalClasses":"is-observe","linkDescription":"AI-powered observability for faster production troubleshooting","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/observe/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Observe"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_511717659_c_1006104884":{"id":"nav-item-ef0e295e86","additionalClasses":"is-observe","linkDescription":"Use any engine on a single governed data copy","flag":"Now GA","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/product/use-cases/interoperable-lakehouse/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Interoperable Lakehouse"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item_copy","nav_item_copy_660590_1293798742","nav_item_511717659_c","nav_item_511717659_c_1443811525","nav_item_511717659_c_1006104884"]}},":itemsOrder":["nav_column","nav_column_copy_copy","nav_column_692142673","nav_column_782221091"]},":type":"snowflake-site/components/nav/nav-dropdown-menu","cq:panelTitle":"Product"},"nav_dropdown_menu_2":{"id":"nav-dropdown-menu-31fb746f8c","enableDropdown":true,"nav_column_container":{"layout":"SIMPLE","id":"container-70bad2b739",":type":"snowflake-site/components/nav/nav-column/nav-column-container",":items":{"nav_column":{"navColumnTitle":"INDUSTRIES","numberOfSubColumns":"one-column","minWidth":"280","layout":"SIMPLE","id":"container-4a6c493f4a",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item_copy_361384_2056203141":{"id":"nav-item-293d4c4e54","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"All Industries"},":type":"snowflake-site/components/nav/nav-item"},"nav_item":{"id":"nav-item-1f780e518c","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/advertising-media-entertainment/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Advertising, Media & Entertainment"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy":{"id":"nav-item-98212c3013","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/financial-services/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Financial Services"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1970515619":{"id":"nav-item-7a41cb0c73","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/healthcare-and-life-sciences/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Healthcare & Life Sciences"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1533429516":{"id":"nav-item-1dd37c478c","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/manufacturing/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Manufacturing"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1444458226":{"id":"nav-item-1741dab5fa","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/public-sector/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Public Sector"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1149488919":{"id":"nav-item-973b83a8e5","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/retail-consumer-goods/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Retail & Consumer Goods"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_57417040":{"id":"nav-item-7df5339146","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/technology/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Technology"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_361384674":{"id":"nav-item-b07ec89b0d","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/telecom/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Telecom"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_361384":{"id":"nav-item-500887dc85","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/industries/travel-hospitality/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Travel & Hospitality"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item_copy_361384_2056203141","nav_item","nav_item_copy","nav_item_copy_1970515619","nav_item_copy_1533429516","nav_item_copy_1444458226","nav_item_copy_1149488919","nav_item_copy_57417040","nav_item_copy_361384674","nav_item_copy_361384"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-extra-small"},"nav_column_copy":{"navColumnTitle":"DEPARTMENTS","numberOfSubColumns":"one-column","minWidth":"160","layout":"SIMPLE","id":"container-e93de3a730",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item":{"id":"nav-item-f7a2f3226a","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/solutions/departments/finance/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Finance"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy":{"id":"nav-item-5c312e52ce","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/solutions/departments/information-technology/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"IT"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1970515619":{"id":"nav-item-17684099eb","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/solutions/departments/marketing/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Marketing"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item","nav_item_copy","nav_item_copy_1970515619"]},"nav_column_833417450":{"navColumnTitle":"Enablement Solutions","numberOfSubColumns":"one-column","layout":"SIMPLE","id":"container-5367757c5b",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item_copy_107772":{"id":"nav-item-f951021028","linkDescription":"Confident migration to a unified platform","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/migrate-to-the-cloud/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Migrate to the AI Data Cloud"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/nav_dropdown_menu_2/nav_column_container/nav_column_833417450/nav_item_copy_107772/icon.coreimg.svg/1723828484100/nav-icon-cloud.svg","lazyEnabled":true,"alt":"Cloud icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_copy":{"id":"nav-item-48f3c85d85","linkDescription":"Snowflake experts to help you accelerate and achieve business goals","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/solutions/services-delivery/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Services Delivery"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/nav_dropdown_menu_2/nav_column_container/nav_column_833417450/nav_item_copy_copy/icon.coreimg.svg/1768354429188/nav-icon--migrate.svg","lazyEnabled":true,"alt":"Migrate icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item_copy_107772","nav_item_copy_copy"]},"nav_column_copy_copy":{"navColumnTitle":"PARTNER SOLUTIONS","numberOfSubColumns":"one-column","layout":"SIMPLE","id":"container-ac8b281f42",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item":{"id":"nav-item-3db964a618","linkDescription":"Programs with product, solutions and cloud partners","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/why-snowflake/partners/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Snowflake Partner Network"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/nav_dropdown_menu_2/nav_column_container/nav_column_copy_copy/nav_item/icon.coreimg.svg/1723828498700/nav-icon--partner-network.svg","lazyEnabled":true,"alt":"Partner Network icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy":{"id":"nav-item-fe75c6d440","linkDescription":"Partners, apps and solutions for enhanced deployment","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/why-snowflake/partners/all-partners/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Partner Finder"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/nav_dropdown_menu_2/nav_column_container/nav_column_copy_copy/nav_item_copy/icon.coreimg.svg/1726173927645/nav-icon--partner-finder.svg","lazyEnabled":true,"alt":"Partner Finder icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1970515619":{"id":"nav-item-e67e6d957f","linkDescription":"Live and virtual events","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/why-snowflake/partners/event-partnership-opportunities/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Event Partnership Opportunities"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/nav_dropdown_menu_2/nav_column_container/nav_column_copy_copy/nav_item_copy_1970515619/icon.coreimg.svg/1726173935655/nav-icon--events.svg","lazyEnabled":true,"alt":"Calendar icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item","nav_item_copy","nav_item_copy_1970515619"]}},":itemsOrder":["nav_column","nav_column_copy","nav_column_833417450","nav_column_copy_copy"]},":type":"snowflake-site/components/nav/nav-dropdown-menu","cq:panelTitle":"Solutions"},"item_1719963657751_c":{"id":"nav-dropdown-menu-f11ee2dca9","enableDropdown":true,"nav_column_container":{"layout":"SIMPLE","id":"container-c30a09a05c",":type":"snowflake-site/components/nav/nav-column/nav-column-container",":items":{"nav_column":{"numberOfSubColumns":"one-column","minWidth":"230","maxWidth":"350","layout":"SIMPLE","id":"container-737f45649d",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item_copy_copy_2_793631646":{"id":"nav-item-f313d6f2dc","additionalClasses":"nav-item__platform-parent-why-sf","linkDescription":"Collaborate locally and globally to reveal new insights, create previously unforeseen business opportunities, and identify your customers with seamless experiences.","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/why-snowflake/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Why Snowflake"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item_copy_copy_2_793631646"]},"nav_column_copy_copy":{"additionalClasses":"meganav-platform-features","numberOfSubColumns":"two-columns","maxWidth":"1200","layout":"SIMPLE","id":"container-a80b0b1174",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item":{"id":"nav-item-619a06c6ca","propertiesId":"testID","linkDescription":"Case studies and videos showcasing how global organizations use Snowflake","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/customers/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Customers"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751_c/nav_column_container/nav_column_copy_copy/nav_item/icon.coreimg.svg/1739839279367/nav-icon--partner-network.svg","lazyEnabled":true,"alt":"Customer icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_258535199":{"id":"nav-item-47986a7bbf","propertiesId":"workload-nav-1","linkDescription":"Learn how to connect, share and integrate the data and apps on the AI Data Cloud","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/why-snowflake/what-is-data-cloud/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"The AI Data Cloud Explained"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751_c/nav_column_container/nav_column_copy_copy/nav_item_258535199/icon.coreimg.svg/1739840490955/nav-icon-cloud.svg","lazyEnabled":true,"alt":"Cloud icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_185565":{"id":"nav-item-5c543750ad","linkDescription":"Comprehensive security through built-in features, robust cloud infrastructure protection, and more","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/why-snowflake/snowflake-security-hub/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Security Hub"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751_c/nav_column_container/nav_column_copy_copy/nav_item_copy_185565/icon.coreimg.svg/1758909528089/user-security-admins-ciso-icon.svg","lazyEnabled":true,"alt":"User with security lock icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy":{"id":"nav-item-5222cb6cc9","additionalClasses":"is-light-gray-icon","linkDescription":"Maximize economic value through minimizing TCO and continuously optimizing price for performance","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/pricing-options/cost-and-performance-optimization/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Cost and Performance Optimization"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751_c/nav_column_container/nav_column_copy_copy/nav_item_copy/icon.coreimg.svg/1758909542267/nav-icon-cost-optimization-performance.svg","lazyEnabled":true,"alt":"Cost Optimization icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_185565_903555964":{"id":"nav-item-19ecf8b62d","linkDescription":"Startups building applications in the AI Data Cloud","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/why-snowflake/startup-program/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Snowflake for Startups"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751_c/nav_column_container/nav_column_copy_copy/nav_item_copy_185565_903555964/icon.coreimg.svg/1758732224323/launch.svg","lazyEnabled":true,"alt":"Launch","height":"64","width":"65",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item","nav_item_258535199","nav_item_copy_185565","nav_item_copy","nav_item_copy_185565_903555964"]}},":itemsOrder":["nav_column","nav_column_copy_copy"]},":type":"snowflake-site/components/nav/nav-dropdown-menu","cq:panelTitle":"Why Snowflake"},"item_1719961362824":{"id":"nav-dropdown-menu-e9a6d194f8","enableDropdown":true,"nav_column_container":{"layout":"SIMPLE","id":"container-5efc9ccc28",":type":"snowflake-site/components/nav/nav-column/nav-column-container",":items":{"nav_column_copy":{"navColumnTitle":"Connect","numberOfSubColumns":"one-column","minWidth":"124","layout":"SIMPLE","id":"container-f05a52f03d",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item":{"id":"nav-item-10141a46d3","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/blog/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Blog"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_180298689":{"id":"nav-item-b131cbc597","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/events/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Events"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_1639361946":{"id":"nav-item-4fe9dde190","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/support/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Support"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_680912746":{"id":"nav-item-a90bf3b33f","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/contact/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Contact us"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item","nav_item_180298689","nav_item_1639361946","nav_item_680912746"]},"nav_column_44600420__826130542":{"navColumnTitle":"Learn","numberOfSubColumns":"two-columns","layout":"SIMPLE","id":"container-2d192a27ee",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item_copy":{"id":"nav-item-8c3e7a4469","linkDescription":"Ebooks, videos, white papers and more","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/resources/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Resource Library"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item_copy/icon.coreimg.svg/1736877128196/nav-icon--notebooks.svg","lazyEnabled":true,"alt":"Notebooks icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item":{"id":"nav-item-08e26b1901","linkDescription":"Overview of Snowflake's educational offerings","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/resources/learn/training/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Training"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item/icon.coreimg.svg/1722385094416/nav-icon--training.svg","lazyEnabled":true,"alt":"Training icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_144634_1984107859":{"id":"nav-item-010e5ffafc","linkDescription":"Expert-led discussions and demos across industries and use cases","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/webinars/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Webinars"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item_copy_144634_1984107859/icon.coreimg.svg/1759424691990/nav-icon--webinars.svg","lazyEnabled":true,"alt":"Webinars icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1438098918":{"id":"nav-item-b475525b47","linkDescription":"Snowflake's technical industry professional certifications","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/en/resources/learn/certifications/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Certifications"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item_copy_1438098918/icon.coreimg.svg/1722382780833/nav-icon--cert.svg","lazyEnabled":true,"alt":"Certification icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_143809":{"id":"nav-item-2d1f3ac319","linkDescription":"Weekly product demos showcasing key features and live Q&A ","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/webinars/demo/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Live Demos"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item_copy_143809/icon.coreimg.svg/1759424359543/nav-icon--live-demo.svg","lazyEnabled":true,"alt":"Live Demo icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_333890638":{"id":"nav-item-615ad296a1","linkDescription":"Training courses for all levels, on-demand or instructor-led","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"attributes":{"target":"_blank"},"url":"https://learn.snowflake.com/en/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Snowflake University"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item_copy_333890638/icon.coreimg.svg/1722382769808/nav-icon--education.svg","lazyEnabled":true,"alt":"Education icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_189945":{"id":"nav-item-555ca288dc","linkDescription":"Instructor-led virtual workshops for exploring key Snowflake features","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/webinars/virtual-hands-on-lab/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Hands-On Labs"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item_copy_189945/icon.coreimg.svg/1759388182903/nav-icon--labs.svg","lazyEnabled":true,"alt":"Hands-on Labs icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_333890":{"id":"nav-item-5a1b862f7c","linkDescription":"Academic papers written by Snowflake researchers","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/resources/publications/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Snowflake Research Publications"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item_copy_333890/icon.coreimg.svg/1756326371387/copy.svg","lazyEnabled":true,"alt":"Copy","height":"64","width":"65",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_333890_930852828":{"id":"nav-item-e9fc831ec8","linkDescription":"Informative articles about AI and data topics","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/fundamentals/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Fundamentals"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719961362824/nav_column_container/nav_column_44600420__826130542/nav_item_copy_333890_930852828/icon.coreimg.svg/1756853637155/data-sheet.svg","lazyEnabled":true,"alt":"Document with list","height":"64","width":"65",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item_copy","nav_item","nav_item_copy_144634_1984107859","nav_item_copy_1438098918","nav_item_copy_143809","nav_item_copy_333890638","nav_item_copy_189945","nav_item_copy_333890","nav_item_copy_333890_930852828"]}},":itemsOrder":["nav_column_copy","nav_column_44600420__826130542"]},"nav_promo_section":{"id":"nav-promo-section-a08590fc60","experience_fragment_1":{"id":"experiencefragment-2d5111bdd5","localizedFragmentVariationPath":"/content/experience-fragments/snowflake-site/language-masters/en/site/nav-promo-card/master1/jcr:content","configured":true,":type":"snowflake-site/components/experiencefragment",":items":{"root":{"columnClassNames":{"nav_promo_card":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-c095455e54",":type":"snowflake-site/components/container",":items":{"nav_promo_card":{"id":"nav-promo-card-40b1534123","openInNewWindow":true,"layout":"horizontal","headline":"Dev Day Virtual - June 25","description":"Don’t just hear about AI — build it. Luminary talks and hands-on labs","linkTitle":"Learn more","linkUrl":"/en/dev-day/americas-virtual/","image":{"id":"image","src":"https://www.snowflake.com/adobe/dynamicmedia/deliver/dm-aid--de231e36-6645-4550-abd9-0f8de758ac66/web-dev-day-26-960x540-1x.png?quality=85&preferwebp=true","lazyEnabled":true,"alt":"dev day","height":"540","width":"960",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-promo-card"}},":itemsOrder":["nav_promo_card"]},"cq:metadata":{":type":"nt:unstructured"}},":itemsOrder":["root","cq:metadata"],"classNames":"aem-xf"},"experience_fragment_2":{"id":"experiencefragment-b867949923","localizedFragmentVariationPath":"/content/experience-fragments/snowflake-site/language-masters/en/site/nav-promo-card/navigation-promo-card-2/jcr:content","configured":true,":type":"snowflake-site/components/experiencefragment",":items":{"root":{"columnClassNames":{"nav_promo_card":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-a9beee8ebf",":type":"snowflake-site/components/container",":items":{"nav_promo_card":{"id":"nav-promo-card-3a8a97737c","openInNewWindow":true,"layout":"horizontal","headline":"The ROI of Gen AI and Agents 2026","description":"Discover how 92% of early adopters are achieving positive ROI with gen AI.","linkTitle":"Learn More","linkUrl":"/en/lp/radical-roi-generative-ai/","image":{"id":"image","src":"https://www.snowflake.com/adobe/dynamicmedia/deliver/dm-aid--0c15edae-1a97-4739-8b16-c7f3941a6d9e/web-roi-of-gen-ai-and-agents-2026-r02-960x540.png?quality=85&preferwebp=true","lazyEnabled":true,"alt":"roi of gen ai and agents","height":"540","width":"960",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-promo-card"}},":itemsOrder":["nav_promo_card"]},"cq:metadata":{":type":"nt:unstructured"}},":itemsOrder":["root","cq:metadata"],"classNames":"aem-xf"},"experience_fragment_3":{"id":"experiencefragment-2f39b95672","localizedFragmentVariationPath":"/content/experience-fragments/snowflake-site/language-masters/en/site/nav-promo-card/navigation-promo-card-3/jcr:content","configured":true,":type":"snowflake-site/components/experiencefragment",":items":{"root":{"columnClassNames":{"nav_promo_card":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-2ea82af430",":type":"snowflake-site/components/container",":items":{"nav_promo_card":{"id":"nav-promo-card-c3b3e71e46","openInNewWindow":true,"layout":"horizontal","headline":"Startup 2026: AI Agents Mean Business","description":"Venture leaders weigh in on agentic AI. ","linkTitle":"Learn more","linkUrl":"/en/lp/building-startup-ai-age/","image":{"id":"image","src":"https://www.snowflake.com/adobe/dynamicmedia/deliver/dm-aid--a320b404-dca1-4477-b033-c79708538657/web-startup-2026-960x540.png?quality=85&preferwebp=true","lazyEnabled":true,"alt":"alt","height":"540","width":"960",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-promo-card"}},":itemsOrder":["nav_promo_card"]},"cq:metadata":{":type":"nt:unstructured"}},":itemsOrder":["root","cq:metadata"],"classNames":"aem-xf"},":type":"snowflake-site/components/nav/nav-promo-section"},":type":"snowflake-site/components/nav/nav-dropdown-menu","cq:panelTitle":"Resources"},"item_1719963657751":{"id":"nav-dropdown-menu-e3ca061aef","enableDropdown":true,"nav_column_container":{"layout":"SIMPLE","id":"container-0b9e370f70",":type":"snowflake-site/components/nav/nav-column/nav-column-container",":items":{"nav_column_copy_copy":{"navColumnTitle":"Build","numberOfSubColumns":"one-column","layout":"SIMPLE","id":"container-60693b663d",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item":{"id":"nav-item-c6252af78a","propertiesId":"testID","linkDescription":"Overview of the dev resources you need to build and scale","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/developers/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Snowflake for Developers"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751/nav_column_container/nav_column_copy_copy/nav_item/icon.coreimg.svg/1731362494574/nav-icon--devs.svg","lazyEnabled":true,"alt":"Developers icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1855651246":{"id":"nav-item-03adaa9652","linkDescription":"Reference architectures, use cases and best practices","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/developers/guides/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Developer Guides"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751/nav_column_container/nav_column_copy_copy/nav_item_copy_1855651246/icon.coreimg.svg/1761677891705/nav-icon--solution-center.svg","lazyEnabled":true,"alt":"Solution Center icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy":{"id":"nav-item-b375c0c28b","additionalClasses":"is-light-gray-icon","linkDescription":"The latest software versions, drivers, libraries and relevant docs","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/developers/downloads/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Downloads"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751/nav_column_container/nav_column_copy_copy/nav_item_copy/icon.coreimg.svg/1731362660050/nav-icon-download.svg","lazyEnabled":true,"alt":"Download icon","height":"28","width":"28",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item","nav_item_copy_1855651246","nav_item_copy"]},"nav_column_copy_copy_1367930678":{"navColumnTitle":"Learn","numberOfSubColumns":"one-column","layout":"SIMPLE","id":"container-d492935624",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item":{"id":"nav-item-5a088a516a","propertiesId":"testID","linkDescription":"Reference docs, guides, tutorials and announcements","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"attributes":{"target":"_blank"},"url":"https://docs.snowflake.com/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Documentation"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751/nav_column_container/nav_column_copy_copy_1367930678/nav_item/icon.coreimg.svg/1731361950527/nav-icon--docs.svg","lazyEnabled":true,"alt":"Docs icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy":{"id":"nav-item-c7cb9ce62d","additionalClasses":"is-light-gray-icon","linkDescription":"Key projects Snowflake engineers maintain and support","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/developers/open-source/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Open Source"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751/nav_column_container/nav_column_copy_copy_1367930678/nav_item_copy/icon.coreimg.svg/1731365437016/nav-icon-open-source.svg","lazyEnabled":true,"alt":"Open Source icon","height":"32","width":"32",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_copy":{"id":"nav-item-92e3cfeeef","additionalClasses":"is-light-gray-icon","linkDescription":"Online and in-person classes and workshops to upskill on Snowflake","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"/en/developers/northstar/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_INTERNAL","text":"Builder Education"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751/nav_column_container/nav_column_copy_copy_1367930678/nav_item_copy_copy/icon.coreimg.svg/1731362475640/nav-icon--northstar.svg","lazyEnabled":true,"alt":"Northstar logo","height":"32","width":"32",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item","nav_item_copy","nav_item_copy_copy"]},"nav_column_copy_copy_1101894776":{"navColumnTitle":"Connect","numberOfSubColumns":"one-column","layout":"SIMPLE","id":"container-f504059385",":type":"snowflake-site/components/nav/nav-column",":items":{"nav_item":{"id":"nav-item-cfe8a76214","propertiesId":"testID","linkDescription":"Snowflake’s technical leaders on what, why and how they build features","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://www.snowflake.com/engineering-blog/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Engineering Blog"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751/nav_column_container/nav_column_copy_copy_1101894776/nav_item/icon.coreimg.svg/1757101368571/nav-icon--developer-center.svg","lazyEnabled":true,"alt":"Developers icon","height":"32","width":"32",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"},"nav_item_copy_1855651246":{"id":"nav-item-b6eb708b43","linkDescription":"Tips, tricks and discussion with fellow Snowflake developers","button":{"id":"button","showOutboundIcon":false,"buttonLink":{"valid":true,"attributes":{"target":"_blank"},"url":"https://community.snowflake.com/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Community"},"icon":{"id":"icon","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/mega-nav-header/master/_jcr_content/root/mega_header/nav_mega/item_1719963657751/nav_column_container/nav_column_copy_copy_1101894776/nav_item_copy_1855651246/icon.coreimg.svg/1731362644348/nav-icon--partner-network.svg","lazyEnabled":true,"alt":"Partner Network icon","height":"64","width":"64",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-item"}},":itemsOrder":["nav_item","nav_item_copy_1855651246"]}},":itemsOrder":["nav_column_copy_copy","nav_column_copy_copy_1367930678","nav_column_copy_copy_1101894776"]},"nav_promo_section":{"id":"nav-promo-section-1bd2d5eeb3","experience_fragment_1":{"id":"experiencefragment-945eaee743","localizedFragmentVariationPath":"/content/experience-fragments/snowflake-site/language-masters/en/site/nav-promo-card/nav-promo-5/jcr:content","configured":true,":type":"snowflake-site/components/experiencefragment",":items":{"root":{"columnClassNames":{"nav_promo_card":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-cee6a417b2",":type":"snowflake-site/components/container",":items":{"nav_promo_card":{"id":"nav-promo-card-e8061ecead","openInNewWindow":false,"layout":"horizontal","headline":"Get started with your first Snowflake Notebook","description":"Write and execute code, visualize results, and tell the story of your analysis all in one place.","linkTitle":"Learn More","linkUrl":"/en/developers/solutions-center/getting-started-with-your-first-snowflake-notebook-project/","image":{"id":"image","src":"https://www.snowflake.com/adobe/dynamicmedia/deliver/dm-aid--dc7e334a-c38b-4283-b1de-fcf829952eef/nav-promo-first-notebook.jpg?quality=85&preferwebp=true","lazyEnabled":true,"alt":"alt","height":"210","width":"415",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-promo-card"}},":itemsOrder":["nav_promo_card"]},"cq:LiveSyncConfig":{"cq:isDeep":true,"cq:rolloutConfigs":[],"cq:master":"/content/experience-fragments/snowflake-site/language-masters/en/site/nav-promo-card/nav-promo-card-4",":type":"cq:LiveCopy"}},":itemsOrder":["root","cq:LiveSyncConfig"],"classNames":"aem-xf"},"experience_fragment_2":{"id":"experiencefragment-134b19b161","localizedFragmentVariationPath":"/content/experience-fragments/snowflake-site/language-masters/en/site/nav-promo-card/nav-promo-card-4/jcr:content","configured":true,":type":"snowflake-site/components/experiencefragment",":items":{"root":{"columnClassNames":{"nav_promo_card":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-28bf9c990f",":type":"snowflake-site/components/container",":items":{"nav_promo_card":{"id":"nav-promo-card-ec29c8eefa","openInNewWindow":true,"layout":"horizontal","headline":"Northstar Builder Workshops","description":"Join other developers as you roll up your sleeves and explore the possibilities of Snowflake.","linkTitle":"Learn More","linkUrl":"/en/nav-promos/northstar-builders-workshop/","image":{"id":"image","src":"https://www.snowflake.com/adobe/dynamicmedia/deliver/dm-aid--14341ced-bc5e-4a29-9762-b7857f6cadfc/nav-promo-northstar.jpg?quality=85&preferwebp=true","lazyEnabled":true,"alt":"Snowflake Northstar logo","height":"700","width":"1440",":type":"snowflake-site/components/image"},":type":"snowflake-site/components/nav/nav-promo-card"}},":itemsOrder":["nav_promo_card"]},"cq:LiveSyncConfig":{"cq:isDeep":true,"cq:rolloutConfigs":[],"cq:master":"/content/experience-fragments/snowflake-site/language-masters/en/site/nav-promo-card/master",":type":"cq:LiveCopy"}},":itemsOrder":["root","cq:LiveSyncConfig"],"classNames":"aem-xf"},":type":"snowflake-site/components/nav/nav-promo-section"},":type":"snowflake-site/components/nav/nav-dropdown-menu","cq:panelTitle":"Developers"},"item_1718247180324":{"id":"nav-dropdown-menu-f4fae3dfe1","enableDropdown":false,"link_url":"/en/pricing-options/",":type":"snowflake-site/components/nav/nav-dropdown-menu","cq:panelTitle":"Pricing"}},":itemsOrder":["item_1719963657751_c_663444255","nav_dropdown_menu_2","item_1719963657751_c","item_1719961362824","item_1719963657751","item_1718247180324"]},"languagenavigation":{"id":"language-navigation-5a039b97e0","languageNavItems":[{"title":"English","path":"/en/developers/guides/end2endmigration/","locale":"en","active":true},{"title":"日本語","path":"/ja/","locale":"ja","active":false},{"title":"한국어","path":"/ko/","locale":"ko","active":false},{"title":"中文（简体）","path":"/zh_cn/","locale":"zh-cn","active":false},{"title":"Português","path":"/pt_br/","locale":"pt-br","active":false},{"title":"Deutsch","path":"/de/","locale":"de","active":false},{"title":"Français","path":"/fr/","locale":"fr","active":false},{"title":"Español","path":"/es/","locale":"es","active":false},{"title":"Italiano","path":"/it/","locale":"it","active":false}],":type":"snowflake-site/components/nav/language-navigation"},"button_1177328691":{"id":"button-3fb2bd9550","heapButtonClasses":["mega-nav__sign-in"],"showOutboundIcon":false,"buttonLink":{"valid":true,"attributes":{"target":"_blank"},"url":"https://app.snowflake.com/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","appliedCssClassNames":"snowflake-button-link snowflake-button-black snowflake-button-compact","linkType":"SNOWFLAKE_EXTERNAL","text":"Sign in"},"button":{"id":"button-f1c6021254","heapButtonClasses":["contact_nav","heap-nav-contact"],"showOutboundIcon":true,"buttonLink":{"valid":true,"url":"/en/contact-sales/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","appliedCssClassNames":"snowflake-button-secondary snowflake-button-blue snowflake-button-compact","linkType":"SNOWFLAKE_INTERNAL","text":"CONTACT SALES"},"button_288358396":{"id":"button-0b19f3432c","heapButtonClasses":["start_for_free_nav","heap-nav-start-for-free"],"showOutboundIcon":false,"buttonLink":{"valid":true,"url":"https://signup.snowflake.com/"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","appliedCssClassNames":"snowflake-button-primary snowflake-button-blue snowflake-button-compact","linkType":"SNOWFLAKE_EXTERNAL","text":"start for free"}},":itemsOrder":["nav_mega","languagenavigation","button_1177328691","button","button_288358396"],"appliedCssClassNames":"snowflake-header-container white"}},":itemsOrder":["markup_editor","mega_header"]},"image":{":type":"nt:unstructured"},"cq:targetMetadata":{"cq:targetStatus":"OUT_OF_SYNC","cq:exportTime":1781280015540,"cq:targetOfferId":860250,":type":"nt:unstructured"}},":itemsOrder":["root","image","cq:targetMetadata"],"classNames":"aem-xf"},"markup_editor_1950346551":{"id":"markup-editor-7a2f7b9edd","title":" ","cssContent":".snowflake-markdown-table code[class*=language-],.snowflake-markdown-table code[class*=language-],.snowflake-markdown .snowflake-text code[class*=language-],.snowflake-markdown .snowflake-text pre[class*=language-]{background-color:rgba(var(--ui-12-rgb),.5);color:var(--text-01);text-shadow:none;padding:var(--spacing-00);border-radius:var(--spacing-00);font-size:smaller}",":type":"snowflake-site/components/markup-editor","isGSAPEnabled":false},"responsivegrid":{"columnClassNames":{"quickstart_hero":"aem-GridColumn aem-GridColumn--default--12","flexible_column_cont":"aem-GridColumn aem-GridColumn--default--12","markup_editor":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","columnCount":12,":items":{"quickstart_hero":{"id":"quickstart-hero-1f58d6ab2f","quickstartHeroFirstCertifiedTag":{"tagText":"Quickstart","tagColor":"#29B5E8","tagPath":"/content/cq:tags/snowflake-site/taxonomy/solution-center/certification/quickstart","tagIcon":""},"quickstartHeroTitle":{"lines":["End-to-End Migration to Snowflake in Action: Data and Pipelines"],"type":"heading2",":type":"snowflake-site/components/title-v2"},"quickstartHeroAuthor":"Kuo Lun Lo","quickstartHeroFirstSnowflakeFeatureTag":{"tagText":"Transformation","tagColor":"#29B5E8","tagPath":"/content/cq:tags/snowflake-site/taxonomy/snowflake-feature/transformation","tagIcon":""},"quickstartHeroForkRepoLink":{"id":"button-2d2588f2b1","showOutboundIcon":false,"buttonLink":{"valid":true,"attributes":{"target":"_blank"},"url":"https://github.com/Snowflake-Labs/sfquickstarts/tree/master/site/sfguides/src/end2endmigration"},"linkTargetContentType":"GENERIC",":type":"snowflake-site/components/button","linkType":"SNOWFLAKE_EXTERNAL","text":"Fork Repo"},"quickstartHeroBreadcrumbs":[{"title":"End-to-End Migration to Snowflake in Action: Data and Pipelines","url":"https://www.snowflake.com/content/snowflake-site/global/en/developers/guides/end2endmigration","currentPage":true},{"title":"Guides","url":"https://www.snowflake.com/content/snowflake-site/global/en/developers/guides","currentPage":false},{"title":"Snowflake for Developers","url":"https://www.snowflake.com/content/snowflake-site/global/en/developers","currentPage":false}],"fragmentPath":"/content/dam/snowflake-site/en/content-fragments/quickstarts/end2endmigration",":type":"snowflake-site/components/quickstart/quickstart-hero","isDeveloperGuidesPage":false},"flexible_column_cont":{"id":"flexible-column-container-825922db8f","propertiesId":"quickstart-template-main-flexible-container","type":"2-column-75-25","alignColumns":"top","containerMaxWidth":"extra-large","topPadding":"none","bottomPadding":"none","spaceBetween":"small","reverseOnMobile":false,"carouselOnMobile":false,"backgroundImageOption":"none","flexible_column_content_container_1":{"layout":"SIMPLE","id":"container-6231c9e31f",":type":"snowflake-site/components/flexible-column-container/flexible-column-content-container",":items":{"contentfragment":{"id":"contentfragment-8669c126b4","paragraphs":["\u003Ch2\u003EOverview\u003C/h2\u003E\n","\u003Cp\u003EWant to migrate to Snowflake in less than 90 minutes? Come to this hands-on lab. We will work through Snowflake's migration strategy and automation tools&mdash;for example, SnowConvert and Snowpark Migration Accelerator&mdash;to assess, move, and validate an entire pipeline with data into Snowflake. Our data will be in another data warehouse, a data lake, and even on prem. Can we move it into Snowflake? Our pipelines are written in Jupyter notebooks utilizing Spark. Can those fully exist in Snowflake? Can all of it happen in Snowflake? Let's find out.\u003C/p\u003E\n","\u003Ch3\u003EPrerequisites\u003C/h3\u003E\n\u003Cul\u003E\u003Cli\u003EBasic knowledge of SnowConvert and SMA\u003C/li\u003E\u003C/ul\u003E\n","\u003Ch3\u003EWhat You'll Learn\u003C/h3\u003E\n\u003Cul\u003E\u003Cli\u003EMigrate SQL Server Adventure Works' DDLs\u003C/li\u003E\u003Cli\u003EFix issues derived from SnowConvert conversion output code\u003C/li\u003E\u003Cli\u003EMigrate SQL Server Adventure Works' Historical Data\u003C/li\u003E\u003Cli\u003EMigrate Spark ETL Pipeline of Adventure Works\u003C/li\u003E\u003C/ul\u003E\n","\u003Ch3\u003EWhat You'll Need\u003C/h3\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Ca href=\"/en/migrate-to-the-cloud/snowconvert-ai/\"\u003ESnowConvert\u003C/a\u003E Installed\u003C/li\u003E\u003Cli\u003E\u003Ca href=\"/en/migrate-to-the-cloud/migration-accelerator/\"\u003ESnowpark Migration Accelerator (SMA)\u003C/a\u003E Installed\u003C/li\u003E\u003Cli\u003E\u003Ca href=\"https://code.visualstudio.com/download\"\u003EVSCode\u003C/a\u003E and \u003Ca href=\"https://marketplace.visualstudio.com/items?itemName=snowflake.snowflake-vsc\"\u003ESnowflake Extension\u003C/a\u003E Installed\u003C/li\u003E\u003Cli\u003E\u003Cem\u003E[Preferred, but not required]\u003C/em\u003E Python and PySpark Installed\u003C/li\u003E\u003C/ul\u003E\n","\u003Ch3\u003EScenario\u003C/h3\u003E\n","\u003Cp\u003EAdventureWorks is a retail company established 20 years ago. They have been using SQL Server since their inception. As they grew and data processing evolved, they decided to start running some pipelines using Spark to take advantage of distributed computing. They also created SQL Server data marts in Azure in an early attempt to move into the cloud.\u003C/p\u003E\n","\u003Cp\u003EOver time, they have decided to consider using Snowflake. They want to take one of their SQL Server datamarts and all of the pipelines that are feeding it&hellip; and move it into Snowflake. This sounds like a daunting task. How should they approach it?\u003C/p\u003E\n","\u003Cp\u003EAdventureWorks has decided to do a Proof of Concept (POC) with a single database. The goal is to move everything from this database into Snowflake, as well as at least one of the pipelines bringing data into this database and at least one of the reporting notebooks they have connected to this database.\u003C/p\u003E\n","\u003Ch2\u003EAssessment\u003C/h2\u003E\n","\u003Cp\u003EAssessing what you have and developing a plan is the most essential part of a migration. When done right, this will set you up for success over the remainder of the migration. Whether it&rsquo;s one table, pipeline, or data warehouse, you need to have a plan that takes into account not just your data, but everything that interacts with your data. A successful migration is built at the beginning with a complete picture of what you want to migrate. With that assessment, comes a plan of attack on how to migrate.\u003C/p\u003E\n","\u003Cp\u003EIn order to migrate across platforms, AdventureWorks needs to utilize a migration framework and strategy that will account for everything required to get the database up and running. Moving some of the data or some of the notebooks&hellip; this will not be enough. They need to move the entire system of pipelines that is connected to that data and those notebooks.\u003C/p\u003E\n","\u003Cp\u003EHere&rsquo;s a migration strategy espoused by Snowflake:\n\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/01_assessment.png\" alt=\"Assessment\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThere are many phases in this migration framework, but most of them can be bucketed into three larger categories:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003EAssessment\u003C/strong\u003E - Understand what you have\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EConversion\u003C/strong\u003E - Move it to the new platform\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EValidation\u003C/strong\u003E - Test that everything works\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EWe will walk through all the nine phases shown above in this Hand-On-Lab, but note that the primary activities we are doing are the three listed above. If you get lost or you&rsquo;re not su    re where to go next, remember to assess -&gt; convert -&gt; test. Let&rsquo;s start with the assessment phase.\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ENote that executing a migration requires a close eye on access and security. You will be running two platforms in parallel, and getting a whole new environment setup. For the purpose of time in this lab, we have already set up the target (Snowflake) environment for you, and we will assume you are the only user of this environment. But full RBAC management and support is something every database administrator will have to deal with.\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Ch3\u003EPlanning and Design\u003C/h3\u003E\n","\u003Cp\u003EAs was mentioned previously, this migration will take a single database that the AdventureWorks company has and migrate it to Snowflake. This will include any data pipelines that are bringing data in and out of this mart.\u003C/p\u003E\n","\u003Cp\u003ETo better understand everything that is bringing this data in and out of the pipeline, let&rsquo;s put together an architecture diagram.\u003C/p\u003E\n","\u003Ch4\u003ESource Architecture\u003C/h4\u003E\n","\u003Cp\u003EGetting an architecture diagram is often one of the first steps in understanding what you have. If you&rsquo;ve never built an architecture diagram, it can seem like a daunting task. A few suggestions would be:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003EStart simple.\u003C/strong\u003E Don&rsquo;t try to create a complete picture in an orderly way. Identify key components, then start listing the components that interact with this data.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003ESocialize your drafts.\u003C/strong\u003E Data owners across any organization will know best how users are interacting with the data. The knowledge is not always consolidated with the systems or IT team.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EIdentify opportunities for improvement.\u003C/strong\u003E When you can visually see the connection points in your system, sometimes the opportunities for improvements in performance, price, or complexity are immediately identifiable.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003ERevisit it.\u003C/strong\u003E Most organizations' data landscapes change wildly in a very short period of time. These static diagrams are quickly out of date, and when the discussion of modernization occurs, the creators are not around anymore or they are simply not aware of changes that have occurred.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ESnowflake won&rsquo;t suggest a specific tool that will help you manage your architecture, but there are tools available.\nHere&rsquo;s a simple diagram.\n\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/02_sourcearchitecture.png\" alt=\"SourceArchitecture\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESpark is being used across this organization in a variety of ways, but for this POC, it is simply being used as ETL. It is being orchestrated by Airflow, but that is assumed to continue for the moment.\u003C/p\u003E\n","\u003Ch4\u003ETarget Architecture\u003C/h4\u003E\n","\u003Cp\u003EThis is the source system that we have. Let&rsquo;s consider the future state\u003C/p\u003E\n","\u003Cp\u003EAs we migrate, this is going to be our goal. Our inventory of artifacts in the source configuration will map over to objects in the target.\n\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/03_targetarchitecture.png\" alt=\"TargetArchitecture\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that this is a \u003Cstrong\u003Elift-and-shift migration\u003C/strong\u003E. This means we are going to recreate exactly what is in the source in the target environment. Everything that was in our SQL Server data mart plus everything that was connected to it will be recreated as is. Once you are on the new platform, then you can optimize.\u003C/p\u003E\n","\u003Cp\u003ESome may advocate for a different approach to the migration that is more of a lift-adjust-shift. While this approach may yield some optimization benefits sooner in the migration process, most projects will take longer and the testing will be considerably more challenging. Most migration projects benefit from the lift-and-shift approach, but it especially makes makes the most sense for our two goals:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EProve the migration is possible\u003C/li\u003E\u003Cli\u003EAnd do it in 90 minutes.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EAs a result, we will be moving the following and testing them in Snowflake:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003ESQL Server Database\u003C/li\u003E\u003Cli\u003EPython Notebooks with references to the Spark API\u003C/li\u003E\u003C/ul\u003E\n","\u003Ch4\u003EInventories\u003C/h4\u003E\n","\u003Cp\u003EIn order to get started with this, we will need to build some object inventories of what we have in our existing system. We may know that we have a SQL Server data mart that we want to move with some Jupyter notebooks, but what objects are we migrating? What local files are we reading from? What dependencies exist across those objects and files?\u003C/p\u003E\n","\u003Cp\u003EFor a smaller migration like this, this may be easy to track down manually. But for large migrations with multiple data sources, multiple different types of pipelines, and multiple different data owners, this can be nearly impossible. The challenge of understanding what you have often leads to the project never even getting off the ground.\u003C/p\u003E\n","\u003Cp\u003ELuckily, Snowflake has some tools to help you understand what you have. These tools are not specific to building an object inventory, but rather are comprehensive tools for assessment and conversion of source code to Snowflake. These two tools will be with us most of the way, so let&rsquo;s go ahead and introduce them now:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003ESnowConvert\u003C/strong\u003E: SnowConvert is a tool that scans source code, reports on what it finds in the code, and inventories that code.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003ESnowpark Migration Accelerator (SMA)\u003C/strong\u003E: The SMA is built on the same framework as SnowConvert, but it takes in code written in Python or Scala with references to the Spark API in them.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EEach of these tools is built on the basic principle that your codebase can be scanned, and a semantic model of everything that is happening in your code can be built. From this model, various reports (like an object inventory) and the output code can be generated.\n\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/04_tool.png\" alt=\"Structure\"\u003E\nSo let&rsquo;s do this for real. Let&rsquo;s start getting our inventory of objects together.\u003C/p\u003E\n","\u003Cp\u003EFor expediency in this lab, we are going to go straight from the assessment into the conversion phase. This means that we will look at the assessment data while we work through the conversion. Note that in reality, we would want to understand everything we have, before we start to move it.\u003C/p\u003E\n","\u003Cp\u003ENow that we have an inventory of the artifacts we&rsquo;re looking to move to Snowflake, let&rsquo;s go ahead and move them.\nWe&rsquo;ll walk through moving the following steps:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EDatabase Code Conversion\u003C/li\u003E\u003Cli\u003EData Migration\u003C/li\u003E\u003Cli\u003EData Ingestion (pipelines)\u003C/li\u003E\u003Cli\u003EReporting and Analytics\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EWhile at the highest level, this phase focuses on moving or converting from the source to Snowflake, there will often be testing and additional pieces of code to assess as we move through it. Meaning that while we do this conversion, it will be iterative with the testing intermixed into the process.\u003C/p\u003E\n","\u003Ch2\u003EDatabase Code Conversion\u003C/h2\u003E\n","\u003Cp\u003EWe will start by assessing what we have. SnowConvert.\u003C/p\u003E\n","\u003Ch3\u003EProject Creation\u003C/h3\u003E\n","\u003Cp\u003EWe have an adventure works database in SQL Server, and we have some spark scripts that load data into the sql server. The customer wants to move all of this into Snowflake.\u003C/p\u003E\n","\u003Cp\u003EThe SQL Server is in an Azure environment that we have in Snowflake, but can make public for a temporary amount of time?\u003C/p\u003E\n","\u003Cp\u003EHow do we connect to it? Through SnowConvert. Let&rsquo;s do that now.\u003C/p\u003E\n","\u003Cp\u003EOpen SnowConvert. Currently, \u003Ca href=\"/en/migrate-to-the-cloud/snowconvert-ai/\"\u003Eit can be downloaded from here\u003C/a\u003E. An installation package file will be created. You can then open this file and install the application on your machine.\u003C/p\u003E\n","\u003Cp\u003EAgree to the Terms of Use by selecting &ldquo;Agree&rdquo;.\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ENote that if you are on a Mac, you may be asked to move the application to your windows folder.\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003E\u003Cstrong\u003EOpen SnowConvert\u003C/strong\u003E: When you open the application it may prompt you to update it:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/initialScreen.png\" alt=\"InitialScreen\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWe won&rsquo;t do that at this time, but when working with the application, it is generally better to keep it as up to date as possible. This is a local application. Keeping it up to date not only ensures the functionality of the application, but also ensures that you have the most up to date version of the conversion core.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003ESelect New Project\u003C/strong\u003E:\n\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/NewProject.png\" alt=\"NewProject\"\u003E\u003C/p\u003E\n","\u003Cp\u003ETo begin using any version of SnowConvert, you will need to create a project. Think of a project as a local config file that will be saved to your machine. This will preserve any settings and will allow you to continue where you left off if you need to step away.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s call our project: \u003Cstrong\u003ESQL Server ADW Test.\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EWe then need to select a source. In this case, it is SQL Server.\u003C/p\u003E\n","\u003Cp\u003ETo run any element of a project in SnowConvert, you will need to provide an access code. Unless you have used SnowConvert before, you will need an access code. Happily, you can request access in the tool by selecting &ldquo;Get an access code&rdquo; next to the access code drop down.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/ProjectCreation.png\" alt=\"ProjectCreation\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that if you have already activated an access code, you will see options in the dropdown menu in the license screen as shown here:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/ProjectCreation2.png\" alt=\"ProjectCreation2\"\u003E\u003C/p\u003E\n","\u003Cp\u003EYou can choose an access code from this list if you have already activated one.\u003C/p\u003E\n","\u003Cp\u003EAssuming we do not yet have an access code, let&rsquo;s request an access code. Choose &ldquo;Get an access code&rdquo; from the menu to the right side of the dropdown menu. When you do this, the access code form will pop up:\n\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/fill_license.png\" alt=\"license\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis information is needed to confirm who you are to Snowflake. It will not be used by the SnowConvert team to start sending you the latest and greatest update on what&rsquo;s new in migrations. (Though if that&rsquo;s something you&rsquo;d be interested in&hellip; let us know.)\u003C/p\u003E\n","\u003Cp\u003EOnce you complete the form, you will receive an email with an access code. It will look something like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/licensedelivery.png\" alt=\"license_delivery\"\u003E\u003C/p\u003E\n","\u003Cp\u003EPaste this access code into the application where it says &ldquo;Enter new access code or select one&rdquo; in SnowConvert. When your access code has been accepted, you will get a small message under the dropdown menu that says \u003Cstrong\u003E&ldquo;Access code activated successfully&rdquo;\u003C/strong\u003E.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/insert_license.png\" alt=\"insert_license\"\u003E\u003C/p\u003E\n","\u003Cp\u003EYou will need to have an active internet connection in order to activate your access code. If you are unable to activate your access code, check out the \u003Ca href=\"https://docs.snowconvert.com/sc/general/frequently-asked-questions-faq#why-am-i-not-receiving-an-access-code\"\u003Etroubleshooting section\u003C/a\u003E of the SnowConvert documentation.\u003C/p\u003E\n","\u003Cp\u003ENow that we&rsquo;re active, let&rsquo;s Extract! It&rsquo;s time!\u003C/p\u003E\n","\u003Ch3\u003EExtract\u003C/h3\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/extractLandscape.png\" alt=\"extractLandscape\"\u003E\u003C/p\u003E\n","\u003Cp\u003EFrom the Project Creation menu, select the blue \u003Cstrong\u003ECONTINUE\u003C/strong\u003E button in the bottom right corner of the application. Then click on \u003Cstrong\u003EFrom SQL Server\u003C/strong\u003E in order to extract code from source database.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/fromSQLServer.png\" alt=\"FromSQLServer\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWhen you scroll down, you will have to enter the connection information:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/connect_sqlserver.png\" alt=\"conn\"\u003E\u003C/p\u003E\n","\u003Cp\u003EFor this Hands on Lab, we will provide you with some credentials. However, if you are executing this outside of the friendly confines of this hands on lab, you will need to enter your own credentials.\u003C/p\u003E\n","\u003Cp\u003EFill the connection information to connect to Adventure Works, you can build your own Adventure Works environment \u003Ca href=\"https://learn.microsoft.com/en-us/sql/samples/adventureworks-install-configure?view=sql-server-ver17&amp;tabs=ssms\"\u003Ehere\u003C/a\u003E as well.\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EAuthentication Method: Standard\u003C/li\u003E\u003Cli\u003EServer URL: \u003Cem\u003EAdventure Works Connection URL\u003C/em\u003E\u003C/li\u003E\u003Cli\u003EDatabase: \u003Cem\u003EAdventureWorks\u003C/em\u003E\u003C/li\u003E\u003Cli\u003EPort: \u003Cem\u003E1433\u003C/em\u003E\u003C/li\u003E\u003Cli\u003EUsername: \u003Cem\u003EFill the user name here\u003C/em\u003E\u003C/li\u003E\u003Cli\u003EPassword: \u003Cem\u003EFill the password here\u003C/em\u003E\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ECheck both boxes for &ldquo;Trust Server Certificate&rdquo; and &ldquo;Encrypt Connection&rdquo;.\u003C/p\u003E\n","\u003Cp\u003ENext, we will have to specify a local path for our project folder. Anything that we do SnowConvert will be preserved in this project path as well as anything that is created locally. Choose a path that is fully accessible to you. This is the path I chose: \u003Ccode\u003E/Users/userName/tmp/SQLServer_ADW_Test\u003C/code\u003E\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/selectProjectPath.png\" alt=\"selectProjectPath\"\u003E\u003C/p\u003E\n","\u003Cp\u003EFinally, you can click \u003Ccode\u003ECONNECT TO DATABASE\u003C/code\u003E, then you will get a pop up that says &ldquo;Connect Established&rdquo; when you have connected. SnowConvert will then take you to the catalog screen:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/extraction_page.png\" alt=\"extract_page\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThe catalog screen allows you to browse objects that were found in the database. For SQL Server, this could be tables, views, procedures, or functions. Nothing has been converted yet. This is merely an inventory of what SnowConvert found in the source.\u003C/p\u003E\n","\u003Cp\u003EUsing the catalog, we can select a set of objects for which we&rsquo;d like to extract the DDL. Using the filter options, you can search for a specific object or set of objects. Using the checkboxes, you can select a specific subset of objects or select the highest checkbox to select everything:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/select_extract.png\" alt=\"select_extract\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIn this example, we will select the top checkbox and select everything. This will include tables, views, and functions. Then select &ldquo;EXTRACT OBJECTS&rdquo; to extract the DDL.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/click_extract.png\" alt=\"click_extract\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis will create a folder on the local machine preservig the structure of the objects in the database with a file for the DDL for each object.\nWhen the extraction is complete, you will see a results screen similar to this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/extract_result.png\" alt=\"extract_result\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis will give you a brief overview of what was extracted. If there were errors or something was not able to be extracted, it will be reported to you here.\u003C/p\u003E\n","\u003Cp\u003EYou can select &ldquo;VIEW EXTRACTED OBJECTS&rdquo; to see where SnowConvert put the extracted DDL. But since the number of objects we have extracted matches what we expected and there are no errors, we can close this dialog menu and return to the catalog.\u003C/p\u003E\n","\u003Cp\u003ENote that now we can see a green checkbox where the DDL was successfully extracted for the object:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/post_extract.png\" alt=\"post_extract\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIf there was an error extracting the DDL, you would see a red X and would need to resolve why that was not extracted.\u003C/p\u003E\n","\u003Ch3\u003EConversion\u003C/h3\u003E\n","\u003Cp\u003EAt this point, we&rsquo;ve extracted the objects in the database and we&rsquo;re ready to assess the compatibility with Snowflake and begin the conversion process. There are some optional steps we can do before we get to the conversion itself. Let&rsquo;s take a look at this by selecting &ldquo;GO TO MAPPINGS -&gt;&rdquo; in the bottom right corner of the application.\u003C/p\u003E\n","\u003Cp\u003EThis brings us to the mapping screen.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/mappings.png\" alt=\"Mappings\"\u003E\u003C/p\u003E\n","\u003Cp\u003EOn this screen, you can choose a new name for a specific object in Snowflake (i.e. map a single object from SQL Server to Snowflake). You can also choose BULK MAPPING to apply a prefix or suffix to all of the objects or a subset of them (such as tables or view).\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/bulkmapping.png\" alt=\"BulkMapping\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that this is completely optional when doing the migration. In this scenario, we will not do any custom mappings.\u003C/p\u003E\n","\u003Cp\u003EBecause this is a poc, we&rsquo;ll apply the suffix &ldquo;_poc&rdquo; to all the objects that we have in our database. Because this applies the change to the object itself, any relationship to the object will be updated to reflect the new name. (For example, if the name of TABLE_A was changed to TABLE_A_poc, and TABLE_A was referenced by FUNCTION_B, the reference to TABLE_A in FUNCTION_B will be changed to TABLE_A_poc.)\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/mapping2.png\" alt=\"BulkMappingInfoAdded\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIn the suffix panel, we can add the suffix we want to fix to each object name. Select &ldquo;APPLY MAPPINGS&rdquo; and you will see the name changes in the Target Name column of the catalog:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/mapping3.png\" alt=\"MappingApplied\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that you can still edit individual object names if so desired.\u003C/p\u003E\n","\u003Cp\u003ENow that we&rsquo;ve mapped all of the source and target objects, we can run the conversion and view the assessment information. Select &ldquo;START CONVERSION&rdquo;.\nSince we are leaving our object names unaffected, let&rsquo;s start the conversion process. Select &ldquo;START CONVERSION&rdquo; in the bottom right had corner of the application.\u003C/p\u003E\n","\u003Cp\u003EIt&rsquo;s possible you will view an error message similar to this one:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/scopeValidation.png\" alt=\"ScopeValidation\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis simply means that SnowConvert has scanned the code that it extracted from the database before it runs its conversion script, and has found some things that COULD cause errors. It will tell you some things you might want to change in the source before converting. These can be found in the &ldquo;Scope validation report&rdquo; that you can read. In this scenario, we&rsquo;ll just click &ldquo;CONTINUE&rdquo;.\u003C/p\u003E\n","\u003Cp\u003ESnowConvert will then execute its conversion engine. This is done by scanning the codebase and creating a semantic model of the source codebase. This model is then used by SnowConvert to create the output Snowflake code as well as the generated reports.\u003C/p\u003E\n","\u003Cp\u003EWhen the conversion is finished, each step will be highlighted:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/conversion_progress.png\" alt=\"ConversionProgress\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESelect &ldquo;VIEW RESULTS&rdquo; to&hellip; well&hellip; view the results.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/conversion_results.png\" alt=\"ConversionResults\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThe results page will give you a code completeness score initially, but there is more information below if you scroll down. There is more information on each element of the output report in the SnowConvert documentation, but we&rsquo;ll just highlight a few elements of the report for this lab, and we&rsquo;ll do the followup for each of them which will explore more in depth.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003ECode Completeness\u003C/strong\u003E: This is a reference to any missing elements or objects that are not present in the codebase. If you have 100% code completeness, then you do not have any missing objects or references to missing elements in the codebase.\u003C/p\u003E\n","\u003Cp\u003EIn this scenario, we have an overall 97.5% completeness. This is broken down in the Assessed Conversion Scope Summary (scroll down in the application to find this section):\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/assessment_summary.png\" alt=\"AssessmentSummary\"\u003E\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003EWe may not need to resolve any missing dependencies (it&rsquo;s unlikely in this scenario because we did a direct extract from the database), but we still want to see why SnowConvert is telling us that something is missing.\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003ETo do this, let&rsquo;s visit the additional reports that are generated by SnowConvert. Select &ldquo;VIEW REPORTS&rdquo; from the bottom of the application:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/clickViewReports.png\" alt=\"ClickViewReports\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis will take you to your file explorer, specifically to the local directory where the reports generated by SnowConvert have been created. This will be within the directory that you specified in the project creation menu at the start of the project.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/outputReports.png\" alt=\"OutputReports\"\u003E\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ENote that the extracted and converted code are also in this directory structure, but we&rsquo;ll get to those later.\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003EIn the SnowConvert folder, you will find many reports. We&rsquo;ll view some of them later, but for now, let&rsquo;s focus on the MissingObjectReferences.csv report. Open this one.\nIn this file, we&rsquo;ll find the name of a referenced element that is not present in the DDL of the output code.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/missingreferences.png\" alt=\"MissingReferences\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s validate that this is a missing reference. There are several ways we can do this, but the first would be to simply open the object inventory and validate that this function actually does exist. The inventory of all code objects in this scan is in the \u003Cstrong\u003ETopLevelCodeUnits.&lt;datetime&gt;.csv\u003C/strong\u003E file:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/clickCodeUnit.png\" alt=\"ClickCodeUnits\"\u003E\u003C/p\u003E\n","\u003Cp\u003EOpening this file, we can search for this reference in the CodeUnitId column:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/codeUnitReport.png\" alt=\"CodeUnitReport\"\u003E\u003C/p\u003E\n","\u003Cp\u003EHuzzah! I have found it here. This means that the object does exist. Why did SnowConvert identify it as a missing object? That&rsquo;s a great question. It could be that it&rsquo;s called in a way that SnowConvert could not resolve. It could be that there was an error with SnowConvert, and it did not resolve the function name with the function DDL. We&rsquo;ll have to dive deeper into why when we work through the issues. For now, it&rsquo;s enough to know that the object does exist and we do not need to figure out how essential it is to track down. This may still come out when working through the issues, but let&rsquo;s find out then.\u003C/p\u003E\n","\u003Cp\u003EIf the result is 100% code completeness:\u003C/p\u003E\n","\u003Cp\u003EIn this scenario, we have 100% code completeness. This makes sense given that we are exporting this directly from the source. If you do have missing objects here, the recommendation would be to open the reports folder and validate that the missing objects are either known to be missing or find the DDL for this object.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EConversion Overview\u003C/strong\u003E: Now that we have seen that we have the code that we need for this, let&rsquo;s see how much of our code was converted. Let&rsquo;s review the Code Units Summary section here:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/codeUnitOverview.png\" alt=\"CodeUnitOverview\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELooks like we have tables, views, and functions in this codebase, but not a lot of code in general (this looks like less than 1000 lines of code in total). There also are only three &ldquo;EWI&rsquo;s&rdquo; (in the last column), meaning that the majority of this extracted DDL can be moved over to Snowflake just by using SnowConvert. We&rsquo;ll look through the EWI&rsquo;s in a moment.\u003C/p\u003E\n","\u003Cp\u003EUnderstanding what we have is essential to successfully completing a migration. If we were pre-migration, we would likely stop here and review the object inventory. We&rsquo;d also want to run the Snowpark Migration Accelerator (SMA) to validate that any pipelines we have include the objects that we are migrating here. In this scenario, we are going to go ahead and move forward to work through any issues that we have and will run the SMA later.\u003C/p\u003E\n","\u003Cp\u003ESince we have a good understanding of what needs to be done and it&rsquo;s relatively small, let&rsquo;s go ahead and attack this. Let&rsquo;s resolve the issues that we have present. Before we do that, let&rsquo;s take a look at the status in our object inventory. Select &ldquo;GO TO DEPLOYMENT&rdquo; in the application.\u003C/p\u003E\n","\u003Ch2\u003EDeployment and solve issues\u003C/h2\u003E\n","\u003Cp\u003EThis will take you back to the inventory screen. It should look something like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentPage.png\" alt=\"DeploymentPage\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis is the same inventory that we have already seen, but note that now we can see a status in the conversion column. There are a few different elements that we can see here that are based on the conversion status. This lets us know which objects were fully converted (a green checkmark), which objects have a warning that you should consider (a green checkmark with a blue &ldquo;i&rdquo; icon), and which objects have a conversion error that must be addressed (a red &ldquo;X&rdquo;). These statuses are determined by the error messaging that SnowConvert has placed into the converted code. The objects with a red &ldquo;X&rdquo; have an error message that will produce an error if you attempt to run that SQL in Snowflake. If you resolve the errors, then you will be able to deploy the output code. SnowConvert is connected to your project directory, so as long as you keep the files in the same location that SnowConvert put them when it converted the code from the source.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s see this in action by resolving the issues.\u003C/p\u003E\n","\u003Ch3\u003EResolving Issues (powered by Cortex AI)\u003C/h3\u003E\n","\u003Cp\u003ESnowConvert will generate you an inventory of all issues that it encounters. This is in the issues.csv file. Let&rsquo;s take a look at issues that we have available in this execution of SnowConvert.\u003C/p\u003E\n","\u003Cp\u003ETo find the issues report, go to &ldquo;VIEW OUTPUT&rdquo; in the bottom of the SnowConvert application.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/viewOutput.png\" alt=\"ViewOutput\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis will take you to a directory title &ldquo;Conversion-&lt;datetime&gt;&rdquo; within the directory you originally created in the project creation screen at the start of the project. This output will have three different sub directories:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003ELogs\u003C/li\u003E\u003Cli\u003EOutput\u003C/li\u003E\u003Cli\u003EReports\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ELet&rsquo;s first visit the reports directory to see what issues SnowConvert identified with this conversion.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/checkIssuesReport.png\" alt=\"ViewIssues\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThe issues.&lt;datetime&gt;.csv report will be available in the reports folder under the SnowConvert subfolder. In this report, you will find the type of each error as well as its location and a description of the error. There are three major types of error generated by SnowConvert:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003EConversion Error (EWI)\u003C/strong\u003E: generally, this is something that the tool could not convert or hasn&rsquo;t seen before\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EFunctional Difference (FDM)\u003C/strong\u003E: this is code that has been converted, but may be functionally different in Snowflake. These errors should be treated as warnings, but paid close attention to during testing.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EPerformance Reviews (PRF)\u003C/strong\u003E: this is something that SnowConvert identifies that will run in Snowflake, but may be suboptimal. You should consider optimizing this once you&rsquo;re up and running in Snowflake.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EThere&rsquo;s more information on each of these \u003Ca href=\"https://docs.snowconvert.com/sc/general/technical-documentation/issues-and-troubleshooting\"\u003Ein the SnowConvert documentation\u003C/a\u003E. Let&rsquo;s look at what we have in this execution:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/checkIssuesReport2.png\" alt=\"ViewIssuesContent\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWe can see that some issues share an issue code, but we can see specifically which file has the issue in it and a description of what the issue is. This is a very low number of issues. There will not be a ton of things to work through.\u003C/p\u003E\n","\u003Cp\u003EThere are many approaches to dealing with the conversion issues generated by SnowConvert. We would recommend that you start in the same order you would want to deploy the objects: tables first, then views and functions, followed by procedures and scripts. You could pivot this table by the root folder (which will have the object name in it), then sort by severity and type. Depending on your timeline and business need, this would allow you to deal programmatically with the most critical issues in the order of deployment.\u003C/p\u003E\n","\u003Cp\u003EHowever, in this lab, there are such a small number of issues, we don&rsquo;t really need to pivot this file. However, we will still start with the same approach. Where are we going to deal with these issues? Snowflake would recommend that you use the Snowflake Visual Studio Code Extension, and that is what we are going to use right now.\u003C/p\u003E\n","\u003Cp\u003EThroughout this lab, anytime that we are going to interact with the code, we are going to use the Snowflake extension for VS Code. We can open it now and leave it open for the duration of this lab. If you have not yet downloaded the \u003Ca href=\"https://docs.snowflake.com/en/user-guide/vscode-ext\"\u003EVS Code extension\u003C/a\u003E, you can do so from within VS Code or from the \u003Ca href=\"https://marketplace.visualstudio.com/items?itemName=snowflake.snowflake-vsc\"\u003EVS Code Marketplace\u003C/a\u003E.\u003C/p\u003E\n","\u003Cp\u003EWhen you first open VS Code, navigate to the settings for your extensions. In the settings for the Snowflake extension, you will need to enable the SnowConvert Migration Assistant:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/enableExtension.png\" alt=\"enableExtension\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that we will be using Snowpark Checkpoints later on in this lab. While you&rsquo;re here, you may as well enable that part of the extension as well.\u003C/p\u003E\n","\u003Cp\u003EWhen you enable this, VS Code may prompt you to restart. Even if it doesn&rsquo;t, it&rsquo;s a good idea to restart the application now.\u003C/p\u003E\n","\u003Cp\u003EOnce you&rsquo;ve restarted VS Code, open the Snowflake extension and login to your Snowflake account. For this lab, Snowflake accounts have been created for you and provided via the information available for this lab.\u003C/p\u003E\n","\u003Cp\u003ENow we should be ready, let&rsquo;s open a new browser window, and make our root directory the same one we created in the project creation menu. This will allow us to access both the extracted source code and the output Snowflake code that has our issues in it.\u003C/p\u003E\n","\u003Cp\u003EYou should see a folder that has the name of the project you created in the project creation screen:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/vsFolderSelect.png\" alt=\"vsFolderSelect\"\u003E\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ENote: if you expand that folder, you should see an input and an output directory\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003EThe extracted code is in the input directory, and the Snowflake code generated by SnowConvert is in the output folder. But we will not need to navigate this folder structure to find our issues. We can use the Snowflake extension.\u003C/p\u003E\n","\u003Cp\u003EOpen the Snowflake extension, and you should see a section called \u003Cstrong\u003ESNOWCONVERT ISSUES\u003C/strong\u003E. It should automatically load the list of issues under the VS Code's root directory.\u003C/p\u003E\n&lt;!-- \n![OpenSCIssues](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/openSCIssues.png)\n\nIt should take you to the same folder that you already choose as your root directory. You can choose the folder that has the name of the project you created in the project creation:\n\n![issuesDir](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesDir.png)\n\nWhen you select this folder, it should show you a few additional folders. These are from the output directory where you generated the output code. These will be each of the files that has an issue in it:\n--&gt;\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesRoot.png\" alt=\"issuesRoot\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s expand this so we can see it a bit better:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesExpanded.png\" alt=\"issuesExpanded\"\u003E\u003C/p\u003E\n","\u003Cp\u003EYou can see each of the files that has an EWI in it and the exact location.\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ENote: for most migrations, this will be exceptionally large. It would be good to plan how to approach the issues using the issues.csv spreadsheet as mentioned earlier.\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003EIf you select a specific error, it will take you to the place in the code where that error can be found:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode.png\" alt=\"issuesCode\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s not start with the first error posted, but instead start in the order that was recommended earlier: tables, views, functions, then procedures/scripts. In this example, we have 1 table (in the file \u003Cstrong\u003Edatabaselog.sql\u003C/strong\u003E) with an error in it. Let&rsquo;s take a look:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode2.png\" alt=\"issuesCode2\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis error is an EWI saying that a datatype was converted to a different datatype in Snowflake. This doesn&rsquo;t look like a problem, but more looks like SnowConvert is trying to tell us to validate that this is not a problem before we deploy.\u003C/p\u003E\n","\u003Cp\u003EIf we&rsquo;re not sure that this datatype should be converted to VARIANT, let&rsquo;s use the SnowConvert Migration Assistant to generate some more information on this error. To do this, let&rsquo;s select the light bulb icon in the SNOWCONVERT ISSUES menu in the extension:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode3.png\" alt=\"issuesCode3\"\u003E\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ENote that a new window will pop up saying &ldquo;Loading explanation and suggestion&hellip;&rdquo;. This is connecting to your Snowflake account and passing the error message that SnowConvert generated to Cortex in your connected Snowflake account. It will give you an explanation after it has had a moment to think through it:\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode4.png\" alt=\"issuesCode4\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis explanation seems to validate that the conversion is correct. VARIANT is the right translation for an XML file. From this menu you can:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003ECopy the replacement code that it has generated\u003C/li\u003E\u003Cli\u003EChat with Cortex\u003C/li\u003E\u003Cli\u003EReturn to the issue\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ELet&rsquo;s say that we are not sure that this is the right conversion. We could ask Cortex if there are any other options for XML data in Snowflake.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode5.png\" alt=\"issuesCode5\"\u003E\u003C/p\u003E\n","\u003Cp\u003EYour results may vary, but this is what Cortex returned to me. VARIANT is still the recommendation. We&rsquo;ll take a look that the data matches depending on how we are using this XML data when we test the result of this conversion.\u003C/p\u003E\n","\u003Cp\u003EFor now, let&rsquo;s replace the code with the EWI with the code that Cortex has recommended (which is essentially, to remove the EWI). Our new code looks like this:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-sql\"\u003ECREATE OR REPLACE TABLE dbo.DatabaseLog (\n    DatabaseLogID INT NOT NULL,\n    PostTime TIMESTAMP_NTZ(3) NOT NULL,\n    DatabaseUser VARCHAR(128) NOT NULL,\n    Event VARCHAR(128) NOT NULL,\n       &quot;Schema&quot; VARCHAR(128) NULL,\n    Object VARCHAR(128) NULL,\n    TSQL VARCHAR NOT NULL,\n    XmlEvent VARIANT NOT NULL\n   )\n   COMMENT = '{ &quot;origin&quot;: &quot;sf_sc&quot;, &quot;name&quot;: &quot;snowconvert&quot;, &quot;version&quot;: {  &quot;major&quot;: 1,  &quot;minor&quot;: 6,  &quot;patch&quot;: &quot;0.0&quot; }, &quot;attributes&quot;: {  &quot;component&quot;: &quot;transact&quot;,  &quot;convertedOn&quot;: &quot;05/22/2025&quot;,  &quot;domain&quot;: &quot;snowflake&quot; }}'\n   ;\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EAnd the EWI message is gone. Note, however, that the file and issue still appear in the SnowConvert Issues section of the Snowflake extension even after we refresh the list:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode7.png\" alt=\"issuesCode7\"\u003E\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ETo remove this, we need to save the changes that we&rsquo;ve made to the file. This will allow us to update the extension and the SnowConvert application. Let&rsquo;s take a look at this.\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesRefresh.png\" alt=\"issuesRefresh\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENow let&rsquo;s refresh the SNOWCONVERT ISSUES list in the Snowflake extension, and the issue no longer appears. We can see this same behavior in the SnowConvert application.\u003C/p\u003E\n","\u003Cp\u003EIf you return to the application, we should still be on the deployment screen. If you expand the &ldquo;Tables&rdquo; section of the deployment screen, you will see that &ldquo;DatabaseLog&rdquo; is marked with a red &ldquo;X&rdquo; indicating that it cannot be deployed.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode8.png\" alt=\"issuesCode8\"\u003E\u003C/p\u003E\n","\u003Cp\u003EHowever, now that we&rsquo;ve resolved the EWIs in the DDL for that object, we can refresh this list by selecting the refresh icon in the top left of the application:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode9.png\" alt=\"issuesCode9\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWe should see that the red &ldquo;X&rdquo; now changes to a green checkmark.\u003C/p\u003E\n","\u003Cp\u003EIf it does not change, then there are still errors associated with that DDL that need to be resolved. That does not appear to be the case here.\u003C/p\u003E\n","\u003Cp\u003EWe can now continue to resolve the rest of the errors in our execution. The tables now all appear to have their issues resolved. (You can validate that they are ready to deploy by scrolling through the catalog in SnowConvert to see that all objects are marked with a green checkmark.)\u003C/p\u003E\n","\u003Cp\u003EIf you are doing a large project, it might be a good idea to go ahead and deploy the tables. To validate that those are correct. However, views and functions should both be done before they are deployed.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s take a look at the remaining issues we have:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode15.png\" alt=\"issuesCode15\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWe have two remaining issues in two different functions, but they have the same error code. Note that this is also a FDM code, so there may not be a solution we need to apply here. Let&rsquo;s select the first issue and call Cortex to tell us more about it.\u003C/p\u003E\n","\u003Cp\u003EThis is an interesting issue. SnowConvert has decided that this UDF should be turned into a stored procedure. This is an interesting decision as a stored procedure cannot be called in the same way as a function. But there are things that you can do with functions in SQL Server that you cannot do with a function in Snowflake. And if you ask Cortex what it thinks, it will disagree with SnowConvert and turn this back into a UDF:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-sql\"\u003ECREATE OR REPLACE FUNCTION dbo.udfMinimumDate(X TIMESTAMP_NTZ(3), Y TIMESTAMP_NTZ(3))\nRETURNS TIMESTAMP_NTZ(3)\nLANGUAGE SQL\nAS\n$$\n  IFF(X &lt;= Y, X, Y)\n$$\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EThis is where you need someone familiar with how this function is used across the workload. Normally, we would inventory any reference to this function across the entire workload. Since this is a small POC, this function is not used anywhere else in this function. You can validate this in a couple of ways. The first is to simply search for the name of this object in the input directory in VS Code:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode17.png\" alt=\"issuesCode17\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThere is only reference to this function and it is in the create function statement. So it is not being used anywhere. We can also check the dependent objects inventory generated by SnowConvert. (You could do this in a file browser or in VS code directly.)\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode18.png\" alt=\"issuesCode18\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWe can see that while there are many mentions of this object in the reports, there is no mention of this object in the \u003Cstrong\u003EObjectReferences.&lt;datetime&gt;.csv\u003C/strong\u003E report. As a result, this function is not used anywhere, so we&rsquo;re not sure how this will be called.\u003C/p\u003E\n","\u003Cp\u003ESo, should we go with what SnowConvert has generated or listen to Cortex? There are advantages and disadvantages to both, but the primary reason would be turning this function into a procedure means it WILL fail if it is called in a way that a procedure cannot be called. This means that you will be immediately alerted to this when you test this output. Leaving it as a function means that there may be functional differences that you may not immediately be aware of.\u003C/p\u003E\n","\u003Cp\u003EIn this situation:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EInterpreting what the function is trying to do versus the way it is written, I think this could be a UDF.\u003C/li\u003E\u003Cli\u003EIt&rsquo;s also likely that this is going to be called in a way that is not going to work for a procedure.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ESo I&rsquo;m going to make the choice to use the Cortex recommended code and put in the function code. However&hellip; I will add my own comment to the file that contains the code as a stored procedure. That way I can revisit it if I need to when it comes time to test these functions. That looks like this:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-sql\"\u003ECREATE OR REPLACE FUNCTION dbo.udfMinimumDate(X TIMESTAMP_NTZ(3), Y TIMESTAMP_NTZ(3))\nRETURNS TIMESTAMP_NTZ(3)\nLANGUAGE SQL\nAS\n$$\n  IFF(X &lt;= Y, X, Y)\n$$\n\n-- SnowConvert turned this into a stored procedure. Leaving as a comment.\n--** SSC-FDM-0029 - USER DEFINED FUNCTION WAS TRANSFORMED TO SNOWFLAKE PROCEDURE **\n/* CREATE OR REPLACE PROCEDURE dbo.udfMinimumDate (X TIMESTAMP_NTZ(3), Y TIMESTAMP_NTZ(3))\nRETURNS VARCHAR\nLANGUAGE SQL\nCOMMENT = '{ &quot;origin&quot;: &quot;sf_sc&quot;, &quot;name&quot;: &quot;snowconvert&quot;, &quot;version&quot;: {  &quot;major&quot;: 1,  &quot;minor&quot;: 6,  &quot;patch&quot;: &quot;0.0&quot; }, &quot;attributes&quot;: {  &quot;component&quot;: &quot;transact&quot;,  &quot;convertedOn&quot;: &quot;05/23/2025&quot;,  &quot;domain&quot;: &quot;snowflake&quot; }}'\nEXECUTE AS CALLER\nAS\n$$\n    DECLARE\n        Z TIMESTAMP_NTZ(3);\n    BEGIN\n         \n        IF (:X &lt;= :Y) THEN\n            Z := :X;\n        ELSE\n            Z := :Y;\n        END IF;\n        RETURN (:Z);\n    END;\n$$; */\n\u003C/code\u003E\u003C/pre\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ENote that I have left the FDM message in the code. This means that the SnowConvert catalog will still alert me that there is a warning present in this function.\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003EI will save the file, and update the SNOWCONVERT ISSUES list in the Snowflake VS Code Extension. Note that this still leaves the FDM.\u003C/p\u003E\n\u003Cblockquote\u003E\n","\u003Cp\u003ENote that this still leaves FDM in the browser because I intentionally left it in the file.\u003C/p\u003E\n\u003C/blockquote\u003E\n","\u003Cp\u003EI will collapse this and take a look at the last issue. Since this issue has the same error code as the previous one, it is likely to be a similar case with a similar solution.\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-sql\"\u003ECREATE OR REPLACE FUNCTION dbo.udfTwoDigitZeroFill(NUMBER INT)\nRETURNS VARCHAR\nLANGUAGE SQL\nAS\n$$\n  CASE \n      WHEN NUMBER &gt; 9 THEN CAST(LEFT(NUMBER::STRING, 2) AS CHAR(2))\n      ELSE CAST(LEFT('0' || NUMBER::STRING, 2) AS CHAR(2))\n  END\n$$\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EAnd&hellip; it is a similar issue. This function is doing something slightly different, but I will make the same decision here that I made before to trust that the function will give me the right result, preserve the procedure as a comment, and test this later on.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s save the file, and now&hellip; we are done! We have successfully navigated through the issues generated by SnowConvert.\u003C/p\u003E\n","\u003Cp\u003ENote that there are a large number of issues that SnowConvert can generate. The ones that we see here are generally something that you need to make a decision on the best solution path for your situation.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s validate this in SnowConvert. If you&rsquo;ve saved each of the files, you should be able to see checkmark across all of the objects, though there will still be some blue &ldquo;i&rdquo; icon:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentReady.png\" alt=\"deployementReady\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s move forward in the process and deploy the code. This will run all of the DDL that we have extracted from SQL Server (and subsequently, resolved any issues with) to Snowflake. To deploy the code, select &ldquo;DEPLOY&rdquo; at the bottom of the application:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentClick.png\" alt=\"deploymentClick\"\u003E\u003C/p\u003E\n","\u003Ch3\u003EDeploy objects\u003C/h3\u003E\n","\u003Cp\u003EIn order to deploy, you need to be connected to a Snowflake account. You will be prompted for your Snowflake connection information:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/connectToSnowflake.png\" alt=\"connectToSnowflake\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWhen you enter your login information, you may be taken to a webpage with this message.\n\u003Ccode\u003EYour identity was confirmed and propagated to Snowflake Node.js driver. You can close this window now and go back where you started from.\u003C/code\u003E\nMost Snowflake accounts require authentication and this is a part of that authentication. You can return to the SnowConvert application to resume the deployment process. You should see some results from your deployment similar to this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentResults.png\" alt=\"deploymentResults\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENow, we know there should be more than 1 object here. There is one error on deployment. You may also see an error message like this one:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deployError.png\" alt=\"deployError\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIf you do see this, you can see why the error occurred in the catalog in SnowConvert by hovering your mouse over the top of the error symbol.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deployErrorStatus.png\" alt=\"deployErrorStatus\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIn this case, there is a permissions problem. Check the connection you are using. If this fails, you may want to create the DATABASE and the SCHEMA manually in the Snowflake, especially if the database you are creating already exists. You could troubleshoot this in SnowConvert, but it would be better to do so in Snowflake.\u003C/p\u003E\n","\u003Cp\u003EI have gone into Snowflake and created the database and schema we are using with the same credentials I am using to deploy in SnowConvert. Now I will try it again. This time, let&rsquo;s just start with the tables. I will select the tables:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deployStatusPane.png\" alt=\"deployStatusPane\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis time, the results will show that we were able to successfully deploy some objects.\n\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentResults2.png\" alt=\"deployResults2\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis can be validated by querying the database in Snowflake:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-sql\"\u003Edescribe schema adventureworks.dbo\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EI can see each of the tables that were just deployed. Now, let&rsquo;s do the views and functions to complete the deployment. When we do, some of the functions and views deploy successfully, but there are still some errors.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s see what is in those errors. If you return to the menu and hover over an object in the deployment column, it will tell you a bit more about why the object was not deployed:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode22.png\" alt=\"issuesCode22\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIn this case, the view that we are hovering over has a dependency on a function. This function exists in this deployment, but SnowConvert is using a &ldquo;brute force&rdquo; method of deployment. This brute force method of deployment will lead to errors oftentimes because it is trying different combinations of deployment methods as opposed to ordering the deployment based on dependencies.\u003C/p\u003E\n","\u003Cp\u003EWe can manually resolve this in the SnowConvert application though. Let&rsquo;s start by seeing which object may have the first level of dependencies:\u003C/p\u003E\n","\u003Cp\u003EIt looks like the view vTimeSeries is dependent on function \u003Cstrong\u003EudfBuildISO8601Date\u003C/strong\u003E, and \u003Cstrong\u003EudfBuildISO8601Date\u003C/strong\u003E is dependent on function \u003Cstrong\u003EudfTwoDigitZeroFill\u003C/strong\u003E.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s try the last two objects in order with the function first, followed by the last view.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode25.png\" alt=\"issuesCode25\"\u003E\u003C/p\u003E\n","\u003Cp\u003EAnother failure! Looks like the timestamp type that was moved from SQL Server to Snowflake was not compatible. So we need to change it to be compatible. If we look at a side by side comparison in VS Code of the source SQL Server code and the Snowflake code, we can see that the code in SQL Server is only returning a &ldquo;datetime&rdquo; value, not a TIMESTAMP_NTZ(3).\u003C/p\u003E\n","\u003Cp\u003ESo let&rsquo;s change the TIMESTAMP_NTZ(3) value to a TIMESTAMP_NTZ(9), but make a note to review this during testing.\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-sql\"\u003ECREATE OR REPLACE FUNCTION dbo.udfBuildISO8601Date (YEAR INT, MONTH INT, DAY INT)\nRETURNS TIMESTAMP_NTZ(9)\nLANGUAGE SQL\nCOMMENT = '{ &quot;origin&quot;: &quot;sf_sc&quot;, &quot;name&quot;: &quot;snowconvert&quot;, &quot;version&quot;: {  &quot;major&quot;: 1,  &quot;minor&quot;: 6,  &quot;patch&quot;: &quot;0.0&quot; }, &quot;attributes&quot;: {  &quot;component&quot;: &quot;transact&quot;,  &quot;convertedOn&quot;: &quot;05/26/2025&quot;,  &quot;domain&quot;: &quot;snowflake&quot; }}'\nAS\n$$\n\tSELECT\n\t\tTO_TIMESTAMP_NTZ(CAST(YEAR AS VARCHAR) || '-' || dbo.udfTwoDigitZeroFill(MONTH) || '-' || dbo.udfTwoDigitZeroFill(DAY) || 'T00:00:00', 'MM-DD-YYYY')\n$$;\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003ESave the file and refresh the SnowConvert application. Let&rsquo;s try to deploy the last function again.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode27.png\" alt=\"issuesCode27\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESuccess! We have all of our tables and functions deployed. There was one view that was not able to be deployed, but it was dependent on the functions being deployed. Let&rsquo;s a take a look at that view right now. Let&rsquo;s try to deploy the last view:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode28.png\" alt=\"issuesCode28\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESuccess again! We have officially deployed all of the schema to Snowflake. There&rsquo;s one more thing we can do in the SnowConvert application, and that is move the data from the original SQL Server database into Snowflake. Select &ldquo;GO TO DATA MIGRATION&rdquo; at the bottom right of the screen.\u003C/p\u003E\n","\u003Ch2\u003EData Migration\u003C/h2\u003E\n","\u003Cp\u003EThe Data Migration page will give you the same catalog that you can see on the previous screens. For the migration of the data, note that only the data in tables will be migrated. The rest of the DDL is already present, but there is no &ldquo;data&rdquo; present in a view or a function, so the tables will be what is migrated. Also note that no data can be migrated to a table where the DDL wasn&rsquo;t already successfully deployed.\u003C/p\u003E\n","\u003Cp\u003ETo migrate the data, select all of the tables (or you can choose the entire database or schema) and choose &ldquo;MIGRATE DATA&rdquo; at the bottom of the application:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md1.png\" alt=\"md1\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWhile the data is migrating, it will let you know how it&rsquo;s doing:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md2.png\" alt=\"md2\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIt&rsquo;s important to note that SnowConvert is migrating the data by passing it through your local machine. It is running a select statement on the data in the original object, then doing a COPY INTO to the data in the Snowflake object. This means that the speed of the data transfer will be directly affected by your connection from the local machine where you are running SnowConvert.\u003C/p\u003E\n","\u003Cp\u003EIf you are moving a large amount of data, this will not be the most efficient way to move that data. You should consider using another data movement or ETL solution. However, in this scenario, this works just fine for us. There are only 30 tables.\u003C/p\u003E\n","\u003Cp\u003ENote that if you are connected via SSO, Snowconvert may take you to a login page in your browser. If you have MFA setup on your Snowflake account, you will be taken to your authenticator to validate your connection.\u003C/p\u003E\n","\u003Cp\u003ELooks like we did not receive any error message for this migration. Let&rsquo;s expand the object catalog to see if all the tables were migrated:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md3.png\" alt=\"md3\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESelect &ldquo;VIEW DATA MIGRATION RESULTS&rdquo; to see a summary of what just happened.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md4.png\" alt=\"md5\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis will show you each table that you just migrated and the corresponding table in Snowflake. For a basic validation of what just happened, you can review the last two columns of the Data Migration Results. These are the row counts for the source table and the count of rows that were loaded into the table in Snowflake.\u003C/p\u003E\n","\u003Cp\u003EYou can go into Snowflake and write a queries to check to see if this will work. Here is a simple &ldquo;select *&rdquo; on the largest table we loaded:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md6.png\" alt=\"md6\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWhere once there was no data, now&hellip; there is data. At this point, we would start running validation tests on the data to confirm that everything is the same. For this POC, we only have a few tests that we want to validate, but they are all related to the pipelines we are running. Let&rsquo;s do our validations once we move the pipelines.\u003C/p\u003E\n","\u003Ch3\u003EReview\u003C/h3\u003E\n","\u003Cp\u003EBefore we get to our pipelines, let&rsquo;s take a quick look back.\u003C/p\u003E\n","\u003Cp\u003EIf we review the steps that we did, let&rsquo;s see how they fit in with our migration process:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003ESetup our project in SnowConvert\u003C/li\u003E\u003Cli\u003EExtracted the schema from the source database\u003C/li\u003E\u003Cli\u003ERan SnowConvert&rsquo;s assessment and conversion engine on the extracted schema\u003C/li\u003E\u003Cli\u003EReviewed the output reporting to better understand what we have\u003C/li\u003E\u003Cli\u003EReviewed what could not be converted in the VS Code Extension\u003C/li\u003E\u003Cli\u003EGenerated new code in the VS Code Extension\u003C/li\u003E\u003Cli\u003EDeployed the code to Snowflake\u003C/li\u003E\u003Cli\u003EMoved the data from the source to Snowflake\u003C/li\u003E\u003Cli\u003EAll the while, we were able to track things in our object inventory.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EThis is what we have just walked through. These steps will generally be recommended regardless of how you used SnowConvert.\u003C/p\u003E\n","\u003Cp\u003EAnd you can generally see that most of these activities fit our \u003Cstrong\u003Eassess\u003C/strong\u003E -&gt; \u003Cstrong\u003Econvert\u003C/strong\u003E -&gt; \u003Cstrong\u003Evalidate\u003C/strong\u003E flow:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003ESetup our project in SnowConvert so we can better assess what we have\u003C/li\u003E\u003Cli\u003EExtracted the schema from the source database so we can see what kind of DDL we are working with\u003C/li\u003E\u003Cli\u003ERan SnowConvert&rsquo;s assessment and conversion engine on the extracted schema to analyze what we have for our initial assessment, and generate the output converted code\u003C/li\u003E\u003Cli\u003EReviewed the output reporting to better understand what we have to build our assessment of what we have\u003C/li\u003E\u003Cli\u003EReviewed what could not be converted in the VS Code Extension to assess what work needs to be done\u003C/li\u003E\u003Cli\u003EGenerated new code in the VS Code Extension to resolve any errors in the initial conversion\u003C/li\u003E\u003Cli\u003EDeployed the code to Snowflake to validate that the schema works\u003C/li\u003E\u003Cli\u003EMoved the data from the source to Snowflake to convert our data into Snowflake data\u003C/li\u003E\u003Cli\u003EAll the while, we were able to track things in our object inventory to better understand where we are in the migration process.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EAs we continue to walk through the migration process, let&rsquo;s take a look at our data pipelines.\u003C/p\u003E\n","\u003Ch2\u003EPipeline Assessment\u003C/h2\u003E\n","\u003Cp\u003EMoving the logic and data in a data warehouse is essential to getting an operational database on a new platform. But to take advantage of the new platform in a functional way, any pipelines running moving data in or out of that data platform need to be repointed or replatformed as well. This can often be challenging as there are usually a variety of pipelines being used. This section of the lab will focus on just one for which Snowflake can provide some acceleration. But note that new ETL and pipeline accelerators are constantly being developed.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s talk about the pipeline and the notebook we are moving.\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EThe pipeline is a Spark script that is reading an accessible file generated by an older POS system in a local directory at regular intervals. (This will be referred to as the pipeline script.)\u003C/li\u003E\u003Cli\u003EThe notebook is a reporting notebook that reads from the existing SQL Server database and reports on a few summary metrics. (This will be referred to as the reporting notebook.)\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EBoth of these use Spark and access the SQL Server database. So our goal is essentially to move the operations in Spark into Snowpark. Let&rsquo;s see how we would do that using Snowpark Migration Accelerator (SMA). Recall that while many of the steps we will walk through are similar to what we just did with SnowConvert, we are still essentially working through the \u003Cstrong\u003Eassessment\u003C/strong\u003E -&gt; \u003Cstrong\u003Econversion\u003C/strong\u003E -&gt; \u003Cstrong\u003Evalidation\u003C/strong\u003E flow.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003ENotes on this Lab Environment\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EThis lab uses the Snowpark Migration Accelerator and the Snowflake VS Code Extension. But to make the most of this, you will need to run Python with a PySpark. The simplest way to start this would be to start an environment with \u003Ca href=\"https://www.anaconda.com/docs/getting-started/anaconda/main\"\u003Ethe anaconda distribution\u003C/a\u003E. This will have most of the packages needed to run the code in this lab.\u003C/p\u003E\n","\u003Cp\u003EYou will still need to make available the following resources:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EPython Libraries (\u003Ca href=\"https://pypi.org/project/pyspark/\"\u003EPySpark\u003C/a\u003E, \u003Ca href=\"https://pypi.org/project/snowflake-snowpark-python/\"\u003ESnowpark Python\u003C/a\u003E, and \u003Ca href=\"https://pypi.org/project/snowflake/\"\u003ESnowflake\u003C/a\u003E)\u003C/li\u003E\u003Cli\u003EVS Code Extensions (\u003Ca href=\"https://marketplace.visualstudio.com/items?itemName=snowflake.snowflake-vsc\"\u003ESnowflake\u003C/a\u003E, \u003Ca href=\"https://marketplace.visualstudio.com/items?itemName=ms-python.python\"\u003EPython\u003C/a\u003E, and \u003Ca href=\"https://marketplace.visualstudio.com/items?itemName=ms-toolsai.jupyter\"\u003EJupyter\u003C/a\u003E)\u003C/li\u003E\u003Cli\u003EOther supporting elements like the \u003Ca href=\"https://learn.microsoft.com/en-us/sql/connect/jdbc/download-microsoft-jdbc-driver-for-sql-server?view=sql-server-ver17\"\u003EPySpark JDBC Driver for SQL Server\u003C/a\u003E.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EHaving said all of this, you can still run this lab with just a Snowflake account, the SMA, and the Snowflake VS Code Extension. You will not be able to run everything that connects to a local machine, but you will be able to use all of the converted elements in Snowflake.\u003C/p\u003E\n","\u003Cp\u003ENow let&rsquo;s get started by assessing what we have.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EExtraction\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EFor the purpose of this lab, we will assume that the notebook and script file that we are converting are already accessible as files. \u003Ca href=\"https://docs.snowconvert.com/sma/use-cases/pipeline-migration-lab/pipeline-lab-assessment\"\u003EDownload those files from the Snowpark Migration Accelerator's docs site\u003C/a\u003E.\u003C/p\u003E\n","\u003Cp\u003ENote that there is also a data file as well: &lsquo;customer_update.csv&rsquo;. This is a sample of the file being generated locally by the Point of Sale (POS) system that AdventureWorks is currently using. While that system is also being updated, this Proof of Concept (POC) is focused on making the existing pipeline work with Snowpark instead of Spark.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s take each of these files, and drop them into a single directory on our local machine:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_source_code_directory_01.png\" alt=\"sma_source_code_directory_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIt would be recommended to create a project directory. This can be called whatever you like, but as a suggestion for this lab, let&rsquo;s go with \u003Cstrong\u003Espark_adw_lab\u003C/strong\u003E. This means we would create a folder with the name spark_adw_lab, then create another folder in that directory called source_files (the path being something like \u003Cstrong\u003E/your/accessible/directory/spark_adw_lab/source_files\u003C/strong\u003E). This isn&rsquo;t required, but will help keep things organized. The SMA will scan any set of subdirectories as well, so you could add specific pipelines in a folder and notebooks in another.\u003C/p\u003E\n","\u003Cp\u003ENote that in general, if you are extracting files that are being orchestrated by a specific tool, you may need to export them. If you are using notebooks in the Cloud elsewhere, you can export those as .ipynb files just as the jupyter notebook we are going to run through the SMA today.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EAccess\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ENow that we have our source files in an accessible directory, it is time to run the SMA.\u003C/p\u003E\n","\u003Cp\u003EIf you have not already downloaded it, the SMA is accessible from \u003Ca href=\"/en/migrate-to-the-cloud/migration-accelerator/\"\u003Ethe Snowflake website\u003C/a\u003E. It is also accessible from the Migrations page in SnowSight in your Snowflake account:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma1.png\" alt=\"sma1\"\u003E\u003C/p\u003E\n","\u003Cp\u003EOnce you download the tool, install it! There is more information on \u003Ca href=\"/en/migrate-to-the-cloud/migration-accelerator/\"\u003Einstalling the SMA\u003C/a\u003E in the SMA documentation.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EUsing the Snowpark Migration Accelerator\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EOnce you have installed the tool, open it! When you launch the SMA, it will look very similar to its partner tool, SnowConvert. Both of these tools are built on a similar concept where you input code files into the tool and it runs. We have seen that SnowConvert can take the DDL and data directly from the source and input it directly into Snowflake. The SMA does not do this. It only takes in code files as a source and outputs those files to something that is compatible with Snowflake. This is primarily because the tool does not know how a user will orchestrate their spark code.\u003C/p\u003E\n","\u003Cp\u003EOnce you have launched the tool, It will ask you if you would like to create a new project or open an already existing one:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma2.png\" alt=\"sma2\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis will take you to the project creation screen:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma3.png\" alt=\"sma3\"\u003E\u003C/p\u003E\n","\u003Cp\u003EOn this screen, you will enter the relevant details for your project. Note that all fields are required. For this project, you could enter something similar to:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EProject Name: \u003Cstrong\u003ESpark ADW Lab\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003EInput Folder Path: \u003Cstrong\u003E/your/accessible/directory/spark_adw_lab/source_files\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003EOutput Folder Path (the SMA will auto generate a directory for the output, but you can modify this): \u003Cstrong\u003E/your/accessible/directory/spark_adw_lab/source_files_output\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003EEmail Address: \u003Cstrong\u003Eyour.name@your_domain.com\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003ECustomer&rsquo;s Company: \u003Cstrong\u003EYour Organization\u003C/strong\u003E\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EA couple of notes about this project creation screen:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EThe email and company fields are to help you track projects that may be ongoing. For example, at any large SI, there may be multiple email addresses and multiple organizations on behalf of whom a single user may run the SMA. This information is stored in the project file created by the SMA.\u003C/li\u003E\u003Cli\u003EThere is a hidden field for SQL. Note that the SMA can scan/analyze SQL, but it does not convert any SQL.It also can only identify SQL in the following circumstances:\n\u003Cul\u003E\u003Cli\u003ESQL that is in .sql files\u003C/li\u003E\u003Cli\u003ESQL that is in SQL cells in a Jupyter Notebook\u003C/li\u003E\u003Cli\u003ESQL that is passed as a single string to a spark.sql statement.\u003C/li\u003E\u003C/ul\u003E\n\u003C/li\u003E\u003Cli\u003EWhile this SQL capability can be helpful to determine where there is incompatible SQL with Snowflake, it is not the primary use for the SMA. More support for Spark SQL and HiveQL are coming soon.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EOnce you&rsquo;ve entered all of your project information, for this lab, we are going to \u003Cstrong\u003Eskip\u003C/strong\u003E the assessment phase. (What&hellip; aren&rsquo;t we building an assessment?) If you do not want to convert any code, running an assessment can be helpful as it will allow you to get the full set of reports generated by the SMA. You can then navigate through those or share them with others in your organization while not creating extra copies of the converted code. However, all of these same assessment reports are also generated during a conversion. So we will skip assessment mode for now and go to conversion by selecting &ldquo;SAVE &amp; SKIP ASSESSMENT&rdquo; in the bottom right corner of the application.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma4.png\" alt=\"sma4\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis will take you to the conversion screen. On this screen you will again see the input and output directory fields, but those will have already been populated by what you have entered in the project creation page. The only new field here will be to enter an access code. Access codes are freely available, but must still be requested an active to use the Snowpark Migration Accelerator (SMA). And while the mechanism for generating these access codes is similar to SnowConvert, the access codes for SnowConvert will not work with the SMA. You will have to get another one.\u003C/p\u003E\n","\u003Cp\u003EYou can request an access code by selecting &ldquo;Inquire about an access code&rdquo; out to the side of the &ldquo;Enter access code&hellip;&rdquo; field:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma5.png\" alt=\"sma5\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWhen you select this, a pop up menu will appear asking you who you are so an access code can be generated:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma6.png\" alt=\"sma6\"\u003E\u003C/p\u003E\n","\u003Cp\u003EFill out all of the fields shown above, and ensure that you enter a valid email address. In the project creation screen earlier, you entered an email address to associate with the project you were creating. However, nothing was actually sent to that email. That was only to track your project locally. This form will trigger an access code to be sent to the email you enter.\u003C/p\u003E\n","\u003Cp\u003EOnce you submit the form, you should receive an email with an access code shortly. The email will come from sma-notifications@snowflake.com, and it will look something like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma7.png\" alt=\"sma7\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIn the image above, where it says &quot;\u003Cstrong\u003Eyour access code here\u003C/strong\u003E&quot;, you should see a series of numbers, letters, and dashes. Copy that string, and paste it in the access code field for the SMA:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma8.png\" alt=\"sma8\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWhen you paste the value into the box, the SMA will validate the access code. A successful validation will show the access code details below the access code dialog box:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma9.png\" alt=\"sma9\"\u003E\u003C/p\u003E\n","\u003Cp\u003ETo validate the access code, the SMA will call out to the Snowflake licensing API. If you are not connected to the internet, the tool will not be able to validate the access code and you will get an error message. If you need to run the tool in a completely offline environment, please reach out to \u003Ca href=\"mailto:sma-support@snowflake.com\"\u003Esma-support@snowflake.com\u003C/a\u003E for help validating an access code.\u003C/p\u003E\n","\u003Cp\u003ENow that the access code has been validated, you can take a look at the conversion settings:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_go_to_conversion_settings_01.png\" alt=\"sma_go_to_conversion_settings_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThere is one setting that will simplify the output of this hands on lab, which would be to \u003Cstrong\u003Edisable\u003C/strong\u003E the attempted conversion of pandas dataframes to the Snowpark API:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_conversion_settings_01.png\" alt=\"sma_conversion_settings_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis one setting is currently being updated, so there will be a lot of additional warnings added if this option is not deselected. These can be worked though, but this will add noise to this lab. Most of the functions related to pandas dataframes can be used as part of the modin implementation of pandas, so a simple import call change should suffice for now.\u003C/p\u003E\n","\u003Cp\u003EYou can look at the other settings, but we will leave them as is. It&rsquo;s important to note that there is a testing library that the output code is compatible with called \u003Cstrong\u003ESnowpark Checkpoints\u003C/strong\u003E. There are settings related to this, but we will not alter them in this lab. Select &ldquo;CLOSE&rdquo; in the dialog box to save and close your settings.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_conversion_settings_02.png\" alt=\"sma_conversion_settings_02\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThe &ldquo;START CONVERSION&rdquo; option will be available in the bottom right corner of the application. Let&rsquo;s start the conversion by selecting this option.\u003C/p\u003E\n","\u003Cp\u003EThe next screen will show the progress of the conversion:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma10.png\" alt=\"sma10\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELike SnowConvert, the SMA is building a semantic model of the entire codebase in the input directory. It is building relationships between code elements, sql objects, and other referenced artifacts, and creating the closest output it can to a functional equivalent for Snowflake. This primarily means converting references from the Spark API to the Snowpark API. The SMA&rsquo;s engineering team is a part of the Snowpark engineering team, so most transformations that take place have been built into the Snowpark API, so the changes may seem minor. But the wealth of assessment information that is generated by the SMA allows a migration project to really get moving forward. An in-depth look at all of the generated assessment information will have to take place elsewhere because the SMA has likely finished this conversion in the time it took to read this paragraph.\u003C/p\u003E\n","\u003Cp\u003EWhen the SMA has finished, the &ldquo;VIEW RESULTS&rdquo; option will be available in the bottom right corner:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma11.png\" alt=\"sma11\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThe results page, shockingly enough, will show the results.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_results_page_01.png\" alt=\"sma_results_page_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThe results page has some &ldquo;Readiness Scores&rdquo; that are very simplified metrics on how &ldquo;ready&rdquo; this codebase is for Snowflake. We will review the results next, but note that running the Snowpark Migration Accelerator is the easy part. Note that this is just an &ldquo;accelerator&rdquo;. It is not a silver bullet or a hands-off automation tool. Pipelines that connect to one data source and output to another are not fully migrated by this tool will always need more attention than a straight SQL-to-SQL migration of DDL as is done by SnowConvert. But Snowflake is continuously working towards making this as simple as possible.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EInterpreting the Output\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EThe SMA, even more so than SnowConvert, generates a large amount of assessment information. It can be difficult to parse through the results. There are many different directions you could go depending on what you want to achieve. We'll try to cut through some of this given the simplicity of the scenario, but note that all of this data can be valualbe when planning a migration, particularly a large migration.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EReadiness Scores\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EWith that in mind, let&rsquo;s take a look at the first part of the output that you will see in the application: the readiness scores. There will be multiple readiness scores and you can expand on each one of them to better understand what is captured by that readiness score.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_scores_01.png\" alt=\"sma_scores_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EEach readiness score is a very basic calculation of the count of functions or elements in an API that are supported in Snowpark/Snowflake divided by the count of all functions or elements related to that API for this execution. The calculation showing you how the score is calculated is shown when you expand the window. You can also learn more about how to interpret the readiness scores by selecting &ldquo;How to read through the scores&rdquo; near the top left corner of this window.\u003C/p\u003E\n","\u003Cp\u003EThis execution has a Spark API Readiness Score of \u003Cstrong\u003E97.92%\u003C/strong\u003E. (Please note that yours may be different! These tools are updated on a biweekly basis and there may be a change as compatibility between the two platforms is ever evolving.) This means that 97.92% of the references to the Spark API that the tool identified are supported in Snowflake. &ldquo;Supported&rdquo; in this case means that there could be a similar function that already exists or that the SMA has created a functionally equivalent output. The higher this score is, the more likely this code can quickly run in Snowflake.\u003C/p\u003E\n","\u003Cp\u003EThere are other readiness scores and you may see more than what is shown in the lab as the readiness scores do change over time. This lab won&rsquo;t walk through each of them, but note that a low score will always be worth investigating.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003ECode Analyzed\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EJust below each of the readiness scores, will be a small indicator that lets you know if there was any code that could not be processed:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_analyzed_01.png\" alt=\"sma_code_analyzed_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis number represents the \u003Cstrong\u003Epercentage of files\u003C/strong\u003E that were fully parsed. If this number is less than 100%, then there is some code that the SMA could not parse or process. This is the first place you should start looking to resolve problems. If it&rsquo;s less than 100%, you should see where the parsing errors occurred by looking at the issue summary. This is the first place you should look when working through the SMA&rsquo;s output because it&rsquo;s the only one where it might make sense to run the tool again if a large amount of code was not able to be scanned.\u003C/p\u003E\n","\u003Cp\u003EIn this case, we only have \u003Cstrong\u003E50%\u003C/strong\u003E of our workload successfully parsed. Tragic. Now, this might seem like something we should panic about, but let&rsquo;s not be too quick to judge. We only have 2 files, and we don&rsquo;t yet know how many parsing errors we have.\u003C/p\u003E\n","\u003Cp\u003ERegardless of the result of this number, the last place we will visit on this page is the \u003Cstrong\u003EIssue Summary\u003C/strong\u003E. Scroll down in the application until you see this summary:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_issue_summary_01.png\" alt=\"sma_issue_summary_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EIssue Summary\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EIssues are one of the key elements of both SnowConvert and the SMA. Each tool is attempting to create a functional equivalent output based on the input that it receives, but no conversion is 100% automated. These tools know this, and mark everything that cannot be converted or even might need extra attention with an issue. These issues are summarized in a spreadsheet, but are also written as comments directly into the output code.\u003C/p\u003E\n","\u003Cp\u003EThe issue summary in the UI highlights issues that were found in this execution of the tool. These issues are often referred to with the acronym EWI (error, warning, and issue). Similar, but not identical to SnowConvert, the SMA generates three types of issues:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003EParsing Error\u003C/strong\u003E - This type of issue is considered critical and will need you deal with it immediately. Because of the way the SMA works, having code that does not parse, could mean missing information in the reports and missing conversion piece as well.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EConversion Error\u003C/strong\u003E - This is something the SMA recognizes (or at least thinks that it recognizes), but it cannot convert for one reason or another. These errors usually have very specific issue codes and should be dealt with next.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EWarning\u003C/strong\u003E - These error codes identify code that the SMA did convert or is something that has an equivalent in Snowpark/Snowflake, but there may be issues when you do testing. There may not be 100% functional equivalence.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EThere is more information on issue types on \u003Ca href=\"https://docs.snowconvert.com/sma/issue-analysis/issue-code-categorization\"\u003Ethe SMA documentation page\u003C/a\u003E, but in the issue summary shown above, one can find the code, count, level, and description for the unique issues present. Even if there are a lot of issues present, the fewer unique issues there are, the more likely they can be dealt with programmatically. For more information on each unique issue code, you can click the code in the UI. This will take you to the SMA documentation page for that specific issue.\u003C/p\u003E\n","\u003Cp\u003ELooks like we have some conversion errors, warnings, and 1 parsing error. This means there was 1 thing the tool could not read. (Note that if you get a lot of error codes that start with PND, then you may not have deselected that option in the conversion settings. Now worries, if you see those, you can ignore them.)\u003C/p\u003E\n","\u003Cp\u003ERegardless of how many issues you have, it is always recommended to explore the detailed issues file if you&rsquo;re ready to start migrating. This is a csv file that is stored locally on the machine where you ran the SMA. You can find this file by selecting the &ldquo;VIEW REPORTS&rdquo; option in the bottom right of the SMA:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_view_reports_01.png\" alt=\"sma_view_reports_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis will take you to the local directory that has all of the reports&hellip; and as of this writing, there are a lot of reports generated by the SMA:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_reports_folder_01.png\" alt=\"sma_reports_folder_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EEach of these reports has some valuable information in it depending on how you are using the SMA. For now, we will only look at the issues.csv file, but note there is more information on EVERY report and inventory generated by the SMA \u003Ca href=\"https://docs.snowconvert.com/sma/user-guide/assessment/output-reports\"\u003Ein the SMA documentation\u003C/a\u003E.\u003C/p\u003E\n","\u003Cp\u003EWhen you open the issues file, it will look something like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_issue_report_01.png\" alt=\"sma_issue_report_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote the schema of this report. It tells you the issue code, a description of the issue, the type of issue (category), the file each issue is in, the line number of the file each issue is in, and provides a link to the documentation page for that specific issue. All of this is helpful information when navigating through the issues.\u003C/p\u003E\n","\u003Cp\u003EYou can pivot this by file to see what type of issue you have by file to find where the parsing error is for this workload:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_issue_report_02.png\" alt=\"sma_issue_report_02\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELooks like we only have a few issues, and our parsing error is in the pipeline python script. That&rsquo;s where we want to start.\u003C/p\u003E\n","\u003Cp\u003ENormally, we would take a look at one other report in our Reports directory, the \u003Cstrong\u003EArtifactDependencyInventory.csv\u003C/strong\u003E file. But this is such a small execution, let&rsquo;s take a look at what&rsquo;s actually in these output files now, and see if we can&rsquo;t get it to run in (or with) Snowflake.\u003C/p\u003E\n","\u003Ch2\u003EPipeline Conversion\u003C/h2\u003E\n","\u003Cp\u003EThe SMA has &ldquo;converted&rdquo; our scripts, but has it really? What it has actually done is converted all references from the Spark API to the Snowpark API, but what it has not done is to replace the connections that may exist in your pipelines.\u003C/p\u003E\n","\u003Cp\u003EThe SMA&rsquo;s power is in the assessment reporting that it does as the conversion is tied to converting references from the Spark API to the Snowpark API. Note that the conversion of these references will not be enough to run any data pipeline. You will have to ensure that the pipeline&rsquo;s connections are resolved manually. The SMA cannot assume to know connection parameters or other elements that are likely not available to be run through it.\u003C/p\u003E\n","\u003Cp\u003EAs with any conversion, dealing with the converted code can be done in a variety of ways. The following steps are how we would \u003Cstrong\u003Erecommend\u003C/strong\u003E that you approach the output of the conversion tool. Like SnowConvert, the SMA requires attention to be paid to the output. No conversion will ever be 100% automated. This is particularly true for the SMA. Since the SMA is converting references from the Spark API to the Snowpark API, you will always need to check how those references are being run. It does not attempt to orchestrate the successful execution of any script or notebook run through it.\u003C/p\u003E\n","\u003Cp\u003ESo we&rsquo;ll follow these steps to work through the output of the SMA that will be slightly different than SnowConvert:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003EResolve All Issues\u003C/strong\u003E: &ldquo;Issues&rdquo; here means the issues generated by the SMA. Take a look at the output code. Resolve parsing errors and conversion errors, and investigate warnings.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EResolve the session calls\u003C/strong\u003E: How the session call is written in the output code depends on where we are going to run the file. We will resolve this for running the code file(s) in the same location as they were originally going to be run, and then for running them in Snowflake.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EResolve the Input/Outputs\u003C/strong\u003E: Connections to different sources cannot be resolved entirely by the SMA. There are differences in the platforms, and the SMA will usually disregard this. This also is affected by where the file is going to be run.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EClean up and Test\u003C/strong\u003E! Let&rsquo;s run the code. See if it works. We will be smoke testing in this lab, but there are tools to do more extensive testing and data validation including Snowpark Python Checkpoints.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ESo let&rsquo;s take a look at what this looks like. We&rsquo;re going to do this with two approaches: the first approach is to run this in Python on the local machine (as the source script is running). The second would be to do everything in Snowflake&hellip; in Snowsight, but for a data pipeline reading from a local source, this will not be 100% possible in Snowsight. That&rsquo;s ok though. We are not converting the orchestration of this script in this POC.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s start with the pipeline script file, and get to the notebook in the next section.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EResolve Issues\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s open our source and our output code in a code editor. You can use any code editor of your choice, but as has been mentioned multiple times, Snowflake would recommend using \u003Cstrong\u003EVS Code with the Snowflake Extension\u003C/strong\u003E. Not only does the Snowflake Extension help navigate through the issues from SnowConvert, but can also run \u003Cstrong\u003ESnowpark Checkpoints\u003C/strong\u003E for Python, which would help with testing and root cause analysis (though just barely out of scope for this lab).\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s open the directory that we originally created in the project creation screen (Spark ADW Lab) in VS Code:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_vscode_files_01.png\" alt=\"sma_vscode_files_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that the \u003Cstrong\u003EOutput\u003C/strong\u003E directory structure will be the same as the input directory. Even the data file will be copied over despite no conversion taking place. There will also be a couple of \u003Cstrong\u003Echeckpoints.json\u003C/strong\u003E files that will be created by the SMA. These are json files that contain instructions for the Snowpark Checkpoints extension. The Snowflake extension can load checkpoints into both the source and output code based on the data in those files. We will ignore them for now.\u003C/p\u003E\n","\u003Cp\u003EFinally, let&rsquo;s compare the input python script with the converted one in the output script.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_01.png\" alt=\"sma_code_editor_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis is a very basic side-by-side comparison with the original Spark code on the left and the output Snowpark compatible code on the right. Looks like some imports were converted as well as the session call(s). We can see an EWI at the bottom of the image above, but let&rsquo;s not start there. We need to find the parsing error before we do anything else.\u003C/p\u003E\n","\u003Cp\u003EWe can search the document for the error code for that parsing error that was shown in both the UI and the issues.csv: \u003Cstrong\u003ESPRKPY1101\u003C/strong\u003E.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_02.png\" alt=\"sma_code_editor_02\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELooking for where this EWI shows up in the pipeline_dimcustomer.py, we can see that the parsing error was present in the very last line of the source code.\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Conversion Input.\nsome rogue code that doesn't make any sense!\n\n# Conversion Output.\nsome\n# EWI: SPRKPY1101 =&gt; Unrecognized or invalid CODE STATEMENT @(131, 6). Last valid token was 'some' @(131, 1), failed token 'rogue' @(131, 6)\n#     rogue code that doesn't make any sense!\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003ELooks like this parsing error was because of&hellip; &ldquo;some rogue code that doesn&rsquo;t make any sense!&rdquo;. This line of code is at the bottom of the pipeline file. This is not unusual to have extra characters or other elements in a code file as part of an extraction from a source. Note have the SMA detected that this was not valid Python code, and it generated the parsing error.\u003C/p\u003E\n","\u003Cp\u003EYou can also see how the SMA inserts both the error code and the description into the output code as a comment where the error occurred. This is how all error messages will appear in the output.\u003C/p\u003E\n","\u003Cp\u003ESince this is not valid code, it is at the end of the file, and there is nothing else that was removed as a result of this error, the original code and the comment can safely be removed from the output code file.\u003C/p\u003E\n","\u003Cp\u003EAnd now we&rsquo;ve resolved our first and most serious issue. Get excited.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s work through the rest of our EWIs in this file. We can search for &ldquo;EWI&rdquo; because we now know that text will appear in the comment every time there is an error code. (Alternatively, we could sort the issues.csv file and order the issues by severity&hellip; but that&rsquo;s not really necessary here.)\u003C/p\u003E\n","\u003Cp\u003EThe next one is actually just a warning, not an error. It&rsquo;s telling us that there was a function used that isn&rsquo;t always equivalent in Spark and Snowpark:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E#EWI: SPRKPY1067 =&gt; Snowpark does not support split functions with more than two parameters or containing regex pattern. See documentation for more info.\nsplit_col = split(df_uppercase['NAME'], '.first:')\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EThe description here though gives away that we probably don&rsquo;t have to worry about this. There are only two parameters being passed. Let&rsquo;s leave this EWI as a comment in the file, so we know to check for it when we are running the file later.\u003C/p\u003E\n","\u003Cp\u003EThe last one for this file is a conversion error saying that something is not supported:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_03.png\" alt=\"sma_code_editor_03\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis is the write call to the spark jdbc driver to write the output dataframe into SQL Server. Since this is part of the &ldquo;resolve all inputs/outputs&rdquo; step that we are going to deal with after we address our issues, we&rsquo;ll leave this for later. Note, however, that this error must be resolved. The previous one was just a warning and may still work with no change being made.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EResolving the Session Calls\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EThe session calls are converted by the SMA, but you should pay special attention to them to make sure they are functional. In our pipeline script, this is the before and after code:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_04.png\" alt=\"sma_code_editor_04\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThe SparkSession reference was changed to Session. You can see that reference change near the top of this file in the import statement as well:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Original PySpark Imports\nfrom pyspark.sql import SparkSession\n\n# New Snowpark Imports\nfrom snowflake.snowpark import Session\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003ENote in the image above, the variable assignment of the session call to &ldquo;spark&rdquo; is not changed. This is because this is a variable assignment. It is not necessary to change this, but if you&rsquo;d like to change the &ldquo;spark&rdquo; decorator to &ldquo;session&rdquo;, that would be more in line with what Snowpark recommends. (Note that the VS Code Extension &ldquo;SMA Assistant&rdquo; will suggest these changes as well.)\u003C/p\u003E\n","\u003Cp\u003EThis is a simple exercise, but it&rsquo;s worth doing. You can do a find and replace using VS Code&rsquo;s own search ability to find the references to &ldquo;spark&rdquo; in this file and replace them with session. You can see the result of this in the image below. The references to the &ldquo;spark&rdquo; variable in the converted code have been replaced with &ldquo;session&rdquo;:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_05.png\" alt=\"sma_code_editor_05\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWe also can remove something else from this session call. Since we are not going to be running &ldquo;spark&rdquo; anymore, we do not need to specify the driver path for the spark driver. So we can remove the config function entirely from the session call like this:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Old Converted output.\n# Spark Session\nsession = Session.builder.config('spark.driver.extraClassPath', driver_path) \\\n                    .app_name('SparkSQLServerExample', True) \\\n                    .getOrCreate()\n\n# New Converted Output\n# Snowpark Session\nsession = Session.builder.app_name('SparkSQLServerExample', True).getOrCreate()\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EMight as well convert it to a single line. The SMA couldn&rsquo;t be sure we didn&rsquo;t need that driver (although that seems logical), so it did not remove it. But now that we have our session call is complete.\u003C/p\u003E\n","\u003Cp\u003E(Note that the SMA also adds a &ldquo;query tag&rdquo; to the session. This is to help troubleshoot issues with this session or query later on, but this is completely optional to leave or remove.)\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003ENotes on the Session Calls\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EBelieve it or not that is all that we need to change in the code for the session call, but that&rsquo;s not all we need to do to create the session. This refers back to the original question that a lot of this depends on where you want to run these files. These original spark session calls used a configuration that was setup elsewhere. If you look at the original Spark session call it&rsquo;s looking for a config file that is being read into a pandas dataframe location at the start of this script file (this is actually true for our notebook file as well).\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_06.png\" alt=\"sma_code_editor_06\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESnowpark can function the same way, and this conversion assumes that is how this user will run this code. However, for the existing session call to work, the user would have to load all of the information for their Snowflake account into the local (or at least accessible) connections.toml file on this machine, and that the account they are attempting to connect to is set as the default. \u003Ca href=\"https://docs.snowflake.com/en/developer-guide/snowpark/python/creating-session#connect-by-using-the-connections-toml-file\"\u003EYou can learn more about updating the connections.toml file in the Snowflake/Snowpark documentation\u003C/a\u003E, but the idea behind it is that there is an accessible location that has the credentials. When a snowpark session is created, it is going to check this&hellip; unless the connection parameters are explicitly passed to the session call.\u003C/p\u003E\n","\u003Cp\u003EThe standard way to do this is to input the connection parameters directly as strings and call them with the session:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Parameters in a dictionary.\nconnection_parameters = {\n  &quot;account&quot;: &quot;&lt;your snowflake account&gt;&quot;,\n  &quot;user&quot;: &quot;&lt;your snowflake user&gt;&quot;,\n  &quot;password&quot;: &quot;&lt;your snowflake password&gt;&quot;,\n  &quot;role&quot;: &quot;&lt;your snowflake role&gt;&quot;,  # optional\n  &quot;warehouse&quot;: &quot;&lt;your snowflake warehouse&gt;&quot;,  # optional\n  &quot;database&quot;: &quot;&lt;your snowflake database&gt;&quot;,  # optional\n  &quot;schema&quot;: &quot;&lt;your snowflake schema&gt;&quot;,  # optional\n}\n\n# The session call\nsession = Session.builder.configs(connection_parameters).app_name(&quot;AdventureWorksSummary&quot;, True).getOrCreate()\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EAdventureWorks appears to have referenced a file with these credentials and called it. Assuming there is a similar file called 'snowflake_credentials.txt' that is accessible, then the syntax that would match that could look something like:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Load into a dataframe.\nsnow_creds = pd.read_csv('snowflake_credentials.txt', index_col=None, header=0)\n\n# Build the parameters.\nconnection_parameters = {\n  &quot;account&quot;: snow_creds.loc[snow_creds['Specific_Element'] == 'Account', 'Value'].item(),\n  &quot;user&quot;: snow_creds.loc[snow_creds['Specific_Element'] == 'Username', 'Value'].item(),\n  &quot;password&quot;: snow_creds.loc[snow_creds['Specific_Element'] == 'Password', 'Value'].item(),\n  &quot;role&quot;: &quot;&lt;your snowflake role&gt;&quot;,  # optional\n  &quot;warehouse&quot;: snow_creds.loc[snow_creds['Specific_Element'] == 'Warehouse', 'Value'].item(),  # optional\n  &quot;database&quot;: snow_creds.loc[snow_creds['Specific_Element'] == 'Database', 'Value'].item(),  # optional\n  &quot;schema&quot;: snow_creds.loc[snow_creds['Specific_Element'] == 'Schema', 'Value'].item(),  # optional\n}\n\n# Then pass the parameters to the configs function of the session builder.\nsession = Session.builder.configs(connection_parameters).app_name(&quot;AdventureWorksSummary&quot;, True).getOrCreate()\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EFor the purpose of the time limit on this lab, the first option may make more sense. \u003Ca href=\"https://docs.snowflake.com/en/developer-guide/snowpark/python/creating-session#connect-by-specifying-connection-parameters\"\u003EThere&rsquo;s more on this in the Snowpark documentation\u003C/a\u003E.\u003C/p\u003E\n","\u003Cp\u003ENote that for our notebook file to run inside of Snowflake using Snowsight, you wouldn&rsquo;t need to do any of this. You would just call the active session and run it.\u003C/p\u003E\n","\u003Cp\u003ENow it's time for the most critical component of this migration, resolving any input/output references.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EResolving the Inputs and Outputs\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ESo let&rsquo;s resolve our inputs and outputs now. Note that this is going to diverge based on whether you&rsquo;re running the files locally or Snowflake. for the python script, Let&rsquo;s make sure what we gain/lose by running directly inside of Snowsight: \u003Cstrong\u003Eyou cannot run the whole operation in Snowsight\u003C/strong\u003E (at least not currently). The local csv file is not accessible from Snowsight. You will have to load the .csv file into a stage manually. This will likely not be an ideal solution, but we can test the conversion by doing this.\u003C/p\u003E\n","\u003Cp\u003ESo we&rsquo;ll first prep this file to be run/orchestrated locally, and then to be run in Snowflake.\u003C/p\u003E\n","\u003Cp\u003ETo get the pipeline script&rsquo;s inputs and output resolved, we need to first identify them. They are pretty simple. This script seems to:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003Eaccess a local file\u003C/li\u003E\u003Cli\u003Eload the result into SQL Server (but now Snowflake)\u003C/li\u003E\u003Cli\u003Emoves the file to make way for the next one\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ESimple enough. So we need to replace each component of the code that does those things. Let&rsquo;s start with accessing the local file.\u003C/p\u003E\n","\u003Cp\u003EAs was mentioned at the start of this, it would be strongly suggested to rearchitect the Point of Sale System and the orchestration tools used to run this python script, to put the output file into a cloud storage location. Then you could turn that location into an External Table, and voila&hellip; you are in Snowflake. However, the current architecture says that this file is not in a cloud storage location and will stay where it is, so we need to create a way for Snowflake to access this file preserving the existing logic.\u003C/p\u003E\n","\u003Cp\u003EWe have options to do this, but we will create an internal stage and move the file into the stage with the script. We would then need to move the file in the local file system, and also move it in the stage. This can all be done with Snowpark. Let&rsquo;s break it down:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003Eaccessing a local file: Create an internal stage (it one doesn&rsquo;t exist already) -&gt; Load the file into the stage -&gt; Read the file into a dataframe\u003C/li\u003E\u003Cli\u003Eloading the result into SQL Server: Load the transformed data into a table in Snowflake\u003C/li\u003E\u003Cli\u003Emoves the file to make way for the next one: Move the local file -&gt; Move the file in the stage.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ELet&rsquo;s look at code that can do each of these things.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EAccess a Locally Accessible File\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EThis source code in Spark looks like this:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Spark read from a local csv file.\ndf = spark.read.csv('customer_update.csv', header=True, inferSchema=True)\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EAnd the transformed snowpark code (by the SMA) looks like this:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Snowpark read from a local csv file.\ndf = session.read.option(&quot;PARSE_HEADER&quot;, True).option(&quot;INFER_SCHEMA&quot;, True).csv('customer_update.csv')\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EWe can replace that with this with code that does the steps above:\u003C/p\u003E\n\u003Col\u003E\u003Cli\u003ECreate an internal stage (if one does not exist already). We will create a stage called 'LOCAL_LOAD_STAGE' and go through a few steps to make sure that the stage is ready for our file.\u003C/li\u003E\u003C/ol\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Additional import needed for this. (Check to see if this is already imported.)\nfrom snowflake.core.stage import Stage, StageEncryption, StageResource\n\n# name the stage we're going to use.\ntarget_stage_name = &quot;LOCAL_LOAD_STAGE&quot;\n\n# Check to see if this stage already exists.\nstages = session.sql(&quot;SHOW STAGES&quot;).collect()\ntarget_stages = [stage for stage in stages if stage['name'] == target_stage_name]\n\n# Create the stage if it does not already exist.\nif(len(target_stages) &lt; 1):\n    from snowflake.core import Root\n    from snowflake.core.stage import Stage, StageEncryption, StageResource\n    root = Root(session)\n    my_stage = Stage(name=&quot;LOCAL_LOAD_STAGE&quot;,encryption=StageEncryption(type=&quot;SNOWFLAKE_SSE&quot;))\n    root.databases[&quot;ADVENTUREWORKS&quot;].schemas[&quot;DBO&quot;].stages.create(my_stage)\n    print('%s created.'%(target_stage_name))\nelse:\n    print('%s already exists.'%(target_stage_name))\n\u003C/code\u003E\u003C/pre\u003E\n\u003Col start=\"2\"\u003E\u003Cli\u003ELoad the file into the stage.\u003C/li\u003E\u003C/ol\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Move the file.\nput_results = session.file.put(local_file_name=&quot;customer_update.csv&quot;,\n                    stage_location=&quot;ADVENTUREWORKS.DBO.LOCAL_LOAD_STAGE&quot;,\n                    overwrite=False,\n                    auto_compress=False)\n\n# Read the results.\nfor r in put_results:\n    str_output = (&quot;File {src}: {stat}&quot;).format(src=r.source,stat=r.status)\n    print(str_output) \n\u003C/code\u003E\u003C/pre\u003E\n\u003Col start=\"3\"\u003E\u003Cli\u003ERead the file into a dataframe. This is the part that the SMA actually converted. We need to specify that the location of the file is now the internal stage.\u003C/li\u003E\u003C/ol\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Location of the file in the stage.\ncsv_file_path = &quot;@LOCAL_LOAD_STAGE/customer_update.csv&quot;\n\n# Spark read from a local csv file.\ndf = session.read.option(&quot;PARSE_HEADER&quot;, True).option(&quot;INFER_SCHEMA&quot;, True).csv(csv_file_path)\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EThe result of that would look like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_07.png\" alt=\"sma_code_editor_07\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELet's move on to the next step.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003ELoad the result into Snowflake\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EThe original script wrote the dataframe into SQL Server. Now we are going to load into Snowflake. This is a much simpler conversion. The dataframe is already a Snowpark dataframe. This is one of the advantages of Snowflake. Now that the data is accessible to Snowflake, everything happens inside Snowflake.\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Original output from the conversion tool.\n# Write the DataFrame to SQL Server.\n#EWI: SPRKPY1002 =&gt; pyspark.sql.readwriter.DataFrameWriter.jdbc is not supported\ndf_transformed.write.jdbc(url=sql_server_url,\n              table='dbo.DimCustomer',\n              mode=&quot;append&quot;,\n              properties={\n                  &quot;user&quot;: sql_server_user,\n                  &quot;password&quot;: sql_server_password,\n                  &quot;driver&quot;: driver_path\n              })\n\n# Corrected Snowflake/Snowpark code.\ndf_transformed.write.save_as_table(&quot;ADVENTUREWORKS.DBO.DIMCUSTOMER&quot;, mode=&quot;append&quot;)\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003ENote that we may want to write to a temp table to do some testing/validation, but this is the behavior in the original script.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EMove the file to make way for the next one\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EThis is the behavior in the orginal script. We don't really need to make this happen in Snowflake, but we can to showcase the exact same functionality in the stage. This is done with an os command in the original file system. That does not depend on Spark and will remain the same. But to emulate this behavior in snowpark, we would need to move this file in the stage to a new directory.\u003C/p\u003E\n","\u003Cp\u003EThis can be done simply enough with the following python code:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# New filename.\noriginal_filepath = '@LOCAL_LOAD_STAGE/customer_update.csv'\nnew_filepath = '@LOCAL_LOAD_STAGE/old_versions/customer_update_%s.csv'%(today_time)\n\ncopy_sql = f&quot;COPY FILES INTO {new_filepath} FROM {original_filepath}&quot;\nsession.sql(copy_sql).collect()\nprint(f&quot;File copied from {original_filepath} to {new_filepath}&quot;)\n\nremove_sql = f&quot;REMOVE {original_filepath}&quot;\nsession.sql(remove_sql).collect()\nprint(f&quot;Original file {original_filepath} removed.&quot;)\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003ENote that this would not replace any of the existing code. Since we already want to keep the existing motion of moving the spark code to snowpark, we will leave the os reference. The final version will look like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_08.png\" alt=\"sma_code_editor_08\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENow we have the same motion completely done. Now let's do our final cleanup, and test this script out.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EClean up and Test\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EWe never looked at our import calls and we have config files that are not necessary at all. We could leave the references to the config files and run the script. In fact, assuming those config files are still accesible, then the code will still run. But if we're taking a close look at our importt statements, we might as well remove them. These files are represented by all of the code between the import statements and the session call:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_09.png\" alt=\"sma_code_editor_09\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThere&rsquo;s a few other things we should do:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003ECheck that all of our imports are still necessary. We can leave them for now. If there is an erorr, we can address it.\u003C/li\u003E\u003Cli\u003EWe also have one EWI that we left in there as a warning to check. So we want to make sure we inspect that output.\u003C/li\u003E\u003Cli\u003EWe need to make sure that our file system behavior mirrors that of the expected file system for the POS system. To do this, we should move the customer_update.csv file into the root folder you chose when first launching VS Code.\u003C/li\u003E\u003Cli\u003ECreate a directory called &ldquo;old_versions&rdquo; in that same directory. This should allow the os operations to run.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EFinally, if you are not comfortable running the code directly into the production table, you can create a copy of that table for this test, and point the load to that copy. Replace the load statement with the one below. Since this is a lab, feel free to write to the &ldquo;production&rdquo; table:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# In case we want to test.\ncreate_sql = &quot;&quot;&quot;\n                CREATE OR REPLACE TABLE ADVENTUREWORKS.DBO.DIMCUSTOMER_1\n                AS select * from ADVENTUREWORKS.DBO.DIMCUSTOMER;\n                &quot;&quot;&quot;\nsession.sql(create_sql).collect()\n\n# Write the DataFrame to SQL Server.\ndf_transformed.write.save_as_table(&quot;ADVENTUREWORKS.DBO.DIMCUSTOMER_1&quot;, mode=&quot;append&quot;)\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003ENow we're finally ready to test this out. We can run this script in Python to a testing table and see if it will fail. So run it!\u003C/p\u003E\n","\u003Cp\u003ETragic! The script failed with the following error:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_10.png\" alt=\"sma_code_editor_10\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIt looks like the way we are referencing an identifier is not the way that Snowpark wanted it. The code that failed is in the exact spot where the remaining EWI is:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_11.png\" alt=\"sma_code_editor_11\"\u003E\u003C/p\u003E\n","\u003Cp\u003EYou could reference the documentation on the link provided by the error, but in the interest of time, Snowpark needs this variable to expressly be a literal. We need to make the following replacement:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Old\nsplit_col = split(df_uppercase['NAME'], '.first:')\n\n# New\nsplit_col = split(df_uppercase['NAME'], lit('.first:'))\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EThis should take care of this error. Note that there are always going to be some functional differences between source and a target platforms. Conversion tools like the SMA like to make these differences as obvious as possible. But note that no conversion is 100% automated.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s run it again. This time&hellip; success!\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_12.png\" alt=\"sma_code_editor_12\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWe can write some queries in python to validate this, but why don&rsquo;t we just go into Snowflake (because that&rsquo;s what we&rsquo;re about to do anyways).\u003C/p\u003E\n","\u003Cp\u003ENavigate to your snowflake account that you have been using to run these scripts. This should be the same one you used to load the database from SQL Server (and if you haven&rsquo;t done that, the above scripts won&rsquo;t work anyways beecause the data has not yet been migrated).\u003C/p\u003E\n","\u003Cp\u003EYou can quickly check this by seeing if the stage was created with the file:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_stage_check_01.png\" alt=\"sma_stage_check_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003EEnable the directory table view to see if the old_versions folder is in there:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_stage_check_02.png\" alt=\"sma_stage_check_02\"\u003E\u003C/p\u003E\n","\u003Cp\u003EAnd it is:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_stage_check_03.png\" alt=\"sma_stage_check_03\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESince that was the last element of our script, it looks like we&rsquo;re good!\u003C/p\u003E\n","\u003Cp\u003EWe can also simply validate that the data was loaded by simply querying the table for the data we uploaded. You can open a new worksheet and simply write this query:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003Eselect * from ADVENTUREWORKS.DBO.DIMCUSTOMER\nwhere FIRSTNAME like '%Brandon%'\nAND LASTNAME like '%Carver%'\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_stage_check_04.png\" alt=\"sma_stage_check_04\"\u003E\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003ERunning the Pipeline Script in Snowsight\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s take a quick look back at the flow we are attempting to convert was doing in Spark:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003Eaccessing a local file\u003C/li\u003E\u003Cli\u003Eloading the result into SQL Server\u003C/li\u003E\u003Cli\u003Emoving the file to make way for the next one\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EThis flow is not possible to run entirely from within Snowsight. Snowsight does not have access to a local file system. The recommendation here would be to move the export from the POS to a data lake&hellip; or any number of other options that would be accessible via Snowsight.\u003C/p\u003E\n","\u003Cp\u003EWe can, however, take a closer look at how Snowpark handles the transformation logic by running the Python script in Snowflake. If you have already made the changes recommended above, you can run the body of the script in a Python Worksheet in Snowflake.\u003C/p\u003E\n","\u003Cp\u003ETo do this, first login to your Snowflake account and navigate to the worksheets section. In this worksheet, create a new Python worksheet:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_01.png\" alt=\"sma_python_worksheet_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESpecify the database, schema, role, and warehouse you&rsquo;d like to use:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_02.png\" alt=\"sma_python_worksheet_02\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENow we do not have to deal with our session call. You will see a template generated in the worksheet window:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_03.png\" alt=\"sma_python_worksheet_03\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s start by bringing over our import calls. After making the previous script ready to use, we should have the following set of imports:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# General Imports\nimport pandas as pd\nimport os\nimport shutil\nimport datetime\n\n# Snowpark Imports\nfrom snowflake.snowpark import Session\nfrom snowflake.snowpark.functions import col\nfrom snowflake.snowpark.functions import upper\nfrom snowflake.snowpark.functions import lower\nfrom snowflake.snowpark.functions import split\nfrom snowflake.snowpark.functions import trim\nfrom snowflake.snowpark.functions import when\nfrom snowflake.snowpark.functions import lit\nfrom snowflake.snowpark.functions import expr\nfrom snowflake.snowpark.functions import regexp_replace\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EWe only need the snowpark imports. We will not be moving files around a file system. We could keep the datetime reference if we want to move the file in the stage. (Let&rsquo;s do it.)\u003C/p\u003E\n","\u003Cp\u003EPaste the Snowpark imports (plus datetime) in the python worksheet below the other imports that are already present. Note that &lsquo;col&rsquo; is already imported, so you can remove one of those:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_04.png\" alt=\"sma_python_worksheet_04\"\u003E\u003C/p\u003E\n","\u003Cp\u003EUnder the &ldquo;def main&rdquo; call, let&rsquo;s paste in all of our transformation code. This will include everything from the assignment of the csv location to the writing of the dataframe to a table.\u003C/p\u003E\n","\u003Cp\u003EFrom here:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_05.png\" alt=\"sma_python_worksheet_05\"\u003E\u003C/p\u003E\n","\u003Cp\u003ETo here:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_06.png\" alt=\"sma_python_worksheet_06\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWe can also add back in the code that moves the files around in the stage to preserve the same functionality for the POC. This part:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_07.png\" alt=\"sma_python_worksheet_07\"\u003E\u003C/p\u003E\n","\u003Cp\u003EBefore you can run the code though, you will have to manually create the stage and move the file into the stage. We can add the create stage statement into the script, but we would still need to manually load the file into the stage.\u003C/p\u003E\n","\u003Cp\u003ESo if you open another worksheet (this time&hellip; a sql worksheet), you can run a basic SQL statement that will create the stage:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-sql\"\u003ECREATE STAGE LOCAL_LOAD_STAGE\n  ENCRYPTION = (TYPE = 'SNOWFLAKE_SSE');\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EMake sure to select the correct database, schema, role, and warehouse in your worksheet:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_08.png\" alt=\"sma_python_worksheet_08\"\u003E\u003C/p\u003E\n","\u003Cp\u003EYou can also \u003Ca href=\"https://docs.snowflake.com/en/user-guide/data-load-local-file-system-create-stage#create-a-named-stage-using-snowsight\"\u003Ecreate an internal stage directly in the Snowsight UI\u003C/a\u003E. Now that the stage exists, we can manually load the file of interest into the stage. Navigate to the Databases section of the Snowsight UI, and find the stage we just created in the appropriate database.schema:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_09.png\" alt=\"sma_python_worksheet_09\"\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s add our csv file by selecting the +Files option in the top right corner of the window. This will launch the Upload Your Files menu:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_10.png\" alt=\"sma_python_worksheet_10\"\u003E\u003C/p\u003E\n","\u003Cp\u003EDrag and drop or browse to our project directory and load the customer_update.csv file into the stage:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_11.png\" alt=\"sma_python_worksheet_11\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESelect Upload in the bottom right corner of the screen. You will be taken back to the stage screen. To view the files, you will need to select Enable Directory Table:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_12.png\" alt=\"sma_python_worksheet_12\"\u003E\u003C/p\u003E\n","\u003Cp\u003EAnd now&hellip; our file appears in the stage:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_13.png\" alt=\"sma_python_worksheet_13\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis is not really a pipeline anymore, of course. But at least we can run the login in Snowflake. Run the rest of the code that you moved into the worksheet. This user had success the first time, but that&rsquo;s no guarantee of success the second time:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_14.png\" alt=\"sma_python_worksheet_14\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that once you&rsquo;ve defined this function in Snowflake, you can call it in other ways. If AdventureWorks is 100% replacing their POS, then it may make sense to have the transformation logic in Snowflake, especially if orchestration and file movement will be handled somewhere else entirely. This allows Snowpark to focus on where it excels with the transformation logic.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EConclusions\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EAnd that's it for the script file. It's not the best example of a pipeline, bu it does hit hard on how to deal wit ht he output from the SMA:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003EResolve All Issues\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EResolve the session calls\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EResolve the Input/Outputs\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EClean up and Test\u003C/strong\u003E!\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ELet's move on to the reporting notebook.\u003C/p\u003E\n","\u003Ch2\u003ENotebook Conversion and Validation\u003C/h2\u003E\n","\u003Cp\u003ELet&rsquo;s step over to the Reporting Notebook in our codebase: \u003Cstrong\u003EBasic Reporting Notebook - SqlServer Spark.ipynb\u003C/strong\u003E. We're going to walk through a similar set of steps as we did with the pipeline script.\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003EResolve All Issues\u003C/strong\u003E: &ldquo;Issues&rdquo; here means the issues generated by the SMA. Take a look at the output code. Resolve parsing errors and conversion errors, and investigate warnings.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EResolve the session calls\u003C/strong\u003E: How the session call is written in the output code depends on where we are going to run the file. We will resolve this for running the code file(s) in the same location as they were originally going to be run, and then for running them in Snowflake.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EResolve the Input/Outputs\u003C/strong\u003E: Connections to different sources cannot be resolved entirely by the SMA. There are differences in the platforms, and the SMA will usually disregard this. This also is affected by where the file is going to be run.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003EClean up and Test\u003C/strong\u003E! Let&rsquo;s run the code. See if it works. We will be smoke testing in this lab, but there are tools to do more extensive testing and data validation including Snowpark Python Checkpoints.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ELet&rsquo;s get started.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EResolve All Issues\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s go ahead and look at the issues present in the notebook.\u003C/p\u003E\n","\u003Cp\u003E(Note that you can open the notebook in VS Code, but to view it appropriately, you may want to install the Jupyter extension for VS Code. Alternatively, you could open this in Jupyter, but Snowflake still recommends VS Code with the Snowflake extension installed).\u003C/p\u003E\n","\u003Cp\u003EYou can use the compare feature to view both of these side by side as we did with the pipeline file, though it will look more like a json if you do so:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_01.png\" alt=\"sma_notebook_01\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that there are only two unique EWI&rsquo;s in this notebook. You can return to the search bar to find them, but since this is so short, you could also just&hellip; scroll down. These are the unique issues:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Cstrong\u003ESPRKPY1002\u003C/strong\u003E \u003Cem\u003E=&gt; pyspark.sql.readwriter.DataFrameReader.jdbc is not supported.\u003C/em\u003E This is a similar issue to the one we saw in the pipeline file, but that was a write call. This is a read call to the SQL Server database. We will resolve this in a bit.\u003C/li\u003E\u003Cli\u003E\u003Cstrong\u003ESPRKPY1068\u003C/strong\u003E \u003Cem\u003E=&gt; &quot;pyspark.sql.dataframe.DataFrame.toPandas is not supported if there are columns of type ArrayType, but it has a workaround. See documentation for more info.\u003C/em\u003E This is another warning. If we pass an array to this function in Snowpark, it may not work. Let&rsquo;s keep an eye on this when we test it.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EAnd that&rsquo;s it for the notebook&hellip; and our issues. We resolved a parsing error in the script file, but we did not have much to do in this notebook. Most of this was converted successfully, though there are a few things we will keep our eyes on. Let&rsquo;s move on to the next step: resolving any session calls.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EResolve the Session Calls\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ETo update the session calls in the reporting notebook, we need to locate the cell with the session call in it. That looks like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_02.png\" alt=\"sma_notebook_02\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENow let&rsquo;s do what we already did for our pipeline file:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EChange all references to the &ldquo;spark&rdquo; session variable to &ldquo;session&rdquo; (note that this is throughout the notebook)\u003C/li\u003E\u003Cli\u003ERemove the config function with the spark driver.\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EThe before and after on this will look like this:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Old Session\nspark = Session.builder.config('spark.driver.extraClassPath', driver_path).app_name(&quot;AdventureWorksSummary&quot;, True).getOrCreate()\nspark.update_query_tag({&quot;origin&quot;:&quot;sf_sit&quot;,&quot;name&quot;:&quot;sma&quot;,&quot;version&quot;:{&quot;major&quot;:7,&quot;minor&quot;:4,&quot;patch&quot;:10},&quot;attributes&quot;:{&quot;language&quot;:&quot;Python&quot;}})\n\n# New Session\n# Session\nsession = Session.builder.app_name(&quot;AdventureWorksSummary&quot;, True).getOrCreate()\nsession.update_query_tag({&quot;origin&quot;:&quot;sf_sit&quot;,&quot;name&quot;:&quot;sma&quot;,&quot;version&quot;:{&quot;major&quot;:7,&quot;minor&quot;:4,&quot;patch&quot;:10},&quot;attributes&quot;:{&quot;language&quot;:&quot;Python&quot;}})\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003ENote that there is other code in this cell. This code:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003Eurl = sql_server_url\nproperties = {'user' : sql_server_user, 'password' : sql_server_password}\n# Spark dataframe.\n#EWI: SPRKPY1002 =&gt; pyspark.sql.readwriter.DataFrameReader.jdbc is not supported\ndf = session.read.jdbc(url = url, table = 'dbo.DimCustomer', properties = properties)\nprint('Session successfully setup.')\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EWe&rsquo;re almost ready to take on the read statement, but we&rsquo;re not there yet. Let&rsquo;s just move all of this to another cell. Create a new cell below this one, and move this code to that cell. It will look like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_03.png\" alt=\"sma_notebook_03\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIs this all we need for the session call? No. Recall (and possibly review) the previous page under \u003Cstrong\u003ENotes on Session Calls\u003C/strong\u003E. You will either need to make sure that your connection.toml file has your connection information or you will need to explicitly specify the connection parameters you intend to use in the session.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EResolving the Inputs/Outputs\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ESo let&rsquo;s resolve our inputs and outputs now. Note that this is going to diverge based on whether you&rsquo;re running the files locally or Snowflake, but for the notebook, everything can be run locally or in Snowflake. The code will be a bit simpler as we won&rsquo;t even need to call a session. We&rsquo;ll just&hellip; get the active session. As with the pipeline file, we&rsquo;ll do this in two parts: to be run/orchestrated locally, and to be run in Snowflake.\u003C/p\u003E\n","\u003Cp\u003EWorking through the inputs and outputs in the reporting notebook will be considerably simpler than it was for the pipeline. There is no reading from a local file or moving data between files. There is simply a read from a table in SQL Server that is now a read from a table in Snowflake. Since we will not be accessing SQL Server, we can ditch any reference to the SQL Server properties. And the read statement can be replaced by a table statement in Snowflake. The before and after for this cell should look like this:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Before\nurl = sql_server_url\nproperties = {'user' : sql_server_user, 'password' : sql_server_password}\n# Spark dataframe.\n#EWI: SPRKPY1002 =&gt; pyspark.sql.readwriter.DataFrameReader.jdbc is not supported\ndf = session.read.jdbc(url = url, table = 'dbo.DimCustomer', properties = properties)\nprint('Session successfully setup.')\n\u003C/code\u003E\u003C/pre\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# After\n# New table call\n# Snowpark Dataframe table.\ndf = session.table('ADVENTUREWORKS.DBO.DIMCUSTOMER')\nprint('Table loaded successfully.')\ndf.show()\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EThis leaves us with an updated session and the correct table read into a dataframe:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_04.png\" alt=\"sma_notebook_04\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThat&rsquo;s actually&hellip; it. Let's move on to the Clean up and test part of the notebook file.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EClean Up and Test\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s do some clean up (like we did previously for the pipeline file). We never looked at our import calls and we have config files that are not necessary at all. Let&rsquo;s start by removing the references to the config files. This will be each of the cells between the import statements and the session call.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_05.png\" alt=\"sma_notebook_05\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENow let&rsquo;s look at our imports. The reference to the os can be deleted. (Seems like that wasn&rsquo;t used in the original file either&hellip;) There is a pandas reference. There does not appear to be any usages of pandas in this notebook anymore now that the config files are referenced. There is a toPandas reference as part of the Snowpark dataframe API in the reporting section, but that&rsquo;s not part of the pandas library.\u003C/p\u003E\n","\u003Cp\u003EYou can optionally replace all of the import calls to pandas with the modin pandas library. This library will optimize pandas dataframes to take advantage of Snowflake&rsquo;s powerhouse computing. This change would look like this:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Old\nimport pandas as pd\n\n# New\nimport modin.pandas as pd\nimport snowflake.snowpark.modin.plugin\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EHaving said that, we can delete that one as well. Note that the SMA has replaced any spark specific import statements with those related to Snowpark. The final import cell would look like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_06.png\" alt=\"sma_notebook_06\"\u003E\u003C/p\u003E\n","\u003Cp\u003EAnd that&rsquo;s it for our cleanup. We still have a couple of EWIs in the reporting and visualization cells, but it looks like we should make it. Let&rsquo;s run this one and see if we get an output.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_07.png\" alt=\"sma_notebook_07\"\u003E\u003C/p\u003E\n","\u003Cp\u003EAnd we did. The reports seem to match what was output by the Spark Notebook. Even though the reporting cells seemed complex, Snowpark is able to work with them. The SMA let us know there could be an issue, but there doesn&rsquo;t appear to be any problems. More testing would help, but our first round of smoke testing has passed.\u003C/p\u003E\n","\u003Cp\u003ENow let&rsquo;s look at this notebook in Snowsight. Unlike the pipeline file, we can do this entirely in Snowsight.\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003ERunning the Notebook in Snowsight\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s take the version of the notebook that we have right now (having worked through the issues, the session calls, and the inputs and outputs) and load it into Snowflake. To do this, go to the notebooks section in SnowSight:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_08.png\" alt=\"sma_notebook_08\"\u003E\u003C/p\u003E\n","\u003Cp\u003EAnd select the down arrow next to the +Notebook button in the top right, and select &ldquo;Import .ipynb file&rdquo; (shown above).\u003C/p\u003E\n","\u003Cp\u003EOnce this has been imported, choose the notebook file that we have been working with in the output directory created by the SMA in your project folder.\u003C/p\u003E\n","\u003Cp\u003EThere will be a create notebook dialog window that opens. For this upload, we will choose the following options:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003ENotebook location:\u003C/li\u003E\u003Cli\u003EDatabase: \u003Cstrong\u003EADVENTUREWORKS\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003ESchema: \u003Cstrong\u003EDBO\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003EPython environment: \u003Cstrong\u003ERun on warehouse\u003C/strong\u003E (This is not a large notebook with a bunch of ml. This is a basic reporting notebook. We can run this on a warehouse.)\u003C/li\u003E\u003Cli\u003EQuery warehouse: \u003Cstrong\u003EDEFAULT_WH\u003C/strong\u003E\u003C/li\u003E\u003Cli\u003ENotebook warehouse: \u003Cstrong\u003EDEFAULT_WH\u003C/strong\u003E (you can leave it as the system chosen warehouse (will be a streamlit warehouse)... for this notebook, it will not matter)\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003EYou can see these selections below:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_09.png\" alt=\"sma_notebook_09\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis should load your notebook into Snowflake and it will look something like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_10.png\" alt=\"sma_notebook_10\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThere are a couple of quick checks/changes we need to make from the version we just tested locally in order to ensure that the notebook runs in Snowsight:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003EChange the session calls to retrieve the active session\u003C/li\u003E\u003Cli\u003EEnsure any dependent libraries we need to install are available\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ELet&rsquo;s start with the first one. It may seem odd to alter the session call again after we spent so much time on it in the first place, but we&rsquo;re running inside of Snowflake now. You can remove anything associated with reading the session call and replacing it with the &ldquo;get_active_session&rdquo; call that is standard at the top of most Snowflake notebooks:\u003C/p\u003E\n\u003Cpre\u003E\u003Ccode class=\"language-python\"\u003E# Old for Jupyter\nsession = Session.builder.app_name(&quot;AdventureWorksSummary&quot;, True).getOrCreate()\n\n# New for Snowsight\nfrom snowflake.snowpark.context import get_active_session\nsession = get_active_session()\n\u003C/code\u003E\u003C/pre\u003E\n","\u003Cp\u003EWe don't need to specify connection parameters or update a .toml file because we are already connected. we are in Snowflake.\u003C/p\u003E\n","\u003Cp\u003ELet&rsquo;s replace the old code in the cell with the new code. That will look something like this:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_11.png\" alt=\"sma_notebook_11\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENow let&rsquo;s address the available packages for this run, but instead of us figuring out what we need to add. Let&rsquo;s let Snowflake. One of the better parts of using a notebook is that we can run individual cells and see what the results are. Let&rsquo;s run our import library cell.\u003C/p\u003E\n","\u003Cp\u003EIf you haven&rsquo;t already, go ahead and start the session by clicking in the top right corner of the screen where it says &ldquo;Start&rdquo;:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_12.png\" alt=\"sma_notebook_12\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIf you run the topmost cell in the notebook, and you will likely discover that matplotlib is not loaded into the session:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_13.png\" alt=\"sma_notebook_13\"\u003E\u003C/p\u003E\n","\u003Cp\u003EThis is a pretty important one for this notebook. You can add that library to your notebook/session by using the &ldquo;Packages&rdquo; option in the top right of the notebook:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_14.png\" alt=\"sma_notebook_14\"\u003E\u003C/p\u003E\n","\u003Cp\u003ESearch for matplotlib, and select it. This will make this package available in the session.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_15.png\" alt=\"sma_notebook_15\"\u003E\u003C/p\u003E\n","\u003Cp\u003EOnce you load this library, you will have to restart the session. Once you have restarted the session, run that first cell again. You will likely be told that it was a success this time.\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_16.png\" alt=\"sma_notebook_16\"\u003E\u003C/p\u003E\n","\u003Cp\u003EWith the packages loaded, the session fixed, and the rest of the issues in the code already resolved, what can we do to check the rest of the notebook? Run it! You can run all the cells in the notebook by selecting &ldquo;Run all&rdquo; in the top right corner of the screen, and see if we get any errors.\u003C/p\u003E\n","\u003Cp\u003EIt looks like there was a successful run:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_17.png\" alt=\"sma_notebook_17\"\u003E\u003C/p\u003E\n","\u003Cp\u003EIf you compare the two notebooks execution, it looks like the only difference is that the Snowflake version put all of the output datasets first followed by the images, whereas they are intermixed in the Spark Jupyter Notebook:\u003C/p\u003E\n","\u003Cp\u003E\u003Cimg src=\"https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_18.png\" alt=\"sma_notebook_18\"\u003E\u003C/p\u003E\n","\u003Cp\u003ENote that this difference is not an API difference, but rather a difference in how notebooks in Snowflake orchestrate this. This is likely a difference AdventureWorks is willing to accept!\u003C/p\u003E\n","\u003Cp\u003E\u003Cstrong\u003EConclusions\u003C/strong\u003E\u003C/p\u003E\n","\u003Cp\u003EBy utilizing the SMA, we were able to accelerate the migration of both a data pipeline and a reporting notebook. The more of each that you have, the more value a tool like the SMA can provide.\u003C/p\u003E\n","\u003Cp\u003EAnd let&rsquo;s go back to the assessment -&gt; conversion -&gt; validation flow that we have consistently come back to. In this migration, we:\u003C/p\u003E\n\u003Cul\u003E\u003Cli\u003ESetup out project in the SMA\u003C/li\u003E\u003Cli\u003ERan SMA&rsquo;s assessment and conversion engine on the code files\u003C/li\u003E\u003Cli\u003EReviewed the output reporting from the SMA to better understand what we have\u003C/li\u003E\u003Cli\u003EReview what could not be converted by the SMA in VS Code\u003C/li\u003E\u003Cli\u003EResolve issues and errors\u003C/li\u003E\u003Cli\u003EResolve session references\u003C/li\u003E\u003Cli\u003EResolve input/output references\u003C/li\u003E\u003Cli\u003ERun the code locally\u003C/li\u003E\u003Cli\u003EAnd run the code in Snowflake\u003C/li\u003E\u003Cli\u003ERan the newly migrated scripts and validated their success\u003C/li\u003E\u003C/ul\u003E\n","\u003Cp\u003ESnowflake has spent a great deal of time improving its ingestion and data engineering capabilities, just as it has spent time improving migration tools like SnowConvert, the SnowConvert Migration Assistant, and the Snowpark Migration Accelerator. Each of these will continue to improve. Please feel free to reach out if you have any suggestions for migration tooling. These teams are always looking for additional feedback to improve the tools.\u003C/p\u003E\n","\u003Ch2\u003EConclusion And Resources\u003C/h2\u003E\n","\u003Cp\u003ECongratulations! You've successfully completed an end-to-end migration from SQL Server to Snowflake, including both the database objects and the ETL pipelines that feed your data mart.\u003C/p\u003E\n","\u003Ch3\u003EWhat You Learned\u003C/h3\u003E\n\u003Cul\u003E\u003Cli\u003EHow to assess a SQL Server environment for migration to Snowflake\u003C/li\u003E\u003Cli\u003EUsing SnowConvert to migrate database schemas and resolve conversion issues\u003C/li\u003E\u003Cli\u003ETransferring data from SQL Server to Snowflake\u003C/li\u003E\u003Cli\u003EConverting Spark ETL pipelines to Snowpark using the SMA\u003C/li\u003E\u003C/ul\u003E\n","\u003Ch3\u003ERelated Resources\u003C/h3\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Ca href=\"https://docs.snowconvert.com/\"\u003ESnowConvert Documentation\u003C/a\u003E\u003C/li\u003E\u003Cli\u003E\u003Ca href=\"https://docs.snowflake.com/en/user-guide/migration-accelerators\"\u003ESnowpark Migration Accelerator Guide\u003C/a\u003E\u003C/li\u003E\u003Cli\u003E\u003Ca href=\"https://resources.snowflake.com/migration/microsoft-sql-server-to-snowflake-migration-guide\"\u003ESnowflake SQL Server Migration Guide\u003C/a\u003E\n\u003Cul\u003E\u003Cli\u003E\u003Ca href=\"https://github.com/Snowflake-Labs/sfguides/tree/master/site/sfguides/src/e2e-sqlserver-migration/\"\u003EGitHub Repository for this Lab\u003C/a\u003E\u003C/li\u003E\u003C/ul\u003E\n\u003C/li\u003E\u003C/ul\u003E"],"description":"Migrate SQL Server databases and Spark pipelines to Snowflake using SnowConvert and Snowpark Migration Accelerator tools.","title":"End-to-End Migration to Snowflake in Action: Data and Pipelines",":type":"snowflake-site/components/contentfragment",":items":{},":itemsOrder":[],"elementsOrder":["quickstartArticleBody","quickstartArticleLogoImage"],"isDeveloperGuidesPage":false,"elements":{"quickstartArticleBody":{"dataType":"string","title":"Quickstart Article Body","value":"## Overview\n\nWant to migrate to Snowflake in less than 90 minutes? Come to this hands-on lab. We will work through Snowflake's migration strategy and automation tools—for example, SnowConvert and Snowpark Migration Accelerator—to assess, move, and validate an entire pipeline with data into Snowflake. Our data will be in another data warehouse, a data lake, and even on prem. Can we move it into Snowflake? Our pipelines are written in Jupyter notebooks utilizing Spark. Can those fully exist in Snowflake? Can all of it happen in Snowflake? Let's find out.\n\n### Prerequisites\n\n- Basic knowledge of SnowConvert and SMA\n\n### What You'll Learn\n\n- Migrate SQL Server Adventure Works' DDLs\n- Fix issues derived from SnowConvert conversion output code\n- Migrate SQL Server Adventure Works' Historical Data\n- Migrate Spark ETL Pipeline of Adventure Works\n\n### What You'll Need\n\n- [SnowConvert](/en/migrate-to-the-cloud/snowconvert-ai/) Installed\n- [Snowpark Migration Accelerator (SMA)](/en/migrate-to-the-cloud/migration-accelerator/) Installed\n- [VSCode](https://code.visualstudio.com/download) and [Snowflake Extension](https://marketplace.visualstudio.com/items?itemName=snowflake.snowflake-vsc) Installed\n- *[Preferred, but not required]* Python and PySpark Installed\n\n### Scenario\n\nAdventureWorks is a retail company established 20 years ago. They have been using SQL Server since their inception. As they grew and data processing evolved, they decided to start running some pipelines using Spark to take advantage of distributed computing. They also created SQL Server data marts in Azure in an early attempt to move into the cloud.\n\nOver time, they have decided to consider using Snowflake. They want to take one of their SQL Server datamarts and all of the pipelines that are feeding it… and move it into Snowflake. This sounds like a daunting task. How should they approach it? \n\nAdventureWorks has decided to do a Proof of Concept (POC) with a single database. The goal is to move everything from this database into Snowflake, as well as at least one of the pipelines bringing data into this database and at least one of the reporting notebooks they have connected to this database.\n\n## Assessment\n\nAssessing what you have and developing a plan is the most essential part of a migration. When done right, this will set you up for success over the remainder of the migration. Whether it’s one table, pipeline, or data warehouse, you need to have a plan that takes into account not just your data, but everything that interacts with your data. A successful migration is built at the beginning with a complete picture of what you want to migrate. With that assessment, comes a plan of attack on how to migrate.\n\nIn order to migrate across platforms, AdventureWorks needs to utilize a migration framework and strategy that will account for everything required to get the database up and running. Moving some of the data or some of the notebooks… this will not be enough. They need to move the entire system of pipelines that is connected to that data and those notebooks. \n\nHere’s a migration strategy espoused by Snowflake:\n![Assessment](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/01_assessment.png)\n\nThere are many phases in this migration framework, but most of them can be bucketed into three larger categories:\n\n- **Assessment** - Understand what you have\n- **Conversion** - Move it to the new platform\n- **Validation** - Test that everything works\n\nWe will walk through all the nine phases shown above in this Hand-On-Lab, but note that the primary activities we are doing are the three listed above. If you get lost or you’re not su    re where to go next, remember to assess -\u003E convert -\u003E test. Let’s start with the assessment phase. \n\n\u003E \n\u003E Note that executing a migration requires a close eye on access and security. You will be running two platforms in parallel, and getting a whole new environment setup. For the purpose of time in this lab, we have already set up the target (Snowflake) environment for you, and we will assume you are the only user of this environment. But full RBAC management and support is something every database administrator will have to deal with.\n\n### Planning and Design\n\nAs was mentioned previously, this migration will take a single database that the AdventureWorks company has and migrate it to Snowflake. This will include any data pipelines that are bringing data in and out of this mart. \n\nTo better understand everything that is bringing this data in and out of the pipeline, let’s put together an architecture diagram. \n\n#### Source Architecture\n\nGetting an architecture diagram is often one of the first steps in understanding what you have. If you’ve never built an architecture diagram, it can seem like a daunting task. A few suggestions would be:\n\n- **Start simple.** Don’t try to create a complete picture in an orderly way. Identify key components, then start listing the components that interact with this data.\n- **Socialize your drafts.** Data owners across any organization will know best how users are interacting with the data. The knowledge is not always consolidated with the systems or IT team.\n- **Identify opportunities for improvement.** When you can visually see the connection points in your system, sometimes the opportunities for improvements in performance, price, or complexity are immediately identifiable.\n- **Revisit it.** Most organizations' data landscapes change wildly in a very short period of time. These static diagrams are quickly out of date, and when the discussion of modernization occurs, the creators are not around anymore or they are simply not aware of changes that have occurred.\n\nSnowflake won’t suggest a specific tool that will help you manage your architecture, but there are tools available. \nHere’s a simple diagram.\n![SourceArchitecture](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/02_sourcearchitecture.png)\n\nSpark is being used across this organization in a variety of ways, but for this POC, it is simply being used as ETL. It is being orchestrated by Airflow, but that is assumed to continue for the moment. \n\n#### Target Architecture\n\nThis is the source system that we have. Let’s consider the future state\n\nAs we migrate, this is going to be our goal. Our inventory of artifacts in the source configuration will map over to objects in the target.\n![TargetArchitecture](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/03_targetarchitecture.png)\n\nNote that this is a **lift-and-shift migration**. This means we are going to recreate exactly what is in the source in the target environment. Everything that was in our SQL Server data mart plus everything that was connected to it will be recreated as is. Once you are on the new platform, then you can optimize. \n\nSome may advocate for a different approach to the migration that is more of a lift-adjust-shift. While this approach may yield some optimization benefits sooner in the migration process, most projects will take longer and the testing will be considerably more challenging. Most migration projects benefit from the lift-and-shift approach, but it especially makes makes the most sense for our two goals:\n\n- Prove the migration is possible\n- And do it in 90 minutes.\n\nAs a result, we will be moving the following and testing them in Snowflake:\n\n- SQL Server Database\n- Python Notebooks with references to the Spark API\n\n#### Inventories\n\nIn order to get started with this, we will need to build some object inventories of what we have in our existing system. We may know that we have a SQL Server data mart that we want to move with some Jupyter notebooks, but what objects are we migrating? What local files are we reading from? What dependencies exist across those objects and files?\n\nFor a smaller migration like this, this may be easy to track down manually. But for large migrations with multiple data sources, multiple different types of pipelines, and multiple different data owners, this can be nearly impossible. The challenge of understanding what you have often leads to the project never even getting off the ground. \n\nLuckily, Snowflake has some tools to help you understand what you have. These tools are not specific to building an object inventory, but rather are comprehensive tools for assessment and conversion of source code to Snowflake. These two tools will be with us most of the way, so let’s go ahead and introduce them now:\n\n- **SnowConvert**: SnowConvert is a tool that scans source code, reports on what it finds in the code, and inventories that code.\n- **Snowpark Migration Accelerator (SMA)**: The SMA is built on the same framework as SnowConvert, but it takes in code written in Python or Scala with references to the Spark API in them.\n\nEach of these tools is built on the basic principle that your codebase can be scanned, and a semantic model of everything that is happening in your code can be built. From this model, various reports (like an object inventory) and the output code can be generated.\n![Structure](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/04_tool.png)\nSo let’s do this for real. Let’s start getting our inventory of objects together. \n\nFor expediency in this lab, we are going to go straight from the assessment into the conversion phase. This means that we will look at the assessment data while we work through the conversion. Note that in reality, we would want to understand everything we have, before we start to move it.\n\nNow that we have an inventory of the artifacts we’re looking to move to Snowflake, let’s go ahead and move them. \nWe’ll walk through moving the following steps:\n\n- Database Code Conversion\n- Data Migration\n- Data Ingestion (pipelines)\n- Reporting and Analytics\n\nWhile at the highest level, this phase focuses on moving or converting from the source to Snowflake, there will often be testing and additional pieces of code to assess as we move through it. Meaning that while we do this conversion, it will be iterative with the testing intermixed into the process.\n\n## Database Code Conversion\n\nWe will start by assessing what we have. SnowConvert.\n\n### Project Creation\n\nWe have an adventure works database in SQL Server, and we have some spark scripts that load data into the sql server. The customer wants to move all of this into Snowflake.\n\nThe SQL Server is in an Azure environment that we have in Snowflake, but can make public for a temporary amount of time?\n\nHow do we connect to it? Through SnowConvert. Let’s do that now.\n\nOpen SnowConvert. Currently, [it can be downloaded from here](/en/migrate-to-the-cloud/snowconvert-ai/). An installation package file will be created. You can then open this file and install the application on your machine.\n\nAgree to the Terms of Use by selecting “Agree”.\n\u003E \n\u003E Note that if you are on a Mac, you may be asked to move the application to your windows folder.\n\n**Open SnowConvert**: When you open the application it may prompt you to update it:\n\n![InitialScreen](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/initialScreen.png)\n\nWe won’t do that at this time, but when working with the application, it is generally better to keep it as up to date as possible. This is a local application. Keeping it up to date not only ensures the functionality of the application, but also ensures that you have the most up to date version of the conversion core.\n\n**Select New Project**:\n![NewProject](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/NewProject.png)\n\nTo begin using any version of SnowConvert, you will need to create a project. Think of a project as a local config file that will be saved to your machine. This will preserve any settings and will allow you to continue where you left off if you need to step away.\n\nLet’s call our project: **SQL Server ADW Test.**\n\nWe then need to select a source. In this case, it is SQL Server.\n\n\nTo run any element of a project in SnowConvert, you will need to provide an access code. Unless you have used SnowConvert before, you will need an access code. Happily, you can request access in the tool by selecting “Get an access code” next to the access code drop down. \n\n![ProjectCreation](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/ProjectCreation.png)\n\nNote that if you have already activated an access code, you will see options in the dropdown menu in the license screen as shown here:\n\n![ProjectCreation2](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/ProjectCreation2.png)\n\nYou can choose an access code from this list if you have already activated one.  \n\nAssuming we do not yet have an access code, let’s request an access code. Choose “Get an access code” from the menu to the right side of the dropdown menu. When you do this, the access code form will pop up:\n![license](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/fill_license.png)\n\nThis information is needed to confirm who you are to Snowflake. It will not be used by the SnowConvert team to start sending you the latest and greatest update on what’s new in migrations. (Though if that’s something you’d be interested in… let us know.) \n\nOnce you complete the form, you will receive an email with an access code. It will look something like this:\n\n![license_delivery](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/licensedelivery.png)\n\nPaste this access code into the application where it says “Enter new access code or select one” in SnowConvert. When your access code has been accepted, you will get a small message under the dropdown menu that says **“Access code activated successfully”**.\n\n![insert_license](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/insert_license.png)\n\nYou will need to have an active internet connection in order to activate your access code. If you are unable to activate your access code, check out the [troubleshooting section](https://docs.snowconvert.com/sc/general/frequently-asked-questions-faq#why-am-i-not-receiving-an-access-code) of the SnowConvert documentation.\n\nNow that we’re active, let’s Extract! It’s time!\n\n### Extract\n\n![extractLandscape](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/extractLandscape.png)\n\nFrom the Project Creation menu, select the blue **CONTINUE** button in the bottom right corner of the application. Then click on **From SQL Server** in order to extract code from source database.\n\n![FromSQLServer](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/fromSQLServer.png)\n\nWhen you scroll down, you will have to enter the connection information:\n\n![conn](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/connect_sqlserver.png)\n\nFor this Hands on Lab, we will provide you with some credentials. However, if you are executing this outside of the friendly confines of this hands on lab, you will need to enter your own credentials. \n\nFill the connection information to connect to Adventure Works, you can build your own Adventure Works environment [here](https://learn.microsoft.com/en-us/sql/samples/adventureworks-install-configure?view=sql-server-ver17&tabs=ssms) as well. \n\n- Authentication Method: Standard\n- Server URL: *Adventure Works Connection URL*\n- Database: *AdventureWorks*\n- Port: *1433*\n- Username: *Fill the user name here*\n- Password: *Fill the password here*\n\nCheck both boxes for “Trust Server Certificate” and “Encrypt Connection”. \n\nNext, we will have to specify a local path for our project folder. Anything that we do SnowConvert will be preserved in this project path as well as anything that is created locally. Choose a path that is fully accessible to you. This is the path I chose: `/Users/userName/tmp/SQLServer_ADW_Test`\n\n![selectProjectPath](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/selectProjectPath.png)\n\nFinally, you can click `CONNECT TO DATABASE`, then you will get a pop up that says “Connect Established” when you have connected. SnowConvert will then take you to the catalog screen:\n\n![extract_page](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/extraction_page.png)\n\nThe catalog screen allows you to browse objects that were found in the database. For SQL Server, this could be tables, views, procedures, or functions. Nothing has been converted yet. This is merely an inventory of what SnowConvert found in the source. \n\nUsing the catalog, we can select a set of objects for which we’d like to extract the DDL. Using the filter options, you can search for a specific object or set of objects. Using the checkboxes, you can select a specific subset of objects or select the highest checkbox to select everything:\n\n![select_extract](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/select_extract.png)\n\nIn this example, we will select the top checkbox and select everything. This will include tables, views, and functions. Then select “EXTRACT OBJECTS” to extract the DDL. \n\n![click_extract](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/click_extract.png)\n\nThis will create a folder on the local machine preservig the structure of the objects in the database with a file for the DDL for each object.\nWhen the extraction is complete, you will see a results screen similar to this: \n\n![extract_result](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/extract_result.png)\n\nThis will give you a brief overview of what was extracted. If there were errors or something was not able to be extracted, it will be reported to you here.\n\nYou can select “VIEW EXTRACTED OBJECTS” to see where SnowConvert put the extracted DDL. But since the number of objects we have extracted matches what we expected and there are no errors, we can close this dialog menu and return to the catalog.\n\nNote that now we can see a green checkbox where the DDL was successfully extracted for the object:\n\n![post_extract](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/post_extract.png)\n\nIf there was an error extracting the DDL, you would see a red X and would need to resolve why that was not extracted.\n\n### Conversion\n\nAt this point, we’ve extracted the objects in the database and we’re ready to assess the compatibility with Snowflake and begin the conversion process. There are some optional steps we can do before we get to the conversion itself. Let’s take a look at this by selecting “GO TO MAPPINGS -\u003E” in the bottom right corner of the application.\n\nThis brings us to the mapping screen.\n\n![Mappings](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/mappings.png)\n\nOn this screen, you can choose a new name for a specific object in Snowflake (i.e. map a single object from SQL Server to Snowflake). You can also choose BULK MAPPING to apply a prefix or suffix to all of the objects or a subset of them (such as tables or view).\n\n![BulkMapping](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/bulkmapping.png)\n\nNote that this is completely optional when doing the migration. In this scenario, we will not do any custom mappings. \n\nBecause this is a poc, we’ll apply the suffix “_poc” to all the objects that we have in our database. Because this applies the change to the object itself, any relationship to the object will be updated to reflect the new name. (For example, if the name of TABLE_A was changed to TABLE_A_poc, and TABLE_A was referenced by FUNCTION_B, the reference to TABLE_A in FUNCTION_B will be changed to TABLE_A_poc.)\n\n![BulkMappingInfoAdded](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/mapping2.png)\n\nIn the suffix panel, we can add the suffix we want to fix to each object name. Select “APPLY MAPPINGS” and you will see the name changes in the Target Name column of the catalog:\n\n![MappingApplied](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/mapping3.png)\n\nNote that you can still edit individual object names if so desired.\n\nNow that we’ve mapped all of the source and target objects, we can run the conversion and view the assessment information. Select “START CONVERSION”.\nSince we are leaving our object names unaffected, let’s start the conversion process. Select “START CONVERSION” in the bottom right had corner of the application.\n\nIt’s possible you will view an error message similar to this one:\n\n![ScopeValidation](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/scopeValidation.png)\n\nThis simply means that SnowConvert has scanned the code that it extracted from the database before it runs its conversion script, and has found some things that COULD cause errors. It will tell you some things you might want to change in the source before converting. These can be found in the “Scope validation report” that you can read. In this scenario, we’ll just click “CONTINUE”.\n\nSnowConvert will then execute its conversion engine. This is done by scanning the codebase and creating a semantic model of the source codebase. This model is then used by SnowConvert to create the output Snowflake code as well as the generated reports. \n\nWhen the conversion is finished, each step will be highlighted:\n\n![ConversionProgress](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/conversion_progress.png)\n\nSelect “VIEW RESULTS” to… well… view the results.\n\n![ConversionResults](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/conversion_results.png)\n\nThe results page will give you a code completeness score initially, but there is more information below if you scroll down. There is more information on each element of the output report in the SnowConvert documentation, but we’ll just highlight a few elements of the report for this lab, and we’ll do the followup for each of them which will explore more in depth. \n\n**Code Completeness**: This is a reference to any missing elements or objects that are not present in the codebase. If you have 100% code completeness, then you do not have any missing objects or references to missing elements in the codebase. \n\nIn this scenario, we have an overall 97.5% completeness. This is broken down in the Assessed Conversion Scope Summary (scroll down in the application to find this section):\n\n![AssessmentSummary](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/assessment_summary.png)\n\n\u003E \n\u003E \n\u003E We may not need to resolve any missing dependencies (it’s unlikely in this scenario because we did a direct extract from the database), but we still want to see why SnowConvert is telling us that something is missing. \n\nTo do this, let’s visit the additional reports that are generated by SnowConvert. Select “VIEW REPORTS” from the bottom of the application:\n\n![ClickViewReports](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/clickViewReports.png)\n\nThis will take you to your file explorer, specifically to the local directory where the reports generated by SnowConvert have been created. This will be within the directory that you specified in the project creation menu at the start of the project. \n\n![OutputReports](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/outputReports.png)\n\n\u003E \n\u003E\n\u003E Note that the extracted and converted code are also in this directory structure, but we’ll get to those later.\n\nIn the SnowConvert folder, you will find many reports. We’ll view some of them later, but for now, let’s focus on the MissingObjectReferences.csv report. Open this one.\nIn this file, we’ll find the name of a referenced element that is not present in the DDL of the output code. \n\n![MissingReferences](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/missingreferences.png)\n\nLet’s validate that this is a missing reference. There are several ways we can do this, but the first would be to simply open the object inventory and validate that this function actually does exist. The inventory of all code objects in this scan is in the **TopLevelCodeUnits.\u003Cdatetime\u003E.csv** file: \n\n![ClickCodeUnits](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/clickCodeUnit.png)\n\nOpening this file, we can search for this reference in the CodeUnitId column:\n\n![CodeUnitReport](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/codeUnitReport.png)\n\nHuzzah! I have found it here. This means that the object does exist. Why did SnowConvert identify it as a missing object? That’s a great question. It could be that it’s called in a way that SnowConvert could not resolve. It could be that there was an error with SnowConvert, and it did not resolve the function name with the function DDL. We’ll have to dive deeper into why when we work through the issues. For now, it’s enough to know that the object does exist and we do not need to figure out how essential it is to track down. This may still come out when working through the issues, but let’s find out then. \n\nIf the result is 100% code completeness:\n\nIn this scenario, we have 100% code completeness. This makes sense given that we are exporting this directly from the source. If you do have missing objects here, the recommendation would be to open the reports folder and validate that the missing objects are either known to be missing or find the DDL for this object.\n\n**Conversion Overview**: Now that we have seen that we have the code that we need for this, let’s see how much of our code was converted. Let’s review the Code Units Summary section here:\n\n![CodeUnitOverview](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/codeUnitOverview.png)\n\nLooks like we have tables, views, and functions in this codebase, but not a lot of code in general (this looks like less than 1000 lines of code in total). There also are only three “EWI’s” (in the last column), meaning that the majority of this extracted DDL can be moved over to Snowflake just by using SnowConvert. We’ll look through the EWI’s in a moment. \n\nUnderstanding what we have is essential to successfully completing a migration. If we were pre-migration, we would likely stop here and review the object inventory. We’d also want to run the Snowpark Migration Accelerator (SMA) to validate that any pipelines we have include the objects that we are migrating here. In this scenario, we are going to go ahead and move forward to work through any issues that we have and will run the SMA later.\n\nSince we have a good understanding of what needs to be done and it’s relatively small, let’s go ahead and attack this. Let’s resolve the issues that we have present. Before we do that, let’s take a look at the status in our object inventory. Select “GO TO DEPLOYMENT” in the application.\n\n## Deployment and solve issues\n\nThis will take you back to the inventory screen. It should look something like this:\n\n![DeploymentPage](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentPage.png)\n\nThis is the same inventory that we have already seen, but note that now we can see a status in the conversion column. There are a few different elements that we can see here that are based on the conversion status. This lets us know which objects were fully converted (a green checkmark), which objects have a warning that you should consider (a green checkmark with a blue “i” icon), and which objects have a conversion error that must be addressed (a red “X”). These statuses are determined by the error messaging that SnowConvert has placed into the converted code. The objects with a red “X” have an error message that will produce an error if you attempt to run that SQL in Snowflake. If you resolve the errors, then you will be able to deploy the output code. SnowConvert is connected to your project directory, so as long as you keep the files in the same location that SnowConvert put them when it converted the code from the source. \n\nLet’s see this in action by resolving the issues.\n\n### Resolving Issues (powered by Cortex AI)\n\nSnowConvert will generate you an inventory of all issues that it encounters. This is in the issues.csv file. Let’s take a look at issues that we have available in this execution of SnowConvert.\n\nTo find the issues report, go to “VIEW OUTPUT” in the bottom of the SnowConvert application. \n\n![ViewOutput](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/viewOutput.png)\n\nThis will take you to a directory title “Conversion-\u003Cdatetime\u003E” within the directory you originally created in the project creation screen at the start of the project. This output will have three different sub directories:\n\n- Logs\n- Output\n- Reports\n\nLet’s first visit the reports directory to see what issues SnowConvert identified with this conversion.\n\n![ViewIssues](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/checkIssuesReport.png)\n\nThe issues.\u003Cdatetime\u003E.csv report will be available in the reports folder under the SnowConvert subfolder. In this report, you will find the type of each error as well as its location and a description of the error. There are three major types of error generated by SnowConvert:\n\n- **Conversion Error (EWI)**: generally, this is something that the tool could not convert or hasn’t seen before\n- **Functional Difference (FDM)**: this is code that has been converted, but may be functionally different in Snowflake. These errors should be treated as warnings, but paid close attention to during testing.\n- **Performance Reviews (PRF)**: this is something that SnowConvert identifies that will run in Snowflake, but may be suboptimal. You should consider optimizing this once you’re up and running in Snowflake.\n\nThere’s more information on each of these [in the SnowConvert documentation](https://docs.snowconvert.com/sc/general/technical-documentation/issues-and-troubleshooting). Let’s look at what we have in this execution:\n\n![ViewIssuesContent](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/checkIssuesReport2.png)\n\nWe can see that some issues share an issue code, but we can see specifically which file has the issue in it and a description of what the issue is. This is a very low number of issues. There will not be a ton of things to work through.\n\nThere are many approaches to dealing with the conversion issues generated by SnowConvert. We would recommend that you start in the same order you would want to deploy the objects: tables first, then views and functions, followed by procedures and scripts. You could pivot this table by the root folder (which will have the object name in it), then sort by severity and type. Depending on your timeline and business need, this would allow you to deal programmatically with the most critical issues in the order of deployment.\n\nHowever, in this lab, there are such a small number of issues, we don’t really need to pivot this file. However, we will still start with the same approach. Where are we going to deal with these issues? Snowflake would recommend that you use the Snowflake Visual Studio Code Extension, and that is what we are going to use right now.\n\nThroughout this lab, anytime that we are going to interact with the code, we are going to use the Snowflake extension for VS Code. We can open it now and leave it open for the duration of this lab. If you have not yet downloaded the [VS Code extension](https://docs.snowflake.com/en/user-guide/vscode-ext), you can do so from within VS Code or from the [VS Code Marketplace](https://marketplace.visualstudio.com/items?itemName=snowflake.snowflake-vsc).\n\nWhen you first open VS Code, navigate to the settings for your extensions. In the settings for the Snowflake extension, you will need to enable the SnowConvert Migration Assistant:\n\n![enableExtension](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/enableExtension.png)\n\nNote that we will be using Snowpark Checkpoints later on in this lab. While you’re here, you may as well enable that part of the extension as well.\n\nWhen you enable this, VS Code may prompt you to restart. Even if it doesn’t, it’s a good idea to restart the application now.\n\nOnce you’ve restarted VS Code, open the Snowflake extension and login to your Snowflake account. For this lab, Snowflake accounts have been created for you and provided via the information available for this lab. \n\nNow we should be ready, let’s open a new browser window, and make our root directory the same one we created in the project creation menu. This will allow us to access both the extracted source code and the output Snowflake code that has our issues in it.\n\nYou should see a folder that has the name of the project you created in the project creation screen:\n\n![vsFolderSelect](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/vsFolderSelect.png)\n\n\u003E \n\u003E Note: if you expand that folder, you should see an input and an output directory\n\nThe extracted code is in the input directory, and the Snowflake code generated by SnowConvert is in the output folder. But we will not need to navigate this folder structure to find our issues. We can use the Snowflake extension.\n\nOpen the Snowflake extension, and you should see a section called **SNOWCONVERT ISSUES**. It should automatically load the list of issues under the VS Code's root directory.\n\u003C!-- \n![OpenSCIssues](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/openSCIssues.png)\n\nIt should take you to the same folder that you already choose as your root directory. You can choose the folder that has the name of the project you created in the project creation:\n\n![issuesDir](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesDir.png)\n\nWhen you select this folder, it should show you a few additional folders. These are from the output directory where you generated the output code. These will be each of the files that has an issue in it:\n --\u003E\n![issuesRoot](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesRoot.png)\n\nLet’s expand this so we can see it a bit better:\n\n![issuesExpanded](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesExpanded.png)\n\nYou can see each of the files that has an EWI in it and the exact location.\n\n\u003E \n\u003E Note: for most migrations, this will be exceptionally large. It would be good to plan how to approach the issues using the issues.csv spreadsheet as mentioned earlier.\n\nIf you select a specific error, it will take you to the place in the code where that error can be found:\n\n![issuesCode](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode.png)\n\nLet’s not start with the first error posted, but instead start in the order that was recommended earlier: tables, views, functions, then procedures/scripts. In this example, we have 1 table (in the file **databaselog.sql**) with an error in it. Let’s take a look:\n\n![issuesCode2](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode2.png)\n\nThis error is an EWI saying that a datatype was converted to a different datatype in Snowflake. This doesn’t look like a problem, but more looks like SnowConvert is trying to tell us to validate that this is not a problem before we deploy. \n\nIf we’re not sure that this datatype should be converted to VARIANT, let’s use the SnowConvert Migration Assistant to generate some more information on this error. To do this, let’s select the light bulb icon in the SNOWCONVERT ISSUES menu in the extension:\n\n![issuesCode3](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode3.png)\n\n\u003E \n\u003E Note that a new window will pop up saying “Loading explanation and suggestion…”. This is connecting to your Snowflake account and passing the error message that SnowConvert generated to Cortex in your connected Snowflake account. It will give you an explanation after it has had a moment to think through it:\n\n![issuesCode4](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode4.png)\n\nThis explanation seems to validate that the conversion is correct. VARIANT is the right translation for an XML file. From this menu you can:\n\n- Copy the replacement code that it has generated\n- Chat with Cortex\n- Return to the issue\n\nLet’s say that we are not sure that this is the right conversion. We could ask Cortex if there are any other options for XML data in Snowflake.\n\n![issuesCode5](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode5.png)\n\nYour results may vary, but this is what Cortex returned to me. VARIANT is still the recommendation. We’ll take a look that the data matches depending on how we are using this XML data when we test the result of this conversion.\n\nFor now, let’s replace the code with the EWI with the code that Cortex has recommended (which is essentially, to remove the EWI). Our new code looks like this:\n\n```sql\nCREATE OR REPLACE TABLE dbo.DatabaseLog (\n    DatabaseLogID INT NOT NULL,\n    PostTime TIMESTAMP_NTZ(3) NOT NULL,\n    DatabaseUser VARCHAR(128) NOT NULL,\n    Event VARCHAR(128) NOT NULL,\n       \"Schema\" VARCHAR(128) NULL,\n    Object VARCHAR(128) NULL,\n    TSQL VARCHAR NOT NULL,\n    XmlEvent VARIANT NOT NULL\n   )\n   COMMENT = '{ \"origin\": \"sf_sc\", \"name\": \"snowconvert\", \"version\": {  \"major\": 1,  \"minor\": 6,  \"patch\": \"0.0\" }, \"attributes\": {  \"component\": \"transact\",  \"convertedOn\": \"05/22/2025\",  \"domain\": \"snowflake\" }}'\n   ;\n```\n\nAnd the EWI message is gone. Note, however, that the file and issue still appear in the SnowConvert Issues section of the Snowflake extension even after we refresh the list:\n\n![issuesCode7](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode7.png)\n\n\u003E \n\u003E To remove this, we need to save the changes that we’ve made to the file. This will allow us to update the extension and the SnowConvert application. Let’s take a look at this.\n\n![issuesRefresh](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesRefresh.png)\n\nNow let’s refresh the SNOWCONVERT ISSUES list in the Snowflake extension, and the issue no longer appears. We can see this same behavior in the SnowConvert application.\n\nIf you return to the application, we should still be on the deployment screen. If you expand the “Tables” section of the deployment screen, you will see that “DatabaseLog” is marked with a red “X” indicating that it cannot be deployed.\n\n![issuesCode8](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode8.png)\n\nHowever, now that we’ve resolved the EWIs in the DDL for that object, we can refresh this list by selecting the refresh icon in the top left of the application:\n\n![issuesCode9](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode9.png)\n\nWe should see that the red “X” now changes to a green checkmark.\n\nIf it does not change, then there are still errors associated with that DDL that need to be resolved. That does not appear to be the case here.\n\nWe can now continue to resolve the rest of the errors in our execution. The tables now all appear to have their issues resolved. (You can validate that they are ready to deploy by scrolling through the catalog in SnowConvert to see that all objects are marked with a green checkmark.)\n\nIf you are doing a large project, it might be a good idea to go ahead and deploy the tables. To validate that those are correct. However, views and functions should both be done before they are deployed. \n\nLet’s take a look at the remaining issues we have:\n\n![issuesCode15](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode15.png)\n\nWe have two remaining issues in two different functions, but they have the same error code. Note that this is also a FDM code, so there may not be a solution we need to apply here. Let’s select the first issue and call Cortex to tell us more about it. \n\nThis is an interesting issue. SnowConvert has decided that this UDF should be turned into a stored procedure. This is an interesting decision as a stored procedure cannot be called in the same way as a function. But there are things that you can do with functions in SQL Server that you cannot do with a function in Snowflake. And if you ask Cortex what it thinks, it will disagree with SnowConvert and turn this back into a UDF:\n\n```sql\nCREATE OR REPLACE FUNCTION dbo.udfMinimumDate(X TIMESTAMP_NTZ(3), Y TIMESTAMP_NTZ(3))\nRETURNS TIMESTAMP_NTZ(3)\nLANGUAGE SQL\nAS\n$$\n  IFF(X \u003C= Y, X, Y)\n$$\n```\n\nThis is where you need someone familiar with how this function is used across the workload. Normally, we would inventory any reference to this function across the entire workload. Since this is a small POC, this function is not used anywhere else in this function. You can validate this in a couple of ways. The first is to simply search for the name of this object in the input directory in VS Code: \n\n![issuesCode17](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode17.png)\n\nThere is only reference to this function and it is in the create function statement. So it is not being used anywhere. We can also check the dependent objects inventory generated by SnowConvert. (You could do this in a file browser or in VS code directly.)\n\n![issuesCode18](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode18.png)\n\nWe can see that while there are many mentions of this object in the reports, there is no mention of this object in the **ObjectReferences.\u003Cdatetime\u003E.csv** report. As a result, this function is not used anywhere, so we’re not sure how this will be called.\n\nSo, should we go with what SnowConvert has generated or listen to Cortex? There are advantages and disadvantages to both, but the primary reason would be turning this function into a procedure means it WILL fail if it is called in a way that a procedure cannot be called. This means that you will be immediately alerted to this when you test this output. Leaving it as a function means that there may be functional differences that you may not immediately be aware of. \n\nIn this situation:\n\n- Interpreting what the function is trying to do versus the way it is written, I think this could be a UDF.\n- It’s also likely that this is going to be called in a way that is not going to work for a procedure. \n\nSo I’m going to make the choice to use the Cortex recommended code and put in the function code. However… I will add my own comment to the file that contains the code as a stored procedure. That way I can revisit it if I need to when it comes time to test these functions. That looks like this:\n\n```sql\nCREATE OR REPLACE FUNCTION dbo.udfMinimumDate(X TIMESTAMP_NTZ(3), Y TIMESTAMP_NTZ(3))\nRETURNS TIMESTAMP_NTZ(3)\nLANGUAGE SQL\nAS\n$$\n  IFF(X \u003C= Y, X, Y)\n$$\n\n-- SnowConvert turned this into a stored procedure. Leaving as a comment.\n--** SSC-FDM-0029 - USER DEFINED FUNCTION WAS TRANSFORMED TO SNOWFLAKE PROCEDURE **\n/* CREATE OR REPLACE PROCEDURE dbo.udfMinimumDate (X TIMESTAMP_NTZ(3), Y TIMESTAMP_NTZ(3))\nRETURNS VARCHAR\nLANGUAGE SQL\nCOMMENT = '{ \"origin\": \"sf_sc\", \"name\": \"snowconvert\", \"version\": {  \"major\": 1,  \"minor\": 6,  \"patch\": \"0.0\" }, \"attributes\": {  \"component\": \"transact\",  \"convertedOn\": \"05/23/2025\",  \"domain\": \"snowflake\" }}'\nEXECUTE AS CALLER\nAS\n$$\n    DECLARE\n        Z TIMESTAMP_NTZ(3);\n    BEGIN\n         \n        IF (:X \u003C= :Y) THEN\n            Z := :X;\n        ELSE\n            Z := :Y;\n        END IF;\n        RETURN (:Z);\n    END;\n$$; */\n```\n\n\u003E \n\u003E Note that I have left the FDM message in the code. This means that the SnowConvert catalog will still alert me that there is a warning present in this function.\n\nI will save the file, and update the SNOWCONVERT ISSUES list in the Snowflake VS Code Extension. Note that this still leaves the FDM.\n\n\u003E \n\u003E Note that this still leaves FDM in the browser because I intentionally left it in the file.\n\nI will collapse this and take a look at the last issue. Since this issue has the same error code as the previous one, it is likely to be a similar case with a similar solution.\n\n```sql\nCREATE OR REPLACE FUNCTION dbo.udfTwoDigitZeroFill(NUMBER INT)\nRETURNS VARCHAR\nLANGUAGE SQL\nAS\n$$\n  CASE \n      WHEN NUMBER \u003E 9 THEN CAST(LEFT(NUMBER::STRING, 2) AS CHAR(2))\n      ELSE CAST(LEFT('0' || NUMBER::STRING, 2) AS CHAR(2))\n  END\n$$\n```\n\nAnd… it is a similar issue. This function is doing something slightly different, but I will make the same decision here that I made before to trust that the function will give me the right result, preserve the procedure as a comment, and test this later on. \n\nLet’s save the file, and now… we are done! We have successfully navigated through the issues generated by SnowConvert.\n\nNote that there are a large number of issues that SnowConvert can generate. The ones that we see here are generally something that you need to make a decision on the best solution path for your situation. \n\nLet’s validate this in SnowConvert. If you’ve saved each of the files, you should be able to see checkmark across all of the objects, though there will still be some blue “i” icon:\n\n![deployementReady](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentReady.png)\n\nLet’s move forward in the process and deploy the code. This will run all of the DDL that we have extracted from SQL Server (and subsequently, resolved any issues with) to Snowflake. To deploy the code, select “DEPLOY” at the bottom of the application:\n\n![deploymentClick](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentClick.png)\n\n### Deploy objects\n\nIn order to deploy, you need to be connected to a Snowflake account. You will be prompted for your Snowflake connection information:\n\n![connectToSnowflake](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/connectToSnowflake.png)\n\nWhen you enter your login information, you may be taken to a webpage with this message.\n```Your identity was confirmed and propagated to Snowflake Node.js driver. You can close this window now and go back where you started from.```\nMost Snowflake accounts require authentication and this is a part of that authentication. You can return to the SnowConvert application to resume the deployment process. You should see some results from your deployment similar to this:\n\n![deploymentResults](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentResults.png)\n\nNow, we know there should be more than 1 object here. There is one error on deployment. You may also see an error message like this one:\n\n![deployError](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deployError.png)\n\nIf you do see this, you can see why the error occurred in the catalog in SnowConvert by hovering your mouse over the top of the error symbol.\n\n![deployErrorStatus](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deployErrorStatus.png)\n\nIn this case, there is a permissions problem. Check the connection you are using. If this fails, you may want to create the DATABASE and the SCHEMA manually in the Snowflake, especially if the database you are creating already exists. You could troubleshoot this in SnowConvert, but it would be better to do so in Snowflake.\n\nI have gone into Snowflake and created the database and schema we are using with the same credentials I am using to deploy in SnowConvert. Now I will try it again. This time, let’s just start with the tables. I will select the tables:\n\n![deployStatusPane](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deployStatusPane.png)\n\nThis time, the results will show that we were able to successfully deploy some objects.\n![deployResults2](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/deploymentResults2.png)\n\nThis can be validated by querying the database in Snowflake:\n\n```sql\ndescribe schema adventureworks.dbo\n```\n\nI can see each of the tables that were just deployed. Now, let’s do the views and functions to complete the deployment. When we do, some of the functions and views deploy successfully, but there are still some errors.\n\nLet’s see what is in those errors. If you return to the menu and hover over an object in the deployment column, it will tell you a bit more about why the object was not deployed:\n\n![issuesCode22](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode22.png)\n\nIn this case, the view that we are hovering over has a dependency on a function. This function exists in this deployment, but SnowConvert is using a “brute force” method of deployment. This brute force method of deployment will lead to errors oftentimes because it is trying different combinations of deployment methods as opposed to ordering the deployment based on dependencies.\n\nWe can manually resolve this in the SnowConvert application though. Let’s start by seeing which object may have the first level of dependencies:\n\nIt looks like the view vTimeSeries is dependent on function **udfBuildISO8601Date**, and **udfBuildISO8601Date** is dependent on function **udfTwoDigitZeroFill**. \n\nLet’s try the last two objects in order with the function first, followed by the last view.\n\n![issuesCode25](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode25.png)\n\nAnother failure! Looks like the timestamp type that was moved from SQL Server to Snowflake was not compatible. So we need to change it to be compatible. If we look at a side by side comparison in VS Code of the source SQL Server code and the Snowflake code, we can see that the code in SQL Server is only returning a “datetime” value, not a TIMESTAMP_NTZ(3).\n\nSo let’s change the TIMESTAMP_NTZ(3) value to a TIMESTAMP_NTZ(9), but make a note to review this during testing.\n\n```sql\nCREATE OR REPLACE FUNCTION dbo.udfBuildISO8601Date (YEAR INT, MONTH INT, DAY INT)\nRETURNS TIMESTAMP_NTZ(9)\nLANGUAGE SQL\nCOMMENT = '{ \"origin\": \"sf_sc\", \"name\": \"snowconvert\", \"version\": {  \"major\": 1,  \"minor\": 6,  \"patch\": \"0.0\" }, \"attributes\": {  \"component\": \"transact\",  \"convertedOn\": \"05/26/2025\",  \"domain\": \"snowflake\" }}'\nAS\n$$\n\tSELECT\n\t\tTO_TIMESTAMP_NTZ(CAST(YEAR AS VARCHAR) || '-' || dbo.udfTwoDigitZeroFill(MONTH) || '-' || dbo.udfTwoDigitZeroFill(DAY) || 'T00:00:00', 'MM-DD-YYYY')\n$$;\n```\n\nSave the file and refresh the SnowConvert application. Let’s try to deploy the last function again.\n\n![issuesCode27](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode27.png)\n\nSuccess! We have all of our tables and functions deployed. There was one view that was not able to be deployed, but it was dependent on the functions being deployed. Let’s a take a look at that view right now. Let’s try to deploy the last view:\n\n![issuesCode28](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/issuesCode28.png)\n\nSuccess again! We have officially deployed all of the schema to Snowflake. There’s one more thing we can do in the SnowConvert application, and that is move the data from the original SQL Server database into Snowflake. Select “GO TO DATA MIGRATION” at the bottom right of the screen.\n\n## Data Migration\n\nThe Data Migration page will give you the same catalog that you can see on the previous screens. For the migration of the data, note that only the data in tables will be migrated. The rest of the DDL is already present, but there is no “data” present in a view or a function, so the tables will be what is migrated. Also note that no data can be migrated to a table where the DDL wasn’t already successfully deployed. \n\nTo migrate the data, select all of the tables (or you can choose the entire database or schema) and choose “MIGRATE DATA” at the bottom of the application:\n\n![md1](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md1.png)\n\nWhile the data is migrating, it will let you know how it’s doing:\n\n![md2](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md2.png)\n\nIt’s important to note that SnowConvert is migrating the data by passing it through your local machine. It is running a select statement on the data in the original object, then doing a COPY INTO to the data in the Snowflake object. This means that the speed of the data transfer will be directly affected by your connection from the local machine where you are running SnowConvert.\n\nIf you are moving a large amount of data, this will not be the most efficient way to move that data. You should consider using another data movement or ETL solution. However, in this scenario, this works just fine for us. There are only 30 tables. \n\nNote that if you are connected via SSO, Snowconvert may take you to a login page in your browser. If you have MFA setup on your Snowflake account, you will be taken to your authenticator to validate your connection.\n\nLooks like we did not receive any error message for this migration. Let’s expand the object catalog to see if all the tables were migrated:\n\n![md3](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md3.png)\n\nSelect “VIEW DATA MIGRATION RESULTS” to see a summary of what just happened.\n\n![md5](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md4.png)\n\nThis will show you each table that you just migrated and the corresponding table in Snowflake. For a basic validation of what just happened, you can review the last two columns of the Data Migration Results. These are the row counts for the source table and the count of rows that were loaded into the table in Snowflake.\n\nYou can go into Snowflake and write a queries to check to see if this will work. Here is a simple “select *” on the largest table we loaded:\n\n![md6](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/md6.png)\n\nWhere once there was no data, now… there is data. At this point, we would start running validation tests on the data to confirm that everything is the same. For this POC, we only have a few tests that we want to validate, but they are all related to the pipelines we are running. Let’s do our validations once we move the pipelines.\n\n### Review\n\nBefore we get to our pipelines, let’s take a quick look back.\n\nIf we review the steps that we did, let’s see how they fit in with our migration process:\n\n- Setup our project in SnowConvert\n- Extracted the schema from the source database\n- Ran SnowConvert’s assessment and conversion engine on the extracted schema\n- Reviewed the output reporting to better understand what we have\n- Reviewed what could not be converted in the VS Code Extension\n- Generated new code in the VS Code Extension\n- Deployed the code to Snowflake\n- Moved the data from the source to Snowflake\n- All the while, we were able to track things in our object inventory.\n\nThis is what we have just walked through. These steps will generally be recommended regardless of how you used SnowConvert. \n\nAnd you can generally see that most of these activities fit our **assess** -\u003E **convert** -\u003E **validate** flow:\n\n- Setup our project in SnowConvert so we can better assess what we have\n- Extracted the schema from the source database so we can see what kind of DDL we are working with\n- Ran SnowConvert’s assessment and conversion engine on the extracted schema to analyze what we have for our initial assessment, and generate the output converted code\n- Reviewed the output reporting to better understand what we have to build our assessment of what we have\n- Reviewed what could not be converted in the VS Code Extension to assess what work needs to be done\n- Generated new code in the VS Code Extension to resolve any errors in the initial conversion\n- Deployed the code to Snowflake to validate that the schema works\n- Moved the data from the source to Snowflake to convert our data into Snowflake data\n- All the while, we were able to track things in our object inventory to better understand where we are in the migration process.\n\nAs we continue to walk through the migration process, let’s take a look at our data pipelines. \n\n## Pipeline Assessment\n\nMoving the logic and data in a data warehouse is essential to getting an operational database on a new platform. But to take advantage of the new platform in a functional way, any pipelines running moving data in or out of that data platform need to be repointed or replatformed as well. This can often be challenging as there are usually a variety of pipelines being used. This section of the lab will focus on just one for which Snowflake can provide some acceleration. But note that new ETL and pipeline accelerators are constantly being developed.\n\nLet’s talk about the pipeline and the notebook we are moving. \n\n- The pipeline is a Spark script that is reading an accessible file generated by an older POS system in a local directory at regular intervals. (This will be referred to as the pipeline script.)\n- The notebook is a reporting notebook that reads from the existing SQL Server database and reports on a few summary metrics. (This will be referred to as the reporting notebook.)\n\nBoth of these use Spark and access the SQL Server database. So our goal is essentially to move the operations in Spark into Snowpark. Let’s see how we would do that using Snowpark Migration Accelerator (SMA). Recall that while many of the steps we will walk through are similar to what we just did with SnowConvert, we are still essentially working through the **assessment** -\u003E **conversion** -\u003E **validation** flow. \n\n**Notes on this Lab Environment**\n\nThis lab uses the Snowpark Migration Accelerator and the Snowflake VS Code Extension. But to make the most of this, you will need to run Python with a PySpark. The simplest way to start this would be to start an environment with [the anaconda distribution](https://www.anaconda.com/docs/getting-started/anaconda/main). This will have most of the packages needed to run the code in this lab.\n\nYou will still need to make available the following resources:\n- Python Libraries ([PySpark](https://pypi.org/project/pyspark/), [Snowpark Python](https://pypi.org/project/snowflake-snowpark-python/), and [Snowflake](https://pypi.org/project/snowflake/))\n- VS Code Extensions ([Snowflake](https://marketplace.visualstudio.com/items?itemName=snowflake.snowflake-vsc), [Python](https://marketplace.visualstudio.com/items?itemName=ms-python.python), and [Jupyter](https://marketplace.visualstudio.com/items?itemName=ms-toolsai.jupyter)) \n- Other supporting elements like the [PySpark JDBC Driver for SQL Server](https://learn.microsoft.com/en-us/sql/connect/jdbc/download-microsoft-jdbc-driver-for-sql-server?view=sql-server-ver17).\n\nHaving said all of this, you can still run this lab with just a Snowflake account, the SMA, and the Snowflake VS Code Extension. You will not be able to run everything that connects to a local machine, but you will be able to use all of the converted elements in Snowflake.\n\nNow let’s get started by assessing what we have.\n\n**Extraction**\n\nFor the purpose of this lab, we will assume that the notebook and script file that we are converting are already accessible as files. [Download those files from the Snowpark Migration Accelerator's docs site](https://docs.snowconvert.com/sma/use-cases/pipeline-migration-lab/pipeline-lab-assessment). \n\nNote that there is also a data file as well: ‘customer_update.csv’. This is a sample of the file being generated locally by the Point of Sale (POS) system that AdventureWorks is currently using. While that system is also being updated, this Proof of Concept (POC) is focused on making the existing pipeline work with Snowpark instead of Spark. \n\nLet’s take each of these files, and drop them into a single directory on our local machine:\n\n![sma_source_code_directory_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_source_code_directory_01.png)\n\nIt would be recommended to create a project directory. This can be called whatever you like, but as a suggestion for this lab, let’s go with **spark_adw_lab**. This means we would create a folder with the name spark_adw_lab, then create another folder in that directory called source_files (the path being something like **/your/accessible/directory/spark_adw_lab/source_files**). This isn’t required, but will help keep things organized. The SMA will scan any set of subdirectories as well, so you could add specific pipelines in a folder and notebooks in another.\n\nNote that in general, if you are extracting files that are being orchestrated by a specific tool, you may need to export them. If you are using notebooks in the Cloud elsewhere, you can export those as .ipynb files just as the jupyter notebook we are going to run through the SMA today. \n\n**Access**\n\nNow that we have our source files in an accessible directory, it is time to run the SMA.\n\nIf you have not already downloaded it, the SMA is accessible from [the Snowflake website](/en/migrate-to-the-cloud/migration-accelerator/). It is also accessible from the Migrations page in SnowSight in your Snowflake account:\n\n![sma1](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma1.png)\n\nOnce you download the tool, install it! There is more information on [installing the SMA](/en/migrate-to-the-cloud/migration-accelerator/) in the SMA documentation.\n\n**Using the Snowpark Migration Accelerator**\n\nOnce you have installed the tool, open it! When you launch the SMA, it will look very similar to its partner tool, SnowConvert. Both of these tools are built on a similar concept where you input code files into the tool and it runs. We have seen that SnowConvert can take the DDL and data directly from the source and input it directly into Snowflake. The SMA does not do this. It only takes in code files as a source and outputs those files to something that is compatible with Snowflake. This is primarily because the tool does not know how a user will orchestrate their spark code. \n\nOnce you have launched the tool, It will ask you if you would like to create a new project or open an already existing one:\n\n![sma2](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma2.png)\n\nThis will take you to the project creation screen:\n\n![sma3](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma3.png)\n\nOn this screen, you will enter the relevant details for your project. Note that all fields are required. For this project, you could enter something similar to:\n\n- Project Name: **Spark ADW Lab**\n- Input Folder Path: **/your/accessible/directory/spark_adw_lab/source_files**\n- Output Folder Path (the SMA will auto generate a directory for the output, but you can modify this): **/your/accessible/directory/spark_adw_lab/source_files_output**\n- Email Address: **your.name@your_domain.com**\n- Customer’s Company: **Your Organization**\n\nA couple of notes about this project creation screen:\n\n- The email and company fields are to help you track projects that may be ongoing. For example, at any large SI, there may be multiple email addresses and multiple organizations on behalf of whom a single user may run the SMA. This information is stored in the project file created by the SMA.\n- There is a hidden field for SQL. Note that the SMA can scan/analyze SQL, but it does not convert any SQL.It also can only identify SQL in the following circumstances:\n  - SQL that is in .sql files\n  - SQL that is in SQL cells in a Jupyter Notebook\n  - SQL that is passed as a single string to a spark.sql statement. \n- While this SQL capability can be helpful to determine where there is incompatible SQL with Snowflake, it is not the primary use for the SMA. More support for Spark SQL and HiveQL are coming soon.\n\nOnce you’ve entered all of your project information, for this lab, we are going to **skip** the assessment phase. (What… aren’t we building an assessment?) If you do not want to convert any code, running an assessment can be helpful as it will allow you to get the full set of reports generated by the SMA. You can then navigate through those or share them with others in your organization while not creating extra copies of the converted code. However, all of these same assessment reports are also generated during a conversion. So we will skip assessment mode for now and go to conversion by selecting “SAVE & SKIP ASSESSMENT” in the bottom right corner of the application.\n\n![sma4](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma4.png)\n\nThis will take you to the conversion screen. On this screen you will again see the input and output directory fields, but those will have already been populated by what you have entered in the project creation page. The only new field here will be to enter an access code. Access codes are freely available, but must still be requested an active to use the Snowpark Migration Accelerator (SMA). And while the mechanism for generating these access codes is similar to SnowConvert, the access codes for SnowConvert will not work with the SMA. You will have to get another one.\n\nYou can request an access code by selecting “Inquire about an access code” out to the side of the “Enter access code…” field: \n\n![sma5](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma5.png)\n\nWhen you select this, a pop up menu will appear asking you who you are so an access code can be generated:\n\n![sma6](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma6.png)\n\nFill out all of the fields shown above, and ensure that you enter a valid email address. In the project creation screen earlier, you entered an email address to associate with the project you were creating. However, nothing was actually sent to that email. That was only to track your project locally. This form will trigger an access code to be sent to the email you enter.\n\nOnce you submit the form, you should receive an email with an access code shortly. The email will come from sma-notifications@snowflake.com, and it will look something like this:\n\n![sma7](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma7.png)\n\nIn the image above, where it says \"**your access code here**\", you should see a series of numbers, letters, and dashes. Copy that string, and paste it in the access code field for the SMA: \n\n![sma8](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma8.png)\n\nWhen you paste the value into the box, the SMA will validate the access code. A successful validation will show the access code details below the access code dialog box:\n\n![sma9](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma9.png)\n\nTo validate the access code, the SMA will call out to the Snowflake licensing API. If you are not connected to the internet, the tool will not be able to validate the access code and you will get an error message. If you need to run the tool in a completely offline environment, please reach out to [sma-support@snowflake.com](mailto:sma-support@snowflake.com) for help validating an access code.\n\nNow that the access code has been validated, you can take a look at the conversion settings:\n\n![sma_go_to_conversion_settings_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_go_to_conversion_settings_01.png)\n\nThere is one setting that will simplify the output of this hands on lab, which would be to **disable** the attempted conversion of pandas dataframes to the Snowpark API:\n\n![sma_conversion_settings_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_conversion_settings_01.png)\n\nThis one setting is currently being updated, so there will be a lot of additional warnings added if this option is not deselected. These can be worked though, but this will add noise to this lab. Most of the functions related to pandas dataframes can be used as part of the modin implementation of pandas, so a simple import call change should suffice for now. \n\nYou can look at the other settings, but we will leave them as is. It’s important to note that there is a testing library that the output code is compatible with called **Snowpark Checkpoints**. There are settings related to this, but we will not alter them in this lab. Select “CLOSE” in the dialog box to save and close your settings. \n\n![sma_conversion_settings_02](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_conversion_settings_02.png)\n\nThe “START CONVERSION” option will be available in the bottom right corner of the application. Let’s start the conversion by selecting this option.\n\nThe next screen will show the progress of the conversion:\n\n![sma10](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma10.png)\n\nLike SnowConvert, the SMA is building a semantic model of the entire codebase in the input directory. It is building relationships between code elements, sql objects, and other referenced artifacts, and creating the closest output it can to a functional equivalent for Snowflake. This primarily means converting references from the Spark API to the Snowpark API. The SMA’s engineering team is a part of the Snowpark engineering team, so most transformations that take place have been built into the Snowpark API, so the changes may seem minor. But the wealth of assessment information that is generated by the SMA allows a migration project to really get moving forward. An in-depth look at all of the generated assessment information will have to take place elsewhere because the SMA has likely finished this conversion in the time it took to read this paragraph.\n\nWhen the SMA has finished, the “VIEW RESULTS” option will be available in the bottom right corner:\n\n![sma11](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma11.png)\n\nThe results page, shockingly enough, will show the results.\n\n![sma_results_page_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_results_page_01.png)\n\nThe results page has some “Readiness Scores” that are very simplified metrics on how “ready” this codebase is for Snowflake. We will review the results next, but note that running the Snowpark Migration Accelerator is the easy part. Note that this is just an “accelerator”. It is not a silver bullet or a hands-off automation tool. Pipelines that connect to one data source and output to another are not fully migrated by this tool will always need more attention than a straight SQL-to-SQL migration of DDL as is done by SnowConvert. But Snowflake is continuously working towards making this as simple as possible.\n\n**Interpreting the Output**\n\nThe SMA, even more so than SnowConvert, generates a large amount of assessment information. It can be difficult to parse through the results. There are many different directions you could go depending on what you want to achieve. We'll try to cut through some of this given the simplicity of the scenario, but note that all of this data can be valualbe when planning a migration, particularly a large migration.\n\n**Readiness Scores**\n\nWith that in mind, let’s take a look at the first part of the output that you will see in the application: the readiness scores. There will be multiple readiness scores and you can expand on each one of them to better understand what is captured by that readiness score. \n\n![sma_scores_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_scores_01.png)\n\nEach readiness score is a very basic calculation of the count of functions or elements in an API that are supported in Snowpark/Snowflake divided by the count of all functions or elements related to that API for this execution. The calculation showing you how the score is calculated is shown when you expand the window. You can also learn more about how to interpret the readiness scores by selecting “How to read through the scores” near the top left corner of this window. \n\nThis execution has a Spark API Readiness Score of **97.92%**. (Please note that yours may be different! These tools are updated on a biweekly basis and there may be a change as compatibility between the two platforms is ever evolving.) This means that 97.92% of the references to the Spark API that the tool identified are supported in Snowflake. “Supported” in this case means that there could be a similar function that already exists or that the SMA has created a functionally equivalent output. The higher this score is, the more likely this code can quickly run in Snowflake. \n\nThere are other readiness scores and you may see more than what is shown in the lab as the readiness scores do change over time. This lab won’t walk through each of them, but note that a low score will always be worth investigating.  \n\n**Code Analyzed**\n\nJust below each of the readiness scores, will be a small indicator that lets you know if there was any code that could not be processed:\n\n![sma_code_analyzed_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_analyzed_01.png)\n\nThis number represents the **percentage of files** that were fully parsed. If this number is less than 100%, then there is some code that the SMA could not parse or process. This is the first place you should start looking to resolve problems. If it’s less than 100%, you should see where the parsing errors occurred by looking at the issue summary. This is the first place you should look when working through the SMA’s output because it’s the only one where it might make sense to run the tool again if a large amount of code was not able to be scanned.\n\nIn this case, we only have **50%** of our workload successfully parsed. Tragic. Now, this might seem like something we should panic about, but let’s not be too quick to judge. We only have 2 files, and we don’t yet know how many parsing errors we have.\n\nRegardless of the result of this number, the last place we will visit on this page is the **Issue Summary**. Scroll down in the application until you see this summary:\n\n![sma_issue_summary_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_issue_summary_01.png)\n\n**Issue Summary**\n\nIssues are one of the key elements of both SnowConvert and the SMA. Each tool is attempting to create a functional equivalent output based on the input that it receives, but no conversion is 100% automated. These tools know this, and mark everything that cannot be converted or even might need extra attention with an issue. These issues are summarized in a spreadsheet, but are also written as comments directly into the output code.\n\nThe issue summary in the UI highlights issues that were found in this execution of the tool. These issues are often referred to with the acronym EWI (error, warning, and issue). Similar, but not identical to SnowConvert, the SMA generates three types of issues:\n- **Parsing Error** - This type of issue is considered critical and will need you deal with it immediately. Because of the way the SMA works, having code that does not parse, could mean missing information in the reports and missing conversion piece as well. \n- **Conversion Error** - This is something the SMA recognizes (or at least thinks that it recognizes), but it cannot convert for one reason or another. These errors usually have very specific issue codes and should be dealt with next.\n- **Warning** - These error codes identify code that the SMA did convert or is something that has an equivalent in Snowpark/Snowflake, but there may be issues when you do testing. There may not be 100% functional equivalence. \n\nThere is more information on issue types on [the SMA documentation page](https://docs.snowconvert.com/sma/issue-analysis/issue-code-categorization), but in the issue summary shown above, one can find the code, count, level, and description for the unique issues present. Even if there are a lot of issues present, the fewer unique issues there are, the more likely they can be dealt with programmatically. For more information on each unique issue code, you can click the code in the UI. This will take you to the SMA documentation page for that specific issue.  \n\nLooks like we have some conversion errors, warnings, and 1 parsing error. This means there was 1 thing the tool could not read. (Note that if you get a lot of error codes that start with PND, then you may not have deselected that option in the conversion settings. Now worries, if you see those, you can ignore them.)\n\nRegardless of how many issues you have, it is always recommended to explore the detailed issues file if you’re ready to start migrating. This is a csv file that is stored locally on the machine where you ran the SMA. You can find this file by selecting the “VIEW REPORTS” option in the bottom right of the SMA:\n\n![sma_view_reports_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_view_reports_01.png)\n\nThis will take you to the local directory that has all of the reports… and as of this writing, there are a lot of reports generated by the SMA:\n\n![sma_reports_folder_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_reports_folder_01.png)\n\nEach of these reports has some valuable information in it depending on how you are using the SMA. For now, we will only look at the issues.csv file, but note there is more information on EVERY report and inventory generated by the SMA [in the SMA documentation](https://docs.snowconvert.com/sma/user-guide/assessment/output-reports). \n\nWhen you open the issues file, it will look something like this:\n\n![sma_issue_report_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_issue_report_01.png)\n\nNote the schema of this report. It tells you the issue code, a description of the issue, the type of issue (category), the file each issue is in, the line number of the file each issue is in, and provides a link to the documentation page for that specific issue. All of this is helpful information when navigating through the issues. \n\nYou can pivot this by file to see what type of issue you have by file to find where the parsing error is for this workload:\n\n![sma_issue_report_02](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_issue_report_02.png)\n\nLooks like we only have a few issues, and our parsing error is in the pipeline python script. That’s where we want to start. \n\nNormally, we would take a look at one other report in our Reports directory, the **ArtifactDependencyInventory.csv** file. But this is such a small execution, let’s take a look at what’s actually in these output files now, and see if we can’t get it to run in (or with) Snowflake.\n\n## Pipeline Conversion\n\nThe SMA has “converted” our scripts, but has it really? What it has actually done is converted all references from the Spark API to the Snowpark API, but what it has not done is to replace the connections that may exist in your pipelines. \n\nThe SMA’s power is in the assessment reporting that it does as the conversion is tied to converting references from the Spark API to the Snowpark API. Note that the conversion of these references will not be enough to run any data pipeline. You will have to ensure that the pipeline’s connections are resolved manually. The SMA cannot assume to know connection parameters or other elements that are likely not available to be run through it. \n\nAs with any conversion, dealing with the converted code can be done in a variety of ways. The following steps are how we would **recommend** that you approach the output of the conversion tool. Like SnowConvert, the SMA requires attention to be paid to the output. No conversion will ever be 100% automated. This is particularly true for the SMA. Since the SMA is converting references from the Spark API to the Snowpark API, you will always need to check how those references are being run. It does not attempt to orchestrate the successful execution of any script or notebook run through it. \n\nSo we’ll follow these steps to work through the output of the SMA that will be slightly different than SnowConvert:\n- **Resolve All Issues**: “Issues” here means the issues generated by the SMA. Take a look at the output code. Resolve parsing errors and conversion errors, and investigate warnings.\n- **Resolve the session calls**: How the session call is written in the output code depends on where we are going to run the file. We will resolve this for running the code file(s) in the same location as they were originally going to be run, and then for running them in Snowflake.\n- **Resolve the Input/Outputs**: Connections to different sources cannot be resolved entirely by the SMA. There are differences in the platforms, and the SMA will usually disregard this. This also is affected by where the file is going to be run. \n- **Clean up and Test**! Let’s run the code. See if it works. We will be smoke testing in this lab, but there are tools to do more extensive testing and data validation including Snowpark Python Checkpoints.\n\nSo let’s take a look at what this looks like. We’re going to do this with two approaches: the first approach is to run this in Python on the local machine (as the source script is running). The second would be to do everything in Snowflake… in Snowsight, but for a data pipeline reading from a local source, this will not be 100% possible in Snowsight. That’s ok though. We are not converting the orchestration of this script in this POC.\n\nLet’s start with the pipeline script file, and get to the notebook in the next section.\n\n**Resolve Issues**\n\nLet’s open our source and our output code in a code editor. You can use any code editor of your choice, but as has been mentioned multiple times, Snowflake would recommend using **VS Code with the Snowflake Extension**. Not only does the Snowflake Extension help navigate through the issues from SnowConvert, but can also run **Snowpark Checkpoints** for Python, which would help with testing and root cause analysis (though just barely out of scope for this lab). \n\nLet’s open the directory that we originally created in the project creation screen (Spark ADW Lab) in VS Code:\n\n![sma_vscode_files_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_vscode_files_01.png)\n\nNote that the **Output** directory structure will be the same as the input directory. Even the data file will be copied over despite no conversion taking place. There will also be a couple of **checkpoints.json** files that will be created by the SMA. These are json files that contain instructions for the Snowpark Checkpoints extension. The Snowflake extension can load checkpoints into both the source and output code based on the data in those files. We will ignore them for now.\n\nFinally, let’s compare the input python script with the converted one in the output script. \n\n![sma_code_editor_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_01.png)\n\nThis is a very basic side-by-side comparison with the original Spark code on the left and the output Snowpark compatible code on the right. Looks like some imports were converted as well as the session call(s). We can see an EWI at the bottom of the image above, but let’s not start there. We need to find the parsing error before we do anything else. \n\nWe can search the document for the error code for that parsing error that was shown in both the UI and the issues.csv: **SPRKPY1101**.\n\n![sma_code_editor_02](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_02.png)\n\nLooking for where this EWI shows up in the pipeline_dimcustomer.py, we can see that the parsing error was present in the very last line of the source code.\n\n```python\n# Conversion Input.\nsome rogue code that doesn't make any sense!\n\n# Conversion Output.\nsome\n# EWI: SPRKPY1101 =\u003E Unrecognized or invalid CODE STATEMENT @(131, 6). Last valid token was 'some' @(131, 1), failed token 'rogue' @(131, 6)\n#     rogue code that doesn't make any sense!\n```\n\nLooks like this parsing error was because of… “some rogue code that doesn’t make any sense!”. This line of code is at the bottom of the pipeline file. This is not unusual to have extra characters or other elements in a code file as part of an extraction from a source. Note have the SMA detected that this was not valid Python code, and it generated the parsing error. \n\nYou can also see how the SMA inserts both the error code and the description into the output code as a comment where the error occurred. This is how all error messages will appear in the output. \n\nSince this is not valid code, it is at the end of the file, and there is nothing else that was removed as a result of this error, the original code and the comment can safely be removed from the output code file.\n\nAnd now we’ve resolved our first and most serious issue. Get excited. \n\nLet’s work through the rest of our EWIs in this file. We can search for “EWI” because we now know that text will appear in the comment every time there is an error code. (Alternatively, we could sort the issues.csv file and order the issues by severity… but that’s not really necessary here.)\n\nThe next one is actually just a warning, not an error. It’s telling us that there was a function used that isn’t always equivalent in Spark and Snowpark:\n\n```python\n#EWI: SPRKPY1067 =\u003E Snowpark does not support split functions with more than two parameters or containing regex pattern. See documentation for more info.\nsplit_col = split(df_uppercase['NAME'], '.first:')\n```\n\nThe description here though gives away that we probably don’t have to worry about this. There are only two parameters being passed. Let’s leave this EWI as a comment in the file, so we know to check for it when we are running the file later.\n\nThe last one for this file is a conversion error saying that something is not supported:\n\n![sma_code_editor_03](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_03.png)\n\nThis is the write call to the spark jdbc driver to write the output dataframe into SQL Server. Since this is part of the “resolve all inputs/outputs” step that we are going to deal with after we address our issues, we’ll leave this for later. Note, however, that this error must be resolved. The previous one was just a warning and may still work with no change being made.\n\n**Resolving the Session Calls**\n\nThe session calls are converted by the SMA, but you should pay special attention to them to make sure they are functional. In our pipeline script, this is the before and after code:\n\n![sma_code_editor_04](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_04.png)\n\nThe SparkSession reference was changed to Session. You can see that reference change near the top of this file in the import statement as well:\n\n```python\n# Original PySpark Imports\nfrom pyspark.sql import SparkSession\n\n# New Snowpark Imports\nfrom snowflake.snowpark import Session\n```\n\nNote in the image above, the variable assignment of the session call to “spark” is not changed. This is because this is a variable assignment. It is not necessary to change this, but if you’d like to change the “spark” decorator to “session”, that would be more in line with what Snowpark recommends. (Note that the VS Code Extension “SMA Assistant” will suggest these changes as well.) \n\nThis is a simple exercise, but it’s worth doing. You can do a find and replace using VS Code’s own search ability to find the references to “spark” in this file and replace them with session. You can see the result of this in the image below. The references to the “spark” variable in the converted code have been replaced with “session”:\n\n![sma_code_editor_05](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_05.png)\n\nWe also can remove something else from this session call. Since we are not going to be running “spark” anymore, we do not need to specify the driver path for the spark driver. So we can remove the config function entirely from the session call like this:\n\n```python\n# Old Converted output.\n# Spark Session\nsession = Session.builder.config('spark.driver.extraClassPath', driver_path) \\\n                    .app_name('SparkSQLServerExample', True) \\\n                    .getOrCreate()\n\n# New Converted Output\n# Snowpark Session\nsession = Session.builder.app_name('SparkSQLServerExample', True).getOrCreate()\n```\n\nMight as well convert it to a single line. The SMA couldn’t be sure we didn’t need that driver (although that seems logical), so it did not remove it. But now that we have our session call is complete. \n\n(Note that the SMA also adds a “query tag” to the session. This is to help troubleshoot issues with this session or query later on, but this is completely optional to leave or remove.)\n\n**Notes on the Session Calls**\n\nBelieve it or not that is all that we need to change in the code for the session call, but that’s not all we need to do to create the session. This refers back to the original question that a lot of this depends on where you want to run these files. These original spark session calls used a configuration that was setup elsewhere. If you look at the original Spark session call it’s looking for a config file that is being read into a pandas dataframe location at the start of this script file (this is actually true for our notebook file as well).\n\n![sma_code_editor_06](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_06.png)\n\nSnowpark can function the same way, and this conversion assumes that is how this user will run this code. However, for the existing session call to work, the user would have to load all of the information for their Snowflake account into the local (or at least accessible) connections.toml file on this machine, and that the account they are attempting to connect to is set as the default. [You can learn more about updating the connections.toml file in the Snowflake/Snowpark documentation](https://docs.snowflake.com/en/developer-guide/snowpark/python/creating-session#connect-by-using-the-connections-toml-file), but the idea behind it is that there is an accessible location that has the credentials. When a snowpark session is created, it is going to check this… unless the connection parameters are explicitly passed to the session call. \n\nThe standard way to do this is to input the connection parameters directly as strings and call them with the session:\n\n```python\n# Parameters in a dictionary.\nconnection_parameters = {\n  \"account\": \"\u003Cyour snowflake account\u003E\",\n  \"user\": \"\u003Cyour snowflake user\u003E\",\n  \"password\": \"\u003Cyour snowflake password\u003E\",\n  \"role\": \"\u003Cyour snowflake role\u003E\",  # optional\n  \"warehouse\": \"\u003Cyour snowflake warehouse\u003E\",  # optional\n  \"database\": \"\u003Cyour snowflake database\u003E\",  # optional\n  \"schema\": \"\u003Cyour snowflake schema\u003E\",  # optional\n}\n\n# The session call\nsession = Session.builder.configs(connection_parameters).app_name(\"AdventureWorksSummary\", True).getOrCreate()\n```\n\nAdventureWorks appears to have referenced a file with these credentials and called it. Assuming there is a similar file called 'snowflake_credentials.txt' that is accessible, then the syntax that would match that could look something like:\n\n```python\n# Load into a dataframe.\nsnow_creds = pd.read_csv('snowflake_credentials.txt', index_col=None, header=0)\n\n# Build the parameters.\nconnection_parameters = {\n  \"account\": snow_creds.loc[snow_creds['Specific_Element'] == 'Account', 'Value'].item(),\n  \"user\": snow_creds.loc[snow_creds['Specific_Element'] == 'Username', 'Value'].item(),\n  \"password\": snow_creds.loc[snow_creds['Specific_Element'] == 'Password', 'Value'].item(),\n  \"role\": \"\u003Cyour snowflake role\u003E\",  # optional\n  \"warehouse\": snow_creds.loc[snow_creds['Specific_Element'] == 'Warehouse', 'Value'].item(),  # optional\n  \"database\": snow_creds.loc[snow_creds['Specific_Element'] == 'Database', 'Value'].item(),  # optional\n  \"schema\": snow_creds.loc[snow_creds['Specific_Element'] == 'Schema', 'Value'].item(),  # optional\n}\n\n# Then pass the parameters to the configs function of the session builder.\nsession = Session.builder.configs(connection_parameters).app_name(\"AdventureWorksSummary\", True).getOrCreate()\n```\n\nFor the purpose of the time limit on this lab, the first option may make more sense. [There’s more on this in the Snowpark documentation](https://docs.snowflake.com/en/developer-guide/snowpark/python/creating-session#connect-by-specifying-connection-parameters). \n\nNote that for our notebook file to run inside of Snowflake using Snowsight, you wouldn’t need to do any of this. You would just call the active session and run it. \n\nNow it's time for the most critical component of this migration, resolving any input/output references.\n\n**Resolving the Inputs and Outputs**\n\nSo let’s resolve our inputs and outputs now. Note that this is going to diverge based on whether you’re running the files locally or Snowflake. for the python script, Let’s make sure what we gain/lose by running directly inside of Snowsight: **you cannot run the whole operation in Snowsight** (at least not currently). The local csv file is not accessible from Snowsight. You will have to load the .csv file into a stage manually. This will likely not be an ideal solution, but we can test the conversion by doing this. \n\nSo we’ll first prep this file to be run/orchestrated locally, and then to be run in Snowflake.\n\nTo get the pipeline script’s inputs and output resolved, we need to first identify them. They are pretty simple. This script seems to:\n- access a local file\n- load the result into SQL Server (but now Snowflake)\n- moves the file to make way for the next one\n\nSimple enough. So we need to replace each component of the code that does those things. Let’s start with accessing the local file.\n\nAs was mentioned at the start of this, it would be strongly suggested to rearchitect the Point of Sale System and the orchestration tools used to run this python script, to put the output file into a cloud storage location. Then you could turn that location into an External Table, and voila… you are in Snowflake. However, the current architecture says that this file is not in a cloud storage location and will stay where it is, so we need to create a way for Snowflake to access this file preserving the existing logic. \n\nWe have options to do this, but we will create an internal stage and move the file into the stage with the script. We would then need to move the file in the local file system, and also move it in the stage. This can all be done with Snowpark. Let’s break it down:\n- accessing a local file: Create an internal stage (it one doesn’t exist already) -\u003E Load the file into the stage -\u003E Read the file into a dataframe\n- loading the result into SQL Server: Load the transformed data into a table in Snowflake\n- moves the file to make way for the next one: Move the local file -\u003E Move the file in the stage.\n\nLet’s look at code that can do each of these things. \n\n**Access a Locally Accessible File**\n\nThis source code in Spark looks like this:\n\n```python\n# Spark read from a local csv file.\ndf = spark.read.csv('customer_update.csv', header=True, inferSchema=True)\n```\n\nAnd the transformed snowpark code (by the SMA) looks like this:\n\n```python\n# Snowpark read from a local csv file.\ndf = session.read.option(\"PARSE_HEADER\", True).option(\"INFER_SCHEMA\", True).csv('customer_update.csv')\n```\n\nWe can replace that with this with code that does the steps above:\n\n1. Create an internal stage (if one does not exist already). We will create a stage called 'LOCAL_LOAD_STAGE' and go through a few steps to make sure that the stage is ready for our file.\n\n```python\n# Additional import needed for this. (Check to see if this is already imported.)\nfrom snowflake.core.stage import Stage, StageEncryption, StageResource\n\n# name the stage we're going to use.\ntarget_stage_name = \"LOCAL_LOAD_STAGE\"\n\n# Check to see if this stage already exists.\nstages = session.sql(\"SHOW STAGES\").collect()\ntarget_stages = [stage for stage in stages if stage['name'] == target_stage_name]\n\n# Create the stage if it does not already exist.\nif(len(target_stages) \u003C 1):\n    from snowflake.core import Root\n    from snowflake.core.stage import Stage, StageEncryption, StageResource\n    root = Root(session)\n    my_stage = Stage(name=\"LOCAL_LOAD_STAGE\",encryption=StageEncryption(type=\"SNOWFLAKE_SSE\"))\n    root.databases[\"ADVENTUREWORKS\"].schemas[\"DBO\"].stages.create(my_stage)\n    print('%s created.'%(target_stage_name))\nelse:\n    print('%s already exists.'%(target_stage_name))\n```\n\n2. Load the file into the stage. \n\n```python\n# Move the file.\nput_results = session.file.put(local_file_name=\"customer_update.csv\",\n                    stage_location=\"ADVENTUREWORKS.DBO.LOCAL_LOAD_STAGE\",\n                    overwrite=False,\n                    auto_compress=False)\n\n# Read the results.\nfor r in put_results:\n    str_output = (\"File {src}: {stat}\").format(src=r.source,stat=r.status)\n    print(str_output) \n```\n\n3. Read the file into a dataframe. This is the part that the SMA actually converted. We need to specify that the location of the file is now the internal stage.\n\n```python\n# Location of the file in the stage.\ncsv_file_path = \"@LOCAL_LOAD_STAGE/customer_update.csv\"\n\n# Spark read from a local csv file.\ndf = session.read.option(\"PARSE_HEADER\", True).option(\"INFER_SCHEMA\", True).csv(csv_file_path)\n```\n\nThe result of that would look like this:\n\n![sma_code_editor_07](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_07.png)\n\nLet's move on to the next step.\n\n**Load the result into Snowflake**\n\nThe original script wrote the dataframe into SQL Server. Now we are going to load into Snowflake. This is a much simpler conversion. The dataframe is already a Snowpark dataframe. This is one of the advantages of Snowflake. Now that the data is accessible to Snowflake, everything happens inside Snowflake.\n\n```python\n# Original output from the conversion tool.\n# Write the DataFrame to SQL Server.\n#EWI: SPRKPY1002 =\u003E pyspark.sql.readwriter.DataFrameWriter.jdbc is not supported\ndf_transformed.write.jdbc(url=sql_server_url,\n              table='dbo.DimCustomer',\n              mode=\"append\",\n              properties={\n                  \"user\": sql_server_user,\n                  \"password\": sql_server_password,\n                  \"driver\": driver_path\n              })\n\n# Corrected Snowflake/Snowpark code.\ndf_transformed.write.save_as_table(\"ADVENTUREWORKS.DBO.DIMCUSTOMER\", mode=\"append\")\n```\n\nNote that we may want to write to a temp table to do some testing/validation, but this is the behavior in the original script.\n\n**Move the file to make way for the next one**\n\nThis is the behavior in the orginal script. We don't really need to make this happen in Snowflake, but we can to showcase the exact same functionality in the stage. This is done with an os command in the original file system. That does not depend on Spark and will remain the same. But to emulate this behavior in snowpark, we would need to move this file in the stage to a new directory. \n\nThis can be done simply enough with the following python code:\n\n```python\n# New filename.\noriginal_filepath = '@LOCAL_LOAD_STAGE/customer_update.csv'\nnew_filepath = '@LOCAL_LOAD_STAGE/old_versions/customer_update_%s.csv'%(today_time)\n\ncopy_sql = f\"COPY FILES INTO {new_filepath} FROM {original_filepath}\"\nsession.sql(copy_sql).collect()\nprint(f\"File copied from {original_filepath} to {new_filepath}\")\n\nremove_sql = f\"REMOVE {original_filepath}\"\nsession.sql(remove_sql).collect()\nprint(f\"Original file {original_filepath} removed.\")\n```\n\nNote that this would not replace any of the existing code. Since we already want to keep the existing motion of moving the spark code to snowpark, we will leave the os reference. The final version will look like this:\n\n![sma_code_editor_08](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_08.png)\n\nNow we have the same motion completely done. Now let's do our final cleanup, and test this script out. \n\n**Clean up and Test**\n\nWe never looked at our import calls and we have config files that are not necessary at all. We could leave the references to the config files and run the script. In fact, assuming those config files are still accesible, then the code will still run. But if we're taking a close look at our importt statements, we might as well remove them. These files are represented by all of the code between the import statements and the session call:\n\n![sma_code_editor_09](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_09.png)\n\nThere’s a few other things we should do:\n- Check that all of our imports are still necessary. We can leave them for now. If there is an erorr, we can address it.\n- We also have one EWI that we left in there as a warning to check. So we want to make sure we inspect that output. \n- We need to make sure that our file system behavior mirrors that of the expected file system for the POS system. To do this, we should move the customer_update.csv file into the root folder you chose when first launching VS Code. \n- Create a directory called “old_versions” in that same directory. This should allow the os operations to run.\n\nFinally, if you are not comfortable running the code directly into the production table, you can create a copy of that table for this test, and point the load to that copy. Replace the load statement with the one below. Since this is a lab, feel free to write to the “production” table: \n\n```python\n# In case we want to test.\ncreate_sql = \"\"\"\n                CREATE OR REPLACE TABLE ADVENTUREWORKS.DBO.DIMCUSTOMER_1\n                AS select * from ADVENTUREWORKS.DBO.DIMCUSTOMER;\n                \"\"\"\nsession.sql(create_sql).collect()\n\n# Write the DataFrame to SQL Server.\ndf_transformed.write.save_as_table(\"ADVENTUREWORKS.DBO.DIMCUSTOMER_1\", mode=\"append\")\n```\n\nNow we're finally ready to test this out. We can run this script in Python to a testing table and see if it will fail. So run it!\n\nTragic! The script failed with the following error:\n\n![sma_code_editor_10](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_10.png)\n\nIt looks like the way we are referencing an identifier is not the way that Snowpark wanted it. The code that failed is in the exact spot where the remaining EWI is:\n\n![sma_code_editor_11](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_11.png)\n\nYou could reference the documentation on the link provided by the error, but in the interest of time, Snowpark needs this variable to expressly be a literal. We need to make the following replacement:\n\n```python\n# Old\nsplit_col = split(df_uppercase['NAME'], '.first:')\n\n# New\nsplit_col = split(df_uppercase['NAME'], lit('.first:'))\n```\n\nThis should take care of this error. Note that there are always going to be some functional differences between source and a target platforms. Conversion tools like the SMA like to make these differences as obvious as possible. But note that no conversion is 100% automated. \n\nLet’s run it again. This time… success! \n\n![sma_code_editor_12](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_code_editor_12.png)\n\nWe can write some queries in python to validate this, but why don’t we just go into Snowflake (because that’s what we’re about to do anyways). \n\nNavigate to your snowflake account that you have been using to run these scripts. This should be the same one you used to load the database from SQL Server (and if you haven’t done that, the above scripts won’t work anyways beecause the data has not yet been migrated). \n\nYou can quickly check this by seeing if the stage was created with the file: \n\n![sma_stage_check_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_stage_check_01.png)\n\nEnable the directory table view to see if the old_versions folder is in there:\n\n![sma_stage_check_02](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_stage_check_02.png)\n\nAnd it is:\n\n![sma_stage_check_03](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_stage_check_03.png)\n\nSince that was the last element of our script, it looks like we’re good!\n\nWe can also simply validate that the data was loaded by simply querying the table for the data we uploaded. You can open a new worksheet and simply write this query:\n\n```python\nselect * from ADVENTUREWORKS.DBO.DIMCUSTOMER\nwhere FIRSTNAME like '%Brandon%'\nAND LASTNAME like '%Carver%'\n```\n\n![sma_stage_check_04](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_stage_check_04.png)\n\n**Running the Pipeline Script in Snowsight**\n\nLet’s take a quick look back at the flow we are attempting to convert was doing in Spark:\n- accessing a local file\n- loading the result into SQL Server\n- moving the file to make way for the next one \n\nThis flow is not possible to run entirely from within Snowsight. Snowsight does not have access to a local file system. The recommendation here would be to move the export from the POS to a data lake… or any number of other options that would be accessible via Snowsight. \n\nWe can, however, take a closer look at how Snowpark handles the transformation logic by running the Python script in Snowflake. If you have already made the changes recommended above, you can run the body of the script in a Python Worksheet in Snowflake. \n\nTo do this, first login to your Snowflake account and navigate to the worksheets section. In this worksheet, create a new Python worksheet:\n\n![sma_python_worksheet_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_01.png)\n\nSpecify the database, schema, role, and warehouse you’d like to use:\n\n![sma_python_worksheet_02](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_02.png)\n\nNow we do not have to deal with our session call. You will see a template generated in the worksheet window:\n\n![sma_python_worksheet_03](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_03.png)\n\nLet’s start by bringing over our import calls. After making the previous script ready to use, we should have the following set of imports:\n\n```python\n# General Imports\nimport pandas as pd\nimport os\nimport shutil\nimport datetime\n\n# Snowpark Imports\nfrom snowflake.snowpark import Session\nfrom snowflake.snowpark.functions import col\nfrom snowflake.snowpark.functions import upper\nfrom snowflake.snowpark.functions import lower\nfrom snowflake.snowpark.functions import split\nfrom snowflake.snowpark.functions import trim\nfrom snowflake.snowpark.functions import when\nfrom snowflake.snowpark.functions import lit\nfrom snowflake.snowpark.functions import expr\nfrom snowflake.snowpark.functions import regexp_replace\n```\n\nWe only need the snowpark imports. We will not be moving files around a file system. We could keep the datetime reference if we want to move the file in the stage. (Let’s do it.)\n\nPaste the Snowpark imports (plus datetime) in the python worksheet below the other imports that are already present. Note that ‘col’ is already imported, so you can remove one of those:\n\n![sma_python_worksheet_04](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_04.png)\n\nUnder the “def main” call, let’s paste in all of our transformation code. This will include everything from the assignment of the csv location to the writing of the dataframe to a table. \n\nFrom here:\n\n![sma_python_worksheet_05](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_05.png)\n\nTo here:\n\n![sma_python_worksheet_06](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_06.png)\n\nWe can also add back in the code that moves the files around in the stage to preserve the same functionality for the POC. This part:\n\n![sma_python_worksheet_07](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_07.png)\n\nBefore you can run the code though, you will have to manually create the stage and move the file into the stage. We can add the create stage statement into the script, but we would still need to manually load the file into the stage. \n\nSo if you open another worksheet (this time… a sql worksheet), you can run a basic SQL statement that will create the stage:\n\n```sql\nCREATE STAGE LOCAL_LOAD_STAGE\n  ENCRYPTION = (TYPE = 'SNOWFLAKE_SSE');\n```\n\nMake sure to select the correct database, schema, role, and warehouse in your worksheet:\n\n![sma_python_worksheet_08](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_08.png)\n\nYou can also [create an internal stage directly in the Snowsight UI](https://docs.snowflake.com/en/user-guide/data-load-local-file-system-create-stage#create-a-named-stage-using-snowsight). Now that the stage exists, we can manually load the file of interest into the stage. Navigate to the Databases section of the Snowsight UI, and find the stage we just created in the appropriate database.schema:\n\n![sma_python_worksheet_09](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_09.png)\n\nLet’s add our csv file by selecting the +Files option in the top right corner of the window. This will launch the Upload Your Files menu:\n\n![sma_python_worksheet_10](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_10.png)\n\nDrag and drop or browse to our project directory and load the customer_update.csv file into the stage:\n\n![sma_python_worksheet_11](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_11.png)\n\nSelect Upload in the bottom right corner of the screen. You will be taken back to the stage screen. To view the files, you will need to select Enable Directory Table:\n\n![sma_python_worksheet_12](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_12.png)\n\nAnd now… our file appears in the stage:\n\n![sma_python_worksheet_13](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_13.png)\n\nThis is not really a pipeline anymore, of course. But at least we can run the login in Snowflake. Run the rest of the code that you moved into the worksheet. This user had success the first time, but that’s no guarantee of success the second time:\n\n![sma_python_worksheet_14](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_python_worksheet_14.png)\n\nNote that once you’ve defined this function in Snowflake, you can call it in other ways. If AdventureWorks is 100% replacing their POS, then it may make sense to have the transformation logic in Snowflake, especially if orchestration and file movement will be handled somewhere else entirely. This allows Snowpark to focus on where it excels with the transformation logic.\n\n**Conclusions**\n\nAnd that's it for the script file. It's not the best example of a pipeline, bu it does hit hard on how to deal wit ht he output from the SMA:\n- **Resolve All Issues**\n- **Resolve the session calls**\n- **Resolve the Input/Outputs**\n- **Clean up and Test**!\n\nLet's move on to the reporting notebook.\n\n## Notebook Conversion and Validation\n\nLet’s step over to the Reporting Notebook in our codebase: **Basic Reporting Notebook - SqlServer Spark.ipynb**. We're going to walk through a similar set of steps as we did with the pipeline script.\n- **Resolve All Issues**: “Issues” here means the issues generated by the SMA. Take a look at the output code. Resolve parsing errors and conversion errors, and investigate warnings.\n- **Resolve the session calls**: How the session call is written in the output code depends on where we are going to run the file. We will resolve this for running the code file(s) in the same location as they were originally going to be run, and then for running them in Snowflake.\n- **Resolve the Input/Outputs**: Connections to different sources cannot be resolved entirely by the SMA. There are differences in the platforms, and the SMA will usually disregard this. This also is affected by where the file is going to be run. \n- **Clean up and Test**! Let’s run the code. See if it works. We will be smoke testing in this lab, but there are tools to do more extensive testing and data validation including Snowpark Python Checkpoints.\n\nLet’s get started.\n\n**Resolve All Issues**\n\nLet’s go ahead and look at the issues present in the notebook.\n\n(Note that you can open the notebook in VS Code, but to view it appropriately, you may want to install the Jupyter extension for VS Code. Alternatively, you could open this in Jupyter, but Snowflake still recommends VS Code with the Snowflake extension installed).\n\nYou can use the compare feature to view both of these side by side as we did with the pipeline file, though it will look more like a json if you do so:\n\n![sma_notebook_01](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_01.png)\n\nNote that there are only two unique EWI’s in this notebook. You can return to the search bar to find them, but since this is so short, you could also just… scroll down. These are the unique issues:\n- **SPRKPY1002** *=\u003E pyspark.sql.readwriter.DataFrameReader.jdbc is not supported.* This is a similar issue to the one we saw in the pipeline file, but that was a write call. This is a read call to the SQL Server database. We will resolve this in a bit.\n- **SPRKPY1068** *=\u003E \"pyspark.sql.dataframe.DataFrame.toPandas is not supported if there are columns of type ArrayType, but it has a workaround. See documentation for more info.* This is another warning. If we pass an array to this function in Snowpark, it may not work. Let’s keep an eye on this when we test it.\n\nAnd that’s it for the notebook… and our issues. We resolved a parsing error in the script file, but we did not have much to do in this notebook. Most of this was converted successfully, though there are a few things we will keep our eyes on. Let’s move on to the next step: resolving any session calls.\n\n**Resolve the Session Calls**\n\nTo update the session calls in the reporting notebook, we need to locate the cell with the session call in it. That looks like this:\n\n![sma_notebook_02](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_02.png)\n\nNow let’s do what we already did for our pipeline file:\n- Change all references to the “spark” session variable to “session” (note that this is throughout the notebook)\n- Remove the config function with the spark driver.\n\nThe before and after on this will look like this:\n\n```python\n# Old Session\nspark = Session.builder.config('spark.driver.extraClassPath', driver_path).app_name(\"AdventureWorksSummary\", True).getOrCreate()\nspark.update_query_tag({\"origin\":\"sf_sit\",\"name\":\"sma\",\"version\":{\"major\":7,\"minor\":4,\"patch\":10},\"attributes\":{\"language\":\"Python\"}})\n\n# New Session\n# Session\nsession = Session.builder.app_name(\"AdventureWorksSummary\", True).getOrCreate()\nsession.update_query_tag({\"origin\":\"sf_sit\",\"name\":\"sma\",\"version\":{\"major\":7,\"minor\":4,\"patch\":10},\"attributes\":{\"language\":\"Python\"}})\n```\n\nNote that there is other code in this cell. This code:\n\n```python\nurl = sql_server_url\nproperties = {'user' : sql_server_user, 'password' : sql_server_password}\n# Spark dataframe.\n#EWI: SPRKPY1002 =\u003E pyspark.sql.readwriter.DataFrameReader.jdbc is not supported\ndf = session.read.jdbc(url = url, table = 'dbo.DimCustomer', properties = properties)\nprint('Session successfully setup.')\n```\n\nWe’re almost ready to take on the read statement, but we’re not there yet. Let’s just move all of this to another cell. Create a new cell below this one, and move this code to that cell. It will look like this:\n\n![sma_notebook_03](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_03.png)\n\nIs this all we need for the session call? No. Recall (and possibly review) the previous page under **Notes on Session Calls**. You will either need to make sure that your connection.toml file has your connection information or you will need to explicitly specify the connection parameters you intend to use in the session. \n\n**Resolving the Inputs/Outputs**\n\nSo let’s resolve our inputs and outputs now. Note that this is going to diverge based on whether you’re running the files locally or Snowflake, but for the notebook, everything can be run locally or in Snowflake. The code will be a bit simpler as we won’t even need to call a session. We’ll just… get the active session. As with the pipeline file, we’ll do this in two parts: to be run/orchestrated locally, and to be run in Snowflake. \n\nWorking through the inputs and outputs in the reporting notebook will be considerably simpler than it was for the pipeline. There is no reading from a local file or moving data between files. There is simply a read from a table in SQL Server that is now a read from a table in Snowflake. Since we will not be accessing SQL Server, we can ditch any reference to the SQL Server properties. And the read statement can be replaced by a table statement in Snowflake. The before and after for this cell should look like this: \n\n```python\n# Before\nurl = sql_server_url\nproperties = {'user' : sql_server_user, 'password' : sql_server_password}\n# Spark dataframe.\n#EWI: SPRKPY1002 =\u003E pyspark.sql.readwriter.DataFrameReader.jdbc is not supported\ndf = session.read.jdbc(url = url, table = 'dbo.DimCustomer', properties = properties)\nprint('Session successfully setup.')\n```\n\n```python\n# After\n# New table call\n# Snowpark Dataframe table.\ndf = session.table('ADVENTUREWORKS.DBO.DIMCUSTOMER')\nprint('Table loaded successfully.')\ndf.show()\n```\n\nThis leaves us with an updated session and the correct table read into a dataframe:\n\n![sma_notebook_04](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_04.png)\n\nThat’s actually… it. Let's move on to the Clean up and test part of the notebook file.\n\n**Clean Up and Test**\n\nLet’s do some clean up (like we did previously for the pipeline file). We never looked at our import calls and we have config files that are not necessary at all. Let’s start by removing the references to the config files. This will be each of the cells between the import statements and the session call. \n\n![sma_notebook_05](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_05.png)\n\nNow let’s look at our imports. The reference to the os can be deleted. (Seems like that wasn’t used in the original file either…) There is a pandas reference. There does not appear to be any usages of pandas in this notebook anymore now that the config files are referenced. There is a toPandas reference as part of the Snowpark dataframe API in the reporting section, but that’s not part of the pandas library. \n\nYou can optionally replace all of the import calls to pandas with the modin pandas library. This library will optimize pandas dataframes to take advantage of Snowflake’s powerhouse computing. This change would look like this:\n\n```python\n# Old\nimport pandas as pd\n\n# New\nimport modin.pandas as pd\nimport snowflake.snowpark.modin.plugin\n```\n\nHaving said that, we can delete that one as well. Note that the SMA has replaced any spark specific import statements with those related to Snowpark. The final import cell would look like this:\n\n![sma_notebook_06](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_06.png)\n\nAnd that’s it for our cleanup. We still have a couple of EWIs in the reporting and visualization cells, but it looks like we should make it. Let’s run this one and see if we get an output.\n\n![sma_notebook_07](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_07.png)\n\nAnd we did. The reports seem to match what was output by the Spark Notebook. Even though the reporting cells seemed complex, Snowpark is able to work with them. The SMA let us know there could be an issue, but there doesn’t appear to be any problems. More testing would help, but our first round of smoke testing has passed.\n\nNow let’s look at this notebook in Snowsight. Unlike the pipeline file, we can do this entirely in Snowsight.\n\n**Running the Notebook in Snowsight**\n\nLet’s take the version of the notebook that we have right now (having worked through the issues, the session calls, and the inputs and outputs) and load it into Snowflake. To do this, go to the notebooks section in SnowSight:\n\n![sma_notebook_08](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_08.png)\n\nAnd select the down arrow next to the +Notebook button in the top right, and select “Import .ipynb file” (shown above). \n\nOnce this has been imported, choose the notebook file that we have been working with in the output directory created by the SMA in your project folder.\n\nThere will be a create notebook dialog window that opens. For this upload, we will choose the following options:\n- Notebook location: \n- Database: **ADVENTUREWORKS**\n- Schema: **DBO**\n- Python environment: **Run on warehouse** (This is not a large notebook with a bunch of ml. This is a basic reporting notebook. We can run this on a warehouse.) \n- Query warehouse: **DEFAULT_WH**\n- Notebook warehouse: **DEFAULT_WH** (you can leave it as the system chosen warehouse (will be a streamlit warehouse)... for this notebook, it will not matter)\n\nYou can see these selections below:\n\n![sma_notebook_09](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_09.png)\n\nThis should load your notebook into Snowflake and it will look something like this:\n\n![sma_notebook_10](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_10.png)\n\nThere are a couple of quick checks/changes we need to make from the version we just tested locally in order to ensure that the notebook runs in Snowsight:\n- Change the session calls to retrieve the active session\n- Ensure any dependent libraries we need to install are available\n\nLet’s start with the first one. It may seem odd to alter the session call again after we spent so much time on it in the first place, but we’re running inside of Snowflake now. You can remove anything associated with reading the session call and replacing it with the “get_active_session” call that is standard at the top of most Snowflake notebooks:\n\n```python\n# Old for Jupyter\nsession = Session.builder.app_name(\"AdventureWorksSummary\", True).getOrCreate()\n\n# New for Snowsight\nfrom snowflake.snowpark.context import get_active_session\nsession = get_active_session()\n```\n\nWe don't need to specify connection parameters or update a .toml file because we are already connected. we are in Snowflake.\n\nLet’s replace the old code in the cell with the new code. That will look something like this:\n\n![sma_notebook_11](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_11.png)\n\nNow let’s address the available packages for this run, but instead of us figuring out what we need to add. Let’s let Snowflake. One of the better parts of using a notebook is that we can run individual cells and see what the results are. Let’s run our import library cell.\n\nIf you haven’t already, go ahead and start the session by clicking in the top right corner of the screen where it says “Start”:\n\n![sma_notebook_12](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_12.png)\n\nIf you run the topmost cell in the notebook, and you will likely discover that matplotlib is not loaded into the session:\n\n![sma_notebook_13](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_13.png)\n\nThis is a pretty important one for this notebook. You can add that library to your notebook/session by using the “Packages” option in the top right of the notebook:\n\n![sma_notebook_14](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_14.png)\n\nSearch for matplotlib, and select it. This will make this package available in the session. \n\n![sma_notebook_15](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_15.png)\n\nOnce you load this library, you will have to restart the session. Once you have restarted the session, run that first cell again. You will likely be told that it was a success this time.\n\n![sma_notebook_16](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_16.png)\n\nWith the packages loaded, the session fixed, and the rest of the issues in the code already resolved, what can we do to check the rest of the notebook? Run it! You can run all the cells in the notebook by selecting “Run all” in the top right corner of the screen, and see if we get any errors.\n\nIt looks like there was a successful run:\n\n![sma_notebook_17](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_17.png)\n\nIf you compare the two notebooks execution, it looks like the only difference is that the Snowflake version put all of the output datasets first followed by the images, whereas they are intermixed in the Spark Jupyter Notebook:\n\n![sma_notebook_18](https://www.snowflake.com/content/dam/snowflake-site/developers/guides/end2endmigration/sma_notebook_18.png)\n\nNote that this difference is not an API difference, but rather a difference in how notebooks in Snowflake orchestrate this. This is likely a difference AdventureWorks is willing to accept! \n\n**Conclusions**\n\nBy utilizing the SMA, we were able to accelerate the migration of both a data pipeline and a reporting notebook. The more of each that you have, the more value a tool like the SMA can provide. \n\nAnd let’s go back to the assessment -\u003E conversion -\u003E validation flow that we have consistently come back to. In this migration, we:\n- Setup out project in the SMA\n- Ran SMA’s assessment and conversion engine on the code files\n- Reviewed the output reporting from the SMA to better understand what we have\n- Review what could not be converted by the SMA in VS Code\n- Resolve issues and errors\n- Resolve session references\n- Resolve input/output references\n- Run the code locally\n- And run the code in Snowflake\n- Ran the newly migrated scripts and validated their success\n\nSnowflake has spent a great deal of time improving its ingestion and data engineering capabilities, just as it has spent time improving migration tools like SnowConvert, the SnowConvert Migration Assistant, and the Snowpark Migration Accelerator. Each of these will continue to improve. Please feel free to reach out if you have any suggestions for migration tooling. These teams are always looking for additional feedback to improve the tools. \n\n\n## Conclusion And Resources\n\nCongratulations! You've successfully completed an end-to-end migration from SQL Server to Snowflake, including both the database objects and the ETL pipelines that feed your data mart.\n\n### What You Learned\n- How to assess a SQL Server environment for migration to Snowflake\n- Using SnowConvert to migrate database schemas and resolve conversion issues\n- Transferring data from SQL Server to Snowflake\n- Converting Spark ETL pipelines to Snowpark using the SMA\n\n### Related Resources\n- [SnowConvert Documentation](https://docs.snowconvert.com/)\n- [Snowpark Migration Accelerator Guide](https://docs.snowflake.com/en/user-guide/migration-accelerators)\n- [Snowflake SQL Server Migration Guide](https://resources.snowflake.com/migration/microsoft-sql-server-to-snowflake-migration-guide)\n  - [GitHub Repository for this Lab](https://github.com/Snowflake-Labs/sfguides/tree/master/site/sfguides/src/e2e-sqlserver-migration/)\n",":type":"text/x-markdown","multiValue":false},"quickstartArticleLogoImage":{"dataType":"string","title":"Quickstart Article Logo Image",":type":"text/plain","multiValue":false}},"model":"snowflake-site/models/quickstart-article"},"flexible_column_cont":{"id":"flexible-column-container-d8a8088728","type":"2-column-75-25","alignColumns":"top","containerMaxWidth":"extra-large","topPadding":"none","bottomPadding":"none","spaceBetween":"none","reverseOnMobile":false,"carouselOnMobile":false,"backgroundImageOption":"none","flexible_column_content_container_1":{"layout":"SIMPLE","id":"container-5804158b75",":type":"snowflake-site/components/flexible-column-container/flexible-column-content-container",":items":{"quickstart_last_modi":{"id":"quickstart-last-modified-89aaaed566","icon":{"id":"icon","icon":"calendar",":type":"snowflake-site/components/icon","appliedCssClassNames":"snowflake-icon-blue"},"lastModifiedDatePrefix":"Updated","lastModifiedDate":"2025-12-20",":type":"snowflake-site/components/quickstart/quickstart-last-modified","appliedCssClassNames":"snowflake-responsive-component-top-padding-small"},"text":{"id":"text-ed60df1b20","additionalClasses":"qs-disclaimer-text","text":"\u003Cp\u003E\u003Cspan style=\"color: #666;\"\u003EThis content is provided as is, and is not maintained on an ongoing basis. It may be out of date with current Snowflake instances\u003C/span\u003E\u003C/p\u003E\r\n","richText":true,":type":"snowflake-site/components/text","appliedCssClassNames":"snowflake-responsive-component-top-padding-small"}},":itemsOrder":["quickstart_last_modi","text"]},"flexible_column_content_container_2":{"layout":"SIMPLE","id":"container-892a8bfcda",":type":"snowflake-site/components/flexible-column-container/flexible-column-content-container",":items":{},":itemsOrder":[]},":type":"snowflake-site/components/flexible-column-container","isActiveTOC":false,"isBlogPage":false}},":itemsOrder":["contentfragment","flexible_column_cont"]},"flexible_column_content_container_2":{"layout":"SIMPLE","id":"container-5e8b55803c",":type":"snowflake-site/components/flexible-column-container/flexible-column-content-container",":items":{"quickstart_table_of_":{"layout":"SIMPLE","id":"container-80f5bba41c","isDeveloperGuidesPage":false,":type":"snowflake-site/components/quickstart/quickstart-table-of-content/quickstart-table-of-content-container",":items":{"quickstart_table_of_":{"id":"quickstart-table-of-content-ea329f27db","fragmentPath":"/content/dam/snowflake-site/en/content-fragments/quickstarts/end2endmigration",":type":"snowflake-site/components/quickstart/quickstart-table-of-content","headings":["\u003Ch2\u003EOverview\u003C/h2\u003E","\u003Ch2\u003EAssessment\u003C/h2\u003E","\u003Ch2\u003EDatabase Code Conversion\u003C/h2\u003E","\u003Ch2\u003EDeployment and solve issues\u003C/h2\u003E","\u003Ch2\u003EData Migration\u003C/h2\u003E","\u003Ch2\u003EPipeline Assessment\u003C/h2\u003E","\u003Ch2\u003EPipeline Conversion\u003C/h2\u003E","\u003Ch2\u003ENotebook Conversion and Validation\u003C/h2\u003E","\u003Ch2\u003EConclusion And Resources\u003C/h2\u003E"]},"quickstart_button":{"id":"quickstart-button-6ddb301db2","fragmentPath":"/content/dam/snowflake-site/en/content-fragments/quickstarts/end2endmigration",":type":"snowflake-site/components/quickstart/quickstart-button","appliedCssClassNames":"snowflake-responsive-component-top-padding-none"}},":itemsOrder":["quickstart_table_of_","quickstart_button"]}},":itemsOrder":["quickstart_table_of_"]},":type":"snowflake-site/components/flexible-column-container","isActiveTOC":false,"isBlogPage":false},"markup_editor":{"id":"markup-editor-003688975f","title":"Page CSS","cssContent":"#quickstart-template-main-flexible-container{padding:24px}#quickstart-template-main-flexible-container \u003E .snowflake-flexible-column-container-items{grid-template-columns:1fr 0}.qs-disclaimer-text p \u003E span{font-size:15px !important}@media (min-width:768px){#quickstart-template-main-flexible-container{padding:24px 32px}#quickstart-template-main-flexible-container \u003E .snowflake-flexible-column-container-items{grid-template-columns:7fr 3fr;gap:48px}}@media (max-width:767px){#quickstart-template-main-flexible-container \u003E .snowflake-flexible-column-container-items{gap:0}}@media (min-width:1024px){#quickstart-template-main-flexible-container{padding:0 92px 48px 92px}#quickstart-template-main-flexible-container \u003E .snowflake-flexible-column-container-items{gap:117px}}",":type":"snowflake-site/components/markup-editor","isGSAPEnabled":false}},":itemsOrder":["quickstart_hero","flexible_column_cont","markup_editor"],":type":"wcm/foundation/components/responsivegrid"},"modal_container":{"layout":"SIMPLE","id":"container-ed82b9a33b",":type":"snowflake-site/components/modal/modal-container",":items":{},":itemsOrder":[]},"experiencefragment-footer":{"id":"experiencefragment-2b948268b7","localizedFragmentVariationPath":"/content/experience-fragments/snowflake-site/language-masters/en/site/footer/master/jcr:content","configured":true,":type":"snowflake-site/components/experiencefragment",":items":{"root":{"additionalClasses":"sf-footer","layout":"SIMPLE","id":"container-d63e7bb229",":type":"snowflake-site/components/container",":items":{"container_copy":{"additionalClasses":"sf-footer__inner","columnClassNames":{"flexible_column_cont":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-af3f6d8ddf",":type":"snowflake-site/components/container",":items":{"flexible_column_cont":{"id":"flexible-column-container-071644a69d","type":"1-column","alignColumns":"top","containerMaxWidth":"extra-large","topPadding":"medium","bottomPadding":"extra-small","spaceBetween":"small","reverseOnMobile":false,"carouselOnMobile":false,"propertiesCSSClasses":"sf-footer-grid","backgroundImageOption":"none","flexible_column_content_container_1":{"layout":"SIMPLE","id":"container-3db2b06a57",":type":"snowflake-site/components/flexible-column-container/flexible-column-content-container",":items":{"container":{"additionalClasses":"sf-footer-grid__inner","columnClassNames":{"container":"aem-GridColumn aem-GridColumn--default--12","container_1622723482":"aem-GridColumn aem-GridColumn--default--12","container_copy_copy_":"aem-GridColumn aem-GridColumn--default--12","container_copy_copy":"aem-GridColumn aem-GridColumn--default--12","container_copy":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-7db31f1820",":type":"snowflake-site/components/container",":items":{"container_1622723482":{"additionalClasses":"sf-footer__column","columnClassNames":{"container":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-0559eb868e",":type":"snowflake-site/components/container",":items":{"container":{"additionalClasses":"sf-footer__newsletter-group","columnClassNames":{"text":"aem-GridColumn aem-GridColumn--default--12","marketo_v2":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-24a512478b",":type":"snowflake-site/components/container",":items":{"text":{"id":"text-502f2d8e1a","additionalClasses":"sf-footer__newsletter-title","text":"\u003Cp\u003E\u003Cb\u003ESubscribe to our monthly newsletter\u003C/b\u003E\u003C/p\u003E\r\n\u003Cp\u003EStay up to date on Snowflake’s latest products, expert insights and resources—right in your inbox!\u003C/p\u003E\r\n","richText":true,":type":"snowflake-site/components/text","appliedCssClassNames":"text-size-regular text-color-text-04"},"marketo_v2":{"id":"marketo-v2-5d4e89364a","marketoForm":{"formId":"45871","successUrl":null,"edit":false,"hidden":null,"script":null,"values":null},"marketoConfigured":true,"formConfigured":true,"munchkinId":"252-RFO-227","serverInstance":"252-RFO-227.mktoweb.com",":type":"snowflake-site/components/form/marketo-v2"}},":itemsOrder":["text","marketo_v2"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-small"}},":itemsOrder":["container"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-small"},"container":{"columnClassNames":{"text_copy":"aem-GridColumn aem-GridColumn--default--12","text":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-e41d080d06",":type":"snowflake-site/components/container",":items":{"text":{"id":"text-a9f6bb9585","additionalClasses":"sf-footer__link-group","text":"\u003Cp class=\"sf-footer__column-title\"\u003EProduct\u003C/p\u003E\r\n\u003Cul\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/product/platform/\"\u003EPlatform\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/product/snowflake-cowork/\"\u003ESnowflake CoWork\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/product/data-engineering/\"\u003EData Engineering\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/product/analytics/\"\u003EAnalytics\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/product/ai/\"\u003EAI\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/product/applications-and-collaboration/\"\u003EApplications &amp; Collaboration\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/pricing-options/\"\u003EPricing\u003C/a\u003E\u003C/li\u003E\r\n\u003C/ul\u003E\r\n","richText":true,":type":"snowflake-site/components/text","appliedCssClassNames":"text-size-small text-color-text-04"},"text_copy":{"id":"text-6d9a5a0b7d","additionalClasses":"sf-footer__link-group","text":"\u003Cp class=\"sf-footer__column-title\"\u003ESupport\u003C/p\u003E\r\n\u003Cul\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/support/\"\u003ESupport\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/legal/addenda/priority-support-services-description/\"\u003EPriority Support\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://status.snowflake.com/\" target=\"_blank\" rel=\"noopener noreferrer\"\u003EStatus\u003C/a\u003E\u003C/li\u003E\r\n\u003C/ul\u003E\r\n","richText":true,":type":"snowflake-site/components/text","appliedCssClassNames":"text-size-small text-color-text-04"}},":itemsOrder":["text","text_copy"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-medium"},"container_copy_copy":{"columnClassNames":{"text":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-ada67b6542",":type":"snowflake-site/components/container",":items":{"text":{"id":"text-5b64b8f830","additionalClasses":"sf-footer__link-group","text":"\u003Cp class=\"sf-footer__column-title\"\u003E\u003Ca href=\"/en/solutions/industries/\"\u003EIndustries\u003C/a\u003E\u003C/p\u003E\r\n\u003Cul\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/solutions/industries/advertising-media-entertainment/\"\u003EAdvertising, Media &amp; Entertainment\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/solutions/industries/financial-services/\"\u003EFinancial Services\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/solutions/industries/healthcare-and-life-sciences/\"\u003EHealthcare &amp; Life Sciences\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/solutions/industries/manufacturing/\"\u003EManufacturing\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/solutions/industries/public-sector/\"\u003EPublic Sector\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/solutions/industries/retail-consumer-goods/\"\u003ERetail &amp; Consumer Goods\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/solutions/industries/telecom/\"\u003ETelecom\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/solutions/industries/technology/\"\u003ETechnology\u003C/a\u003E\u003C/li\u003E\r\n\u003C/ul\u003E\r\n","richText":true,":type":"snowflake-site/components/text","appliedCssClassNames":"text-size-small text-color-text-04"}},":itemsOrder":["text"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-small"},"container_copy":{"columnClassNames":{"text":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-d38836069b",":type":"snowflake-site/components/container",":items":{"text":{"id":"text-dd5946409d","additionalClasses":"sf-footer__link-group","text":"\u003Cp class=\"sf-footer__column-title\"\u003ECompany\u003C/p\u003E\r\n\u003Cul\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/company/overview/about-snowflake/\"\u003EAbout Snowflake\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/company/overview/leadership-and-board/\"\u003ELeadership &amp; Board\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://careers.snowflake.com/us/en\" target=\"_blank\" rel=\"noopener noreferrer\"\u003ECareers\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://investors.snowflake.com/overview/default.aspx\" target=\"_blank\" rel=\"noopener noreferrer\"\u003EInvestor Relations\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://trust.snowflake.com/\" target=\"_blank\" rel=\"noopener noreferrer\"\u003ETrust Center\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/brand-guidelines/\" target=\"_blank\" rel=\"noopener noreferrer\"\u003EBrand Guidelines\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/contact/\"\u003EContact\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/news/\"\u003ENewsroom\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/company/overview/esg/\"\u003EEnvironmental, Social &amp; Governance\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/company/overview/snowflake-ventures/\"\u003ESnowflake Ventures\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/company/overview/end-data-disparity/\"\u003EEnd Data Disparity\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/summit/\"\u003ESnowflake Summit 26\u003C/a\u003E\u003C/li\u003E\r\n\u003C/ul\u003E\r\n","richText":true,":type":"snowflake-site/components/text","appliedCssClassNames":"text-size-small text-color-text-04"}},":itemsOrder":["text"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-small"},"container_copy_copy_":{"columnClassNames":{"text":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-031913836b",":type":"snowflake-site/components/container",":items":{"text":{"id":"text-6da182a07a","additionalClasses":"sf-footer__link-group","text":"\u003Cp class=\"sf-footer__column-title\"\u003ELearn\u003C/p\u003E\r\n\u003Cul\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://snowflake.com/en/resources/\"\u003EResource Library\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/webinars/demo/\"\u003ELive Demos\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/fundamentals/\"\u003EFundamentals\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/resources/learn/training/\"\u003ETraining\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/resources/learn/certifications/\"\u003ECertifications\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca rel=\"noopener noreferrer\" target=\"_blank\" href=\"https://learn.snowflake.com/en/\"\u003ESnowflake University\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/developers/guides\"\u003EDeveloper Guides\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca rel=\"noopener noreferrer\" target=\"_blank\" href=\"https://docs.snowflake.com/\"\u003EDocumentation\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"/en/data-governance/\"\u003EData Governance\u003C/a\u003E\u003C/li\u003E\r\n\u003C/ul\u003E\r\n","richText":true,":type":"snowflake-site/components/text","appliedCssClassNames":"text-size-small text-color-text-04"}},":itemsOrder":["text"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-small"}},":itemsOrder":["container_1622723482","container","container_copy_copy","container_copy","container_copy_copy_"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-small"}},":itemsOrder":["container"]},":type":"snowflake-site/components/flexible-column-container","isActiveTOC":false,"isBlogPage":false}},":itemsOrder":["flexible_column_cont"],"appliedCssClassNames":"snowflake-container snowflake-responsive-container-inner-padding-small"},"container_573483281_":{"additionalClasses":"sf-footer__bottom","columnClassNames":{"container_112062425":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-3971afa1ea",":type":"snowflake-site/components/container",":items":{"container_112062425":{"columnClassNames":{"flexible_column_cont":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-5aded295c2",":type":"snowflake-site/components/container",":items":{"flexible_column_cont":{"id":"flexible-column-container-e168398f4b","type":"1-column","alignColumns":"top","containerMaxWidth":"extra-large","topPadding":"none","bottomPadding":"none","spaceBetween":"small","reverseOnMobile":false,"carouselOnMobile":false,"backgroundImageOption":"none","flexible_column_content_container_1":{"layout":"SIMPLE","id":"container-26faecb02e",":type":"snowflake-site/components/flexible-column-container/flexible-column-content-container",":items":{"container":{"additionalClasses":"sf-footer__legal-container","columnClassNames":{"container":"aem-GridColumn aem-GridColumn--default--12","text_copy_copy_16360":"aem-GridColumn aem-GridColumn--default--12","markup_editor":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-014ad5c2a5",":type":"snowflake-site/components/container",":items":{"container":{"columnClassNames":{"image":"aem-GridColumn aem-GridColumn--default--12"},"gridClassNames":"aem-Grid aem-Grid--12 aem-Grid--default--12","layout":"RESPONSIVE_GRID","columnCount":12,"id":"container-550c9a0c4d",":type":"snowflake-site/components/container",":items":{"image":{"id":"image-23df6c91f6","additionalClasses":"sf-footer__logo","src":"https://www.snowflake.com/content/experience-fragments/snowflake-site/language-masters/en/site/footer/master/_jcr_content/root/container_573483281_/container_112062425/flexible_column_cont/flexible_column_content_container_1/container/container/image.coreimg.svg/1747882370694/nav-icon-snowflake-bug.svg","lazyEnabled":true,"alt":"Snowflake logo","imageLink":{"valid":true,"url":"/en/"},"height":"64","width":"64",":type":"snowflake-site/components/image"}},":itemsOrder":["image"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-extra-small"},"text_copy_copy_16360":{"id":"text-2996c34fea","additionalClasses":"sf-footer__legal-links","text":"\u003Cul\u003E\r\n\u003Cli\u003E© 2026 Snowflake Inc. All Rights Reserved\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/legal/privacy/privacy-policy/\"\u003EPrivacy Policy\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://snowflake.com/en/legal/snowflake-site-terms/\"\u003ESite Terms\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://info.snowflake.com/Preference-center.html\"\u003ECommunication Preferences\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Cbutton id=\"ot-sdk-btn\" class=\"ot-sdk-show-settings\"\u003ECookie Settings\u003C/button\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/legal/privacy/privacy-policy/#12\"\u003EDo Not Share My Personal Information\u003C/a\u003E\u003C/li\u003E\r\n\u003Cli\u003E\u003Ca href=\"https://www.snowflake.com/en/legal/\"\u003ELegal\u003C/a\u003E\u003C/li\u003E\r\n\u003C/ul\u003E\r\n","richText":true,":type":"snowflake-site/components/text","appliedCssClassNames":"text-size-small text-color-text-04"},"markup_editor":{"id":"markup-editor-26ee015992","title":" ","htmlContent":"\u003Cdiv class=\"sf-footer__social\"\u003E\r\n\u003Cdiv data-testid=\"snowflake-footer-twitter\" class=\"snowflake-button-icon snowflake-button-white snowflake-footer-social-item\"\u003E\u003Cdiv class=\"snowflake-button-icon \"\u003E\u003Ca href=\"https://x.com/Snowflake\" data-testid=\"button-external\" aria-label=\"X (Twitter)\" role=\"button\" class=\"snowflake-button-container\" title=\"X (Twitter)\" tabindex=\"0\" target=\"_blank\" rel=\"noreferrer\"\u003E\u003Cdiv data-testid=\"button-icon-wrapper\"\u003E\u003Csvg xmlns=\"http://www.w3.org/2000/svg\" fill=\"none\" viewBox=\"0 0 59 53\" class=\"button-icon\"\u003E\u003Cpath fill=\"currentColor\" d=\"M46.614 0h9.044L35.8 22.49 59 53H40.795L26.54 34.46 10.223 53H1.18l21.036-24.055L0 0h18.657l12.878 16.937zM43.45 47.72h5.013L16.023 5.085h-5.387z\"\u003E\u003C/path\u003E\u003C/svg\u003E\u003C/div\u003E\u003C/a\u003E\u003Cdiv\u003E\u003C/div\u003E\u003C/div\u003E\u003C/div\u003E\u003Cdiv data-testid=\"snowflake-footer-linkedin\" class=\"snowflake-button-icon snowflake-button-white snowflake-footer-social-item\"\u003E\u003Cdiv class=\"snowflake-button-icon \"\u003E\u003Ca href=\"https://www.linkedin.com/company/3653845\" data-testid=\"button-external\" aria-label=\"LinkedIn\" role=\"button\" class=\"snowflake-button-container\" title=\"LinkedIn\" tabindex=\"0\" target=\"_blank\" rel=\"noreferrer\"\u003E\u003Cdiv data-testid=\"button-icon-wrapper\"\u003E\u003Csvg xmlns=\"http://www.w3.org/2000/svg\" fill=\"currentColor\" viewBox=\"0 0 24 24\" class=\"button-icon\"\u003E\u003Cpath d=\"M22.223 0H1.772C.792 0 0 .773 0 1.73v20.536C0 23.222.792 24 1.772 24h20.451c.98 0 1.777-.778 1.777-1.73V1.73C24 .773 23.203 0 22.223 0ZM7.12 20.452H3.558V8.995H7.12v11.457ZM5.34 7.434a2.064 2.064 0 1 1 0-4.125 2.063 2.063 0 0 1 0 4.125Zm15.112 13.018h-3.558v-5.57c0-1.326-.024-3.037-1.852-3.037-1.851 0-2.133 1.449-2.133 2.944v5.663H9.356V8.995h3.413v1.566h.047c.473-.9 1.636-1.852 3.365-1.852 3.605 0 4.27 2.372 4.27 5.457v6.286Z\"\u003E\u003C/path\u003E\u003C/svg\u003E\u003C/div\u003E\u003C/a\u003E\u003Cdiv\u003E\u003C/div\u003E\u003C/div\u003E\u003C/div\u003E\u003Cdiv data-testid=\"snowflake-footer-facebook\" class=\"snowflake-button-icon snowflake-button-white snowflake-footer-social-item\"\u003E\u003Cdiv class=\"snowflake-button-icon \"\u003E\u003Ca href=\"https://www.facebook.com/snowflakedb/\" data-testid=\"button-external\" aria-label=\"Facebook\" role=\"button\" class=\"snowflake-button-container\" title=\"Facebook\" tabindex=\"0\" target=\"_blank\" rel=\"noreferrer\"\u003E\u003Cdiv data-testid=\"button-icon-wrapper\"\u003E\u003Csvg xmlns=\"http://www.w3.org/2000/svg\" fill=\"currentColor\" viewBox=\"0 0 24 24\" class=\"button-icon\"\u003E\u003Cpath d=\"M24 12c0-6.627-5.373-12-12-12S0 5.373 0 12c0 5.99 4.388 10.954 10.125 11.854V15.47H7.078V12h3.047V9.356c0-3.007 1.792-4.668 4.533-4.668 1.312 0 2.686.234 2.686.234v2.953H15.83c-1.491 0-1.956.925-1.956 1.875V12h3.328l-.532 3.469h-2.796v8.385C19.612 22.954 24 17.99 24 12Z\"\u003E\u003C/path\u003E\u003C/svg\u003E\u003C/div\u003E\u003C/a\u003E\u003Cdiv\u003E\u003C/div\u003E\u003C/div\u003E\u003C/div\u003E\u003Cdiv data-testid=\"snowflake-footer-youtube\" class=\"snowflake-button-icon snowflake-button-white snowflake-footer-social-item\"\u003E\u003Cdiv class=\"snowflake-button-icon \"\u003E\u003Ca href=\"https://www.youtube.com/user/snowflakecomputing\" data-testid=\"button-external\" aria-label=\"YouTube\" role=\"button\" class=\"snowflake-button-container\" title=\"YouTube\" tabindex=\"0\" target=\"_blank\" rel=\"noreferrer\"\u003E\u003Cdiv data-testid=\"button-icon-wrapper\"\u003E\u003Csvg xmlns=\"http://www.w3.org/2000/svg\" fill=\"currentColor\" viewBox=\"0 0 24 24\" class=\"button-icon\"\u003E\u003Cpath d=\"M23.76 7.2s-.233-1.655-.955-2.381c-.914-.956-1.936-.961-2.405-1.017-3.356-.244-8.395-.244-8.395-.244h-.01s-5.039 0-8.395.244c-.469.056-1.49.06-2.405 1.017C.473 5.545.244 7.2.244 7.2S0 9.145 0 11.086v1.819c0 1.94.24 3.886.24 3.886s.233 1.654.95 2.38c.915.957 2.115.924 2.65 1.027 1.92.183 8.16.24 8.16.24s5.044-.01 8.4-.249c.469-.056 1.49-.06 2.405-1.017.722-.727.956-2.381.956-2.381S24 14.85 24 12.905v-1.819c0-1.94-.24-3.886-.24-3.886ZM9.52 15.113V8.367l6.483 3.385-6.483 3.36Z\"\u003E\u003C/path\u003E\u003C/svg\u003E\u003C/div\u003E\u003C/a\u003E\u003Cdiv\u003E\u003C/div\u003E\u003C/div\u003E\u003C/div\u003E\r\n\u003C/div\u003E",":type":"snowflake-site/components/markup-editor","isGSAPEnabled":false}},":itemsOrder":["container","text_copy_copy_16360","markup_editor"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-none"}},":itemsOrder":["container"]},":type":"snowflake-site/components/flexible-column-container","isActiveTOC":false,"isBlogPage":false}},":itemsOrder":["flexible_column_cont"],"appliedCssClassNames":"snowflake-container snowflake-responsive-container-inner-padding-small"}},":itemsOrder":["container_112062425"],"appliedCssClassNames":"snowflake-responsive-container-inner-padding-none"},"markup_editor_copy":{"id":"markup-editor-92460140e1","title":"New css","cssContent":".snowflake-image-container img{background-color:transparent}div.snowflake-person-chip-avatar{width:80px !important}#snowflake-blog-template-main-container .aem-GridColumn:has(.vertical-video){background-color:#000;border-radius:16px;overflow:hidden}#snowflake-blog-template-main-container .vertical-video{max-width:240px;margin-left:auto;margin-right:auto}@media screen and (min-width:1367px){.dynamic .heading-1-v2 .snowflake-title-v2-line{font-size:72px !important;line-height:60px !important}}.snowflake-flexible-column-container-items-alignment-match-height .download-card,.snowflake-flexible-column-container-items-alignment-match-height .download-card\u003E.container{height:100%}.download-card div.code-toolbar\u003E.toolbar .copy-to-clipboard-button{background-color:white;border:1px solid #a9e1f6;margin-right:4px;top:6px;border-radius:16px;height:26px;width:40px}.download-card .snowflake-code-snippet\u003Ediv.code-toolbar\u003E.toolbar\u003E.toolbar-item\u003Ebutton:before{content:'';background-image:url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 24 24' fill='none' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='9' y='9' width='13' height='13' rx='2' ry='2' style='stroke:%23249EDC;'%3E%3C/rect%3E%3Cpath d='M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1' style='stroke:%23249EDC;'%3E%3C/path%3E%3C/svg%3E\");background-size:auto 65%;background-position:center;background-repeat:no-repeat;top:0;left:0;width:100%;height:100%}.download-card .snowflake-code-snippet\u003Ediv.code-toolbar\u003E.toolbar\u003E.toolbar-item\u003Ebutton:hover:before{background-image:url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' viewBox='0 0 24 24' fill='none' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Crect x='9' y='9' width='13' height='13' rx='2' ry='2' style='stroke:%23fff;'%3E%3C/rect%3E%3Cpath d='M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1' style='stroke:%23fff;'%3E%3C/path%3E%3C/svg%3E\")}.download-card\u003Ediv{background-color:#fff;border:1px solid #ccc;border-radius:8px;padding:24px}.download-chip__headline{border-bottom:1px solid #ccc;padding-bottom:16px;margin-bottom:16px}.download-chip{padding:8px 12px !important;border-radius:4px;transition:300ms ease background-color}.download-chip .black-blue-text-color .snowflake-title-v2-line{color:#000 !important;padding-right:24px;font-family:'Lato',sans-serif;font-size:14px !important;font-weight:500 !important}.download-chip .black-blue-text-color .snowflake-title-v2-line:not(:first-child){opacity:.6;font-style:italic !important}.download-chip .snowflake-content-chip-button{display:none}.download-chip.is-external-link{background-size:16px 16px;background-image:url(\"data:image/svg+xml,%3Csvg width='15' height='15' viewBox='0 0 15 15' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M1.06055 13.0607L11.8605 2.26067M13.0605 10.6607V1.06067H3.46055' stroke='%23249EDC' stroke-width='2.12132' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\")}.download-chip{background-image:url(\"data:image/svg+xml,%3Csvg width='18' height='18' viewBox='0 0 18 18' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cg clip-path='url(%23clip0_883_7979)'%3E%3Cpath d='M3.375 16.875H14.625' stroke='%23249EDC' stroke-width='1.40625' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M9 1.125V11.25' stroke='%23249EDC' stroke-width='1.40625' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M4.5 7.875L9 12.375L13.5 7.875' stroke='%23249EDC' stroke-width='1.40625' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/g%3E%3Cdefs%3E%3CclipPath id='clip0_883_7979'%3E%3Crect width='18' height='18' fill='white'/%3E%3C/clipPath%3E%3C/defs%3E%3C/svg%3E%0A\");background-size:24px auto;background-repeat:no-repeat;background-position:calc(100% - 12px) center}.download-chip__headline{display:flex;gap:16px;flex-direction:row !important;flex-wrap:nowrap}.download-chip__headline::before{content:'';display:inline-block;width:24px;height:24px;background-position:center;background-image:url(\"data:image/svg+xml,%3Csvg width='21' height='21' viewBox='0 0 21 21' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M7.50005 9.89999C8.13657 9.89999 8.74702 9.64713 9.19711 9.19704C9.64719 8.74696 9.90005 8.13651 9.90005 7.49999V2.69999C9.90005 2.06347 9.64719 1.45302 9.19711 1.00293C8.74702 .552844 8.13657 .299988 7.50005 .299988H2.70005C2.06353 .299988 1.45308 .552844 1.00299 1.00293C.552905 1.45302 .300049 2.06347 .300049 2.69999V7.49999C.300049 8.13651 .552905 8.74696 1.00299 9.19704C1.45308 9.64713 2.06353 9.89999 2.70005 9.89999H7.50005ZM7.50005 7.49999H2.70005V2.69999H7.50005V7.49999Z' fill='%23249EDC' stroke='white' stroke-width='.6'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M7.50005 20.3C8.13657 20.3 8.74702 20.0472 9.19711 19.5971C9.64719 19.147 9.90005 18.5365 9.90005 17.9V13.1C9.90005 12.4635 9.64719 11.853 9.19711 11.403C8.74702 10.9529 8.13657 10.7 7.50005 10.7H2.70005C2.06353 10.7 1.45308 10.9529 1.00299 11.403C.552905 11.853 .300049 12.4635 .300049 13.1V17.9C.300049 18.5365 .552905 19.147 1.00299 19.5971C1.45308 20.0472 2.06353 20.3 2.70005 20.3H7.50005ZM7.50005 17.9H2.70005V13.1H7.50005V17.9Z' fill='%23249EDC' stroke='white' stroke-width='.6'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M17.9001 9.89999C18.5366 9.89999 19.147 9.64713 19.5971 9.19704C20.0472 8.74696 20.3001 8.13651 20.3001 7.49999V2.69999C20.3001 2.06347 20.0472 1.45302 19.5971 1.00293C19.147 .552844 18.5366 .299988 17.9001 .299988H13.1001C12.4636 .299988 11.8531 .552844 11.403 1.00293C10.9529 1.45302 10.7001 2.06347 10.7001 2.69999V7.49999C10.7001 8.13651 10.9529 8.74696 11.403 9.19704C11.8531 9.64713 12.4636 9.89999 13.1001 9.89999H17.9001ZM17.9001 7.49999H13.1001V2.69999H17.9001V7.49999Z' fill='%23249EDC' stroke='white' stroke-width='.6'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M17.9001 20.3C18.5366 20.3 19.147 20.0472 19.5971 19.5971C20.0472 19.147 20.3001 18.5365 20.3001 17.9V13.1C20.3001 12.4635 20.0472 11.853 19.5971 11.403C19.147 10.9529 18.5366 10.7 17.9001 10.7H13.1001C12.4636 10.7 11.8531 10.9529 11.403 11.403C10.9529 11.853 10.7001 12.4635 10.7001 13.1V17.9C10.7001 18.5365 10.9529 19.147 11.403 19.5971C11.8531 20.0472 12.4636 20.3 13.1001 20.3H17.9001ZM17.9001 17.9H13.1001V13.1H17.9001V17.9Z' fill='%23249EDC' stroke='white' stroke-width='.6'/%3E%3C/svg%3E%0A\");background-size:contain;background-repeat:no-repeat}.download-chip__headline.is-cli::before{background-image:url(\"data:image/svg+xml,%3Csvg width='24' height='24' viewBox='0 0 24 24' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M4 17L10 11L4 5' stroke='%23000' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M12 19H20' stroke='%23000' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\")}.download-card pre[class*=language-]{padding:8px 12px;background-color:var(--ui-background-05);overflow:hidden}.download-chip__headline.is-windows,.download-chip__headline.is-mac{gap:12px}.download-chip__headline.is-windows::before{width:16px;height:20px;background-image:url(\"data:image/svg+xml,%3Csvg width='4875' height='4875' viewBox='0 0 4875 4875' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cg clip-path='url(%23clip0_122_201)'%3E%3Cpath d='M0 0H2311V2310H0V0ZM2564 0H4875V2310H2564V0ZM0 2564H2311V4875H0V2564ZM2564 2564H4875V4875H2564' fill='%23000'/%3E%3C/g%3E%3C/svg%3E\")}.download-chip__headline.is-mac::before{width:16px;height:20px;background-image:url(\"data:image/svg+xml,%3Csvg version='1.1' id='Layer_1' xmlns:x='ns_extend;' xmlns:i='ns_ai;' xmlns:graph='ns_graphs;' xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink' x='0' y='0' viewBox='0 0 41.5 51' style='enable-background:new 0 0 41.5 51;' xml:space='preserve'%3E%3Cmetadata%3E%3Csfw xmlns='ns_sfw;'%3E%3Cslices%3E%3C/slices%3E%3CsliceSourceBounds bottomLeftOrigin='true' height='51' width='41.5' x='166.1' y='-208.1'%3E%3C/sliceSourceBounds%3E%3C/sfw%3E%3C/metadata%3E%3Cg%3E%3Cpath d='M40.2,17.4c-3.4,2.1-5.5,5.7-5.5,9.7c0,4.5,2.7,8.6,6.8,10.3c-.8,2.6-2,5-3.5,7.2c-2.2,3.1-4.5,6.3-7.9,6.3s-4.4-2-8.4-2 c-3.9,0-5.3,2.1-8.5,2.1s-5.4-2.9-7.9-6.5C2,39.5,.1,33.7,0,27.6c0-9.9,6.4-15.2,12.8-15.2c3.4,0,6.2,2.2,8.3,2.2 c2,0,5.2-2.3,9-2.3C34.1,12.2,37.9,14.1,40.2,17.4z M28.3,8.1C30,6.1,30.9,3.6,31,1c0-.3,0-.7-.1-1c-2.9,.3-5.6,1.7-7.5,3.9 c-1.7,1.9-2.7,4.3-2.8,6.9c0,.3,0,.6,.1,.9c.2,0,.5,.1,.7,.1C24.1,11.6,26.6,10.2,28.3,8.1z'%3E%3C/path%3E%3C/g%3E%3C/svg%3E\")}.download-chip__headline.is-desktop::before{background-image:url(\"data:image/svg+xml,%3Csvg width='24' height='24' viewBox='0 0 24 24' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cg opacity='.8'%3E%3Cpath d='M1.5 21H22.5V18H1.5V21Z' fill='%23000' stroke='white' stroke-width='.75'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M19.5 15C20.2956 15 21.0587 14.6839 21.6213 14.1213C22.1839 13.5587 22.5 12.7956 22.5 12V6C22.5 5.20435 22.1839 4.44129 21.6213 3.87868C21.0587 3.31607 20.2956 3 19.5 3H4.5C3.70435 3 2.94129 3.31607 2.37868 3.87868C1.81607 4.44129 1.5 5.20435 1.5 6V12C1.5 12.7956 1.81607 13.5587 2.37868 14.1213C2.94129 14.6839 3.70435 15 4.5 15H19.5ZM19.5 12H4.5V6H19.5V12Z' fill='%23000' stroke='white' stroke-width='.75'/%3E%3C/g%3E%3C/svg%3E%0A\")}.download-card .snowflake-code-snippet,.download-card .snowflake-code-snippet code,.download-card .snowflake-code-snippet pre{font-size:14px;color:#000;text-shadow:none !important}.download-chip:hover{background-color:var(--ui-background-05) !important;transition:300ms ease background-color}body:has(.snowflake-skip-to-content[style]) #subNav,.pushdown-banner-dismissed #subNav{top:var(--scroll-padding-top) !important;transition:300ms ease top}body:has(.snowflake-skip-to-content[style*=\"58\"]) #subNav{top:34px !important}body:has(.snowflake-skip-to-content[style*=\"82\"]) #subNav{top:58px !important}body:has(.snowflake-skip-to-content[style*=\"130\"]) #subNav{top:106px !important}body:has(.snowflake-skip-to-content[style*=\"138\"]) #subNav{top:114px !important}body:has(.snowflake-skip-to-content[style*=\"146\"]) #subNav{top:122px !important}.is-hidden .snowflake-person-chip-avatar{display:none}.is-small .snowflake-person-chip-avatar{width:56px;height:56px}.ai-summary ul{margin:16px 0 0 0 !important;padding:0 !important;list-style-type:none}.ai-summary li{margin:0;padding:0 0 0 32px;position:relative}.ai-summary li::before{content:\"\";display:block;border-radius:100%;background:#29b5e8;width:18px;height:18px;position:absolute;top:4px;left:0;border:5px solid #e5f2f7;box-sizing:border-box}.ai-summary li:not(:last-child){margin-bottom:1rem}.snowflake-content-chip-image__image{aspect-ratio:5 / 3 !important}.content-chip-new .snowflake-content-chip-image__image{height:100% !important;aspect-ratio:unset !important}.snapshot-card .snowflake-text p:not(:first-child){margin-top:var(--spacing-01)}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:nth-child(2) p:has(b){font-family:'Texta',sans-serif;margin-top:24px}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:nth-child(2) p b{font-weight:700 !important}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:nth-child(2){border-bottom:1px solid #ccc;padding-bottom:24px}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:nth-child(3) p:first-child:has(b){font-family:'Texta',sans-serif;font-size:20px !important;margin-bottom:1rem !important}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:nth-child(3) li{display:inline-block}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:nth-child(3) li a{display:inline-block;text-decoration:none;padding:4px 16px !important;border:1px solid #ccc;border-radius:24px;color:#666 !important}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:nth-child(3) ul{list-style-type:none;display:flex;padding:0 !important;margin:0 !important;gap:12px}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container img{width:90%;max-width:240px;margin:0 auto}.snapshot-card\u003E.container\u003E.cmp-container\u003E.aem-container{padding:40px;max-width:450px;margin:0 0 0 auto;background-color:#fff;box-shadow:0 2px 6px 0 rgba(152,162,179,.25),0 10px 20px 0 rgba(152,162,179,.10);border-radius:8px;border-top:4px solid var(--ui-01)}.ai-summary{background-color:#f3fbfe;border-left:2px solid var(--ui-01);padding:40px}.ai-summary\u003Espan p:last-child:has(i){color:#666;font-size:14px !important}.ai-summary\u003Espan p:last-child:has(i) a{color:#666 !important;text-decoration:underline !important}.ai-summary\u003Espan p:last-child:has(i) a:hover{color:var(--ui-01) !Important}.ai-summary\u003Espan p:first-child:has(b)::after{content:'';display:inline-block;width:20px;height:20px;background-image:url(\"data:image/svg+xml,%3Csvg width='24' height='24' viewBox='0 0 24 24' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M9.3158 3.15226C8.6475 6.2258 6.22698 8.64545 3.15232 9.31587C2.94923 9.36072 2.94923 9.63928 3.15232 9.68413C6.22698 10.3522 8.6475 12.7742 9.3158 15.8477C9.36067 16.0508 9.63933 16.0508 9.6842 15.8477C10.3525 12.7742 12.773 10.3545 15.8477 9.68413C16.0508 9.63928 16.0508 9.36072 15.8477 9.31587C12.773 8.64781 10.3525 6.2258 9.6842 3.15226C9.63933 2.94925 9.36067 2.94925 9.3158 3.15226Z' fill='%23249EDC'/%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M17.3725 11.5461C16.9098 13.6739 15.2341 15.3491 13.1054 15.8132C12.9649 15.8443 12.9649 16.0371 13.1054 16.0681C15.2341 16.5307 16.9098 18.2074 17.3725 20.3353C17.4035 20.4758 17.5965 20.4758 17.6275 20.3353C18.0902 18.2074 19.7659 16.5323 21.8946 16.0681C22.0352 16.0371 22.0352 15.8443 21.8946 15.8132C19.7659 15.3507 18.0902 13.6739 17.6275 11.5461C17.5965 11.4055 17.4035 11.4055 17.3725 11.5461Z' fill='%23249EDC'/%3E%3C/svg%3E%0A\");background-repeat:no-repeat;background-size:contain;background-position:center;vertical-align:middle;margin-left:8px}.ai-summary\u003Espan p:first-child:has(b){color:var(--ui-01) !important;text-transform:uppercase}.border-top{border-top:1px solid rgba(0,0,0,.2)}.border-top\u003Espan{display:block;padding-top:32px}body .snowflake-card-v2-advanced-image__image{aspect-ratio:16 / 9 !important}.content-chip-new .snowflake-content-chip-image__image{border-radius:0;object-fit:cover;height:100%}.sf-footer #ot-sdk-btn.ot-sdk-show-settings,.sf-footer #ot-sdk-btn.optanon-show-settings{color:rgba(255,255,255,.7) !important;text-underline-offset:4px;border-top:none;border-left:none;border-right:none;border-bottom:1px dotted transparent;background-color:transparent !important;background-image:none !important;transition:300ms ease text-decoration-color;padding:0 !important;font-size:12px;font-family:'Lato',sans-serif}.sf-footer #ot-sdk-btn.ot-sdk-show-settings:hover,.sf-footer #ot-sdk-btn.optanon-show-settings:hover{color:rgba(255,255,255,1) !important;border-bottom:1px dotted var(--ui-01);transition:300ms ease text-decoration-color}.sf-footer__legal-container\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:last-child{flex-shrink:0}.sf-footer__disclaimers{background-color:#042130}.sf-footer__disclaimers .snowflake-simple-stat-disclaimer p a{color:inherit;text-decoration:none !important}.sf-footer__disclaimers .snowflake-simple-stat-disclaimer p sup{margin-right:2px}.sf-footer__disclaimers .snowflake-simple-stat-disclaimer p{text-indent:-5px;padding-left:5px}.sf-footer__disclaimers-inner{border-top:1px solid rgba(255,255,255,.25);padding:40px 0}.sf-footer__disclaimers .snowflake-simple-stat{align-items:flex-start;text-align:left;color:rgba(255,255,255,.7);margin-bottom:10px}.sf-footer__social{display:flex;justify-content:center;gap:12px}.sf-footer .snowflake-footer-social-item{margin:0 !important}.sf-footer .snowflake-footer-social-item a{line-height:0;background-color:rgba(3,24,35,.8);display:inline-block;width:48px !important;height:48px;border-radius:8px;display:inline-flex;justify-content:center;align-items:center;transition:300ms ease background-color}.sf-footer .snowflake-footer-social-item a:hover{background-color:var(--ui-01) !important;transition:300ms ease background-color}.sf-footer__bottom{padding-bottom:40px}.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoError .mktoErrorMsg{max-width:100%;color:#fff}.sf-footer .mktoForm .mktoError .mktoErrorMsg .mktoErrorDetail{display:inline-block}.sf-footer .mktoFormRow:has(.mktoHtmlText:empty){display:none}.sf-footer .mktoFormRow .mktoHtmlText span{color:#fff !important}.sf-footer{background-color:#042130}.sf-footer .optanon-toggle-display:hover{text-decoration-color:var(--ui-01) !important;cursor:pointer !important;text-underline-offset:4px;text-decoration-style:dotted !important;text-decoration-color:var(--ui-01);color:#fff !important;transition:300ms ease text-decoration-color;text-decoration:underline;opacity:1}.sf-footer__logo{width:40px}.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container{row-gap:32px}.sf-footer__legal-container\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;justify-content:space-between;align-items:center;text-align:center;row-gap:16px}.sf-footer__legal-container\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:nth-child(2){text-align:center;flex-grow:1}.sf-footer__legal-links li button,.sf-footer__legal-links li a,.sf-footer__legal-links li{margin:0;color:rgba(255,255,255,.7) !important;font-weight:500}.sf-footer__legal-links li a:hover{color:rgba(255,255,255,1) !important}.sf-footer div.sf-footer__copyright p,.sf-footer div.sf-footer__legal-links li,.sf-footer div.sf-footer__legal-links a,.sf-footer div.sf-footer__legal-links p{font-size:12px !important}.sf-footer__legal-links ul{list-style-type:none;margin:0;padding:0;display:flex;gap:20px;row-gap:4px;justify-content:center;flex-wrap:wrap;text-align:center}.sf-footer__legal-links li:last-child{width:100%}.sf-footer .mktoFormRow:has(.mktoPlaceholder),.sf-footer .mktoFormRow:has(input[type=\"hidden\"]){display:none !important}.sf-footer .mktoFormCol{margin-bottom:0 !important}.sf-footer label[for=\"adhoc1\"]{width:auto !important;flex-grow:1;margin-left:16px}.sf-footer .mktoFieldWrap:has(label[for=\"adhoc1\"]){display:flex;flex-direction:row-reverse;margin-top:22px}.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoCheckboxList input[type=checkbox]{background-color:transparent !important;border:1px solid rgba(255,255,255,.4) !important;border-radius:4px !important}.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoEmailField,.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoTelField,.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoTextField,.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap select{background-color:transparent !important;color:#fff !important;height:auto !important;border:1px solid rgba(255,255,255,.4) !important;border-radius:4px !important;padding:12px 18px !important}.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoEmailField:focus,.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoTelField:focus,.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoTextField:focus,.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap select:focus{border-color:var(--ui-01) !important}.sf-footer .mktoForm *{padding:0 !important}.sf-footer .mktoForm,.sf-footer .snowflake-marketo-form-container{padding:0 !important;background:transparent;margin-bottom:0;box-shadow:none}.sf-footer .mktoHtmlText.mktoHasWidth{width:100% !important;margin:24px 0}.sf-footer .mktoFormRow{flex-direction:column}.sf-footer .mktoForm .mktoButtonWrap{margin:0 !important}.sf-footer select{background-image:url(\"data:image/svg+xml,%3Csvg width='14' height='8' viewBox='0 0 14 8' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M.981445 1.43496L6.90897 7.32496L12.9314 1.33496' stroke='white' stroke-width='1.33333' stroke-miterlimit='10' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\") !important}.sf-footer .snowflake-marketo-form .mktoButtonWrap.mktoNative{justify-content:flex-start}.sf-footer *::placeholder{color:#fff !important;opacity:.8}.sf-footer .mktoForm .mktoButtonWrap.mktoSimple .mktoButton{background-color:var(--ui-01) !important;color:#fff !important;width:100% !important;padding:12px 16px !important;border:1px solid var(--ui-01) !important;background-image:none !important;border-radius:48px;text-transform:uppercase;font-weight:800 !important;font-family:'Texta',sans-serif !important;font-size:16px !important;line-height:1.2}.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoHtmlText\u003Espan,.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap .mktoLabel\u003Espan,.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap label.mktoLabel{color:#fff !important}.sf-footer__newsletter-title p:not(:first-child){margin-top:8px !important}.sf-footer__newsletter-title p b{font-weight:800 !important;font-family:'Texta',sans-serif !important;font-size:22px !important;line-height:1.2}.sf-footer__newsletter-title p:last-child{font-size:14px !important;opacity:.8}.sf-footer__link-group li a[target=\"_blank\"]::after{content:'';display:inline-block;width:10px;height:10px;margin-left:5px;background-image:url(\"data:image/svg+xml,%3Csvg width='11' height='11' viewBox='0 0 11 11' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M6.72222 1.22222C6.38471 1.22222 6.11111 .948616 6.11111 .611111C6.11111 .273607 6.38471 0 6.72222 0H10.3889C10.551 0 10.7064 .0643867 10.821 .178988C10.9356 .293596 11 .449032 11 .611111V4.27778C11 4.61529 10.7264 4.88889 10.3889 4.88889C10.0514 4.88889 9.77778 4.61529 9.77778 4.27778V2.08647L4.09879 7.76545C3.86013 8.00409 3.4732 8.00409 3.23454 7.76545C2.99589 7.52681 2.99589 7.13986 3.23454 6.90122L8.91355 1.22222H6.72222ZM0 2.44444C0 1.76943 .547207 1.22222 1.22222 1.22222H4.27778C4.61529 1.22222 4.88889 1.49583 4.88889 1.83333C4.88889 2.17084 4.61529 2.44444 4.27778 2.44444H1.22222V9.77778H8.55556V6.72222C8.55556 6.38471 8.82915 6.11111 9.16667 6.11111C9.50418 6.11111 9.77778 6.38471 9.77778 6.72222V9.77778C9.77778 10.4528 9.23059 11 8.55556 11H1.22222C.547207 11 0 10.4528 0 9.77778V2.44444Z' fill='white'/%3E%3C/svg%3E%0A\");background-size:contain;background-repeat:no-repeat;background-position:center}.sf-footer__link-group ul,.sf-footer__link-group li{margin:0;padding:0;list-style-type:none}.sf-footer__link-group ul{margin-top:20px !important}.sf-footer__link-group li{margin-top:15px}.sf-footer div.sf-footer__link-group\u003Espan\u003Ep\u003Ea,.sf-footer div.sf-footer__link-group\u003Espan\u003Ep{color:var(--ui-01) !important;font-weight:800 !important;font-family:'Texta',sans-serif !important;font-size:20px !important;line-height:1.2}.sf-footer__link-group li a{opacity:.9;color:#fff !important;font-weight:500 !important;font-size:15px !important;line-height:1.3}.sf-footer__link-group li a:hover{opacity:1}.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container::before,.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container::after{display:none}.sf-footer__column{flex-grow:1}.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:not(:first-child){width:50%}@media (min-width:800px){.sf-footer__legal-links ul{justify-content:flex-start;text-align:left}.sf-footer__social{justify-content:flex-end}.sf-footer__legal-links ul{padding-left:24px}.sf-footer__legal-container\u003E.container\u003E.cmp-container\u003E.aem-container{text-align:right;flex-wrap:nowrap}.sf-footer__legal-links.align-left ul{justify-content:flex-start}.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;justify-content:space-between;flex-direction:row}.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:auto !important;max-width:200px}.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:first-child{flex-grow:1;order:2;width:100% !important;max-width:none}.sf-footer__legal-container\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:auto}}@media screen and (min-width:1380px){.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container{flex-wrap:nowrap}.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:first-child{padding-right:48px;max-width:380px;background-color:rgba(3,24,35,.4);padding:32px;margin-left:48px;border-radius:16px}.sf-footer__link-group li,.sf-footer__link-group li a{font-size:14px !important;line-height:1.3}}@media screen and (max-width:991px){.sf-footer-grid__inner\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv:first-child{order:2;margin-top:24px !important}}@media screen and (max-width:420px){.is-reduced-mobile .heading-1-v2,.is-reduced-mobile .heading-1-v2-sm{font-size:32px;line-height:28px}}.quote-content-chip{background-color:var(--ui-background-05);padding:24px;border-radius:12px;position:relative}.quote-content-chip .black-blue-text-color .snowflake-title-v2-line\u003Espan{color:rgba(0,0,0,.8) !important;font-size:15px !important;line-height:1.5 !important;font-family:'Lato',sans-serif;font-weight:400 !important}.quote-content-chip .black-blue-text-color .snowflake-title-v2-line\u003Espan:not(:first-child){max-width:calc(100% - 200px)}.quote-content-chip .black-blue-text-color .snowflake-title-v2-line\u003Espan:nth-child(2){font-family:'Texta',sans-serif;color:#000 !important;font-size:20px !important;font-weight:800 !important;margin-top:24px}.quote-content-chip .snowflake-content-chip-image{width:140px !important}@media screen and (min-width:992px){.quote-content-chip .snowflake-content-chip-image{position:absolute !important;bottom:24px;right:16px}}@media screen and (max-width:991px){.quote-content-chip .snowflake-content-chip-image{margin-bottom:40px}.quote-content-chip{flex-direction:column}}#spa-root{background-color:#fff}.lowercase .snowflake-title-v2-line{text-transform:none !important}.centered .snowflake-logo-content-container-inner{justify-content:center}div.snowflake-linklist-dropdown-menu{max-height:380px}.first-line-blue .snowflake-typographyv2 .snowflake-title-v2-line:first-child{color:var(--ui-01) !important}.is-front{position:relative;z-index:2}.use-case-body .snowflake-text h1,.use-case-body .snowflake-text h2,.use-case-body .snowflake-text h3,.use-case-body .snowflake-text h4,.use-case-body .snowflake-text h5,.use-case-body .snowflake-text h6{font-family:'Texta',sans-serif;color:#000;margin:.25rem 0 0 0}.pc-hero .button-group\u003E.container\u003E.cmp-container\u003E.aem-container{justify-content:flex-start}.sf-footer .mktoFormRow .mktoHtmlText span{font-family:'Lato',sans-serif !important}.snowflake-button-primary.snowflake-button-blue .snowflake-button-container{justify-content:center}.related-chip-25{background-color:#fff;border:1px solid rgba(204,204,204,.5);border-radius:8px;padding:20px;position:relative}.related-chip-25:hover{box-shadow:rgba(152,162,179,.1) 0 10px 20px 0}.related-chip-25:hover::after{right:24px;transition:300ms ease right}.related-chip-25::after{content:'';display:block;transition:300ms ease right;background-image:url(\"data:image/svg+xml,%3Csvg width='8' height='14' viewBox='0 0 8 14' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M7.66699 7C7.66699 6.6571 7.53559 6.32825 7.30169 6.08578L2.34446 .947072C1.84529 .429617 1.0164 .429617 .517219 .947072C.0427878 1.43887 .042788 2.21798 .517219 2.70978L4.65591 7L.51722 11.2902C.0427889 11.782 .0427887 12.5611 .51722 13.0529C1.0164 13.5704 1.84529 13.5704 2.34447 13.0529L7.30169 7.91421C7.53559 7.67175 7.66699 7.34289 7.66699 7Z' fill='%2329B5E8'/%3E%3C/svg%3E%0A\");width:8px;height:14px;display:block;position:absolute;right:30px;top:50%;transform:translateY(-50%);background-size:contain;background-position:center;background-repeat:no-repeat}.related-chip-25 .heading-5-v2{font-size:22px;line-height:1.1}.related-chip-25 .snowflake-content-chip-image{width:48px;flex-shrink:0}.related-chip-25 .snowflake-content-chip-image__image{aspect-ratio:1;height:auto;object-fit:contain}.related-chip-25 .snowflake-content-chip-button{display:none}.related-chip-25 .snowflake-content-chip-content-without-tag{flex-grow:1;padding-right:24px}.case-study-25.small-logo .snowflake-case-study-card-logo img{width:60px !important}.swiper-slide .case-study-25{width:95%;margin-left:auto;margin-right:auto}.case-study-25 .snowflake-case-study-card-logo img{width:140px !important;height:auto !important;transform:none !important;margin:24px 0 8px 0}.case-study-25 .snowflake-case-study-card-image__image{object-position:left center}.case-study-25 .snowflake-case-study-card-information-container{padding-right:24px}.case-study-25 ul{list-style-type:none;padding:0;margin:8px 0 0 0}.case-study-25 li{font-size:15px !important;line-height:1.3 !important;display:flex;flex-direction:column;border-left:4px solid var(--ui-01);padding-left:24px;margin-top:24px;color:#535862;gap:4px}.case-study-25 li b{display:block;font-family:'Texta',sans-serif;font-weight:900 !important;font-size:48px !important;line-height:.9 !important;color:var(--ui-01)}.case-study-25 .snowflake-case-study-card-description p{color:#535862}.case-study-25 .snowflake-case-study-card-description p:nth-child(2):not(:has(a)){color:#000;font-family:Texta;font-size:30px !important;line-height:1 !important;font-style:normal;font-weight:700;text-indent:-8px}.case-study-25.is-story .snowflake-case-study-card-description p:nth-child(2):not(:has(a)){text-indent:0}.case-study-25 .snowflake-case-study-card-key-card{background-color:transparent}.case-study-25 .snowflake-case-study-card-button{display:none}.case-study-25{border-radius:24px;overflow:hidden}@media screen and (min-width:1024px){.case-study-25 .snowflake-case-study-card-left-container{position:static;width:60%;min-height:0}.case-study-25 .snowflake-case-study-card-right-container::after{content:'';display:block;width:60%;max-width:340px;padding-bottom:50%;background-image:url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 22 16' class='snowflake-pushdown-banner-placeholder-arrow'%3E%3Cpath fill='%2329B5E8' fill-rule='evenodd' d='M17.865 8.756c.088-.274.124-.555.118-.834a2.551 2.551 0 0 0-1.3-2.142L7.887.76C6.645.055 5.063.475 4.35 1.7a2.535 2.535 0 0 0 .947 3.494l4.916 2.809-4.916 2.801a2.543 2.543 0 0 0-.947 3.502c.713 1.222 2.295 1.64 3.537.934l8.796-5.024a2.541 2.541 0 0 0 1.182-1.46Z' clip-rule='evenodd'%3E%3C/path%3E%3C/svg%3E\");background-size:contain;background-repeat:no-repeat;position:absolute;top:-10%;left:-20%}.case-study-25 .snowflake-case-study-card-right-container{max-width:none;width:40%;position:absolute;top:-5%;right:-5%;z-index:0;height:110%}}@media screen and (min-width:768px){.case-study-25 li{max-width:50%}.case-study-25 ul{display:flex;gap:48px}}.snowflake-text.section-eyebrow p{margin-left:auto;margin-right:auto;margin-bottom:16px !important}.snowflake-text.section-eyebrow p,.snowflake-text.eyebrow-text p{text-transform:uppercase;font-family:'Texta',sans-serif !important;font-weight:800 !important;letter-spacing:.025em;margin-bottom:12px;line-height:1.1 !important}.snowflake-title-v2.dynamic .heading-2-v2 span.snowflake-title-v2-line{font-size:clamp(2.5rem,4.5vw,4rem) !important;line-height:.82 !important}.checklist ul{padding:0;margin:0}.checklist ul li{list-style-type:none;padding-left:32px;position:relative}.checklist ul li:not(:last-child){margin-bottom:1em}.checklist ul li::before{content:'';display:inline-block;width:20px;height:20px;background-image:url(\"data:image/svg+xml,%3Csvg width='24' height='25' viewBox='0 0 24 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Crect y='.985352' width='24' height='24' rx='12' fill='%23D4F0FA'/%3E%3Cpath d='M7.28613 13.2967L10.7147 16.7253L17.5718 9.86816' stroke='%2329B5E8' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");background-size:contain;background-repeat:no-repeat;position:absolute;top:3px;left:0}.last-line-blue .snowflake-typographyv2 .snowflake-title-v2-line:last-child{color:var(--ui-01)}.snowflake-text p sup{line-height:0}.snowflake-title-v2.lowercase .heading-3-v2{font-size:28px;line-height:1;text-transform:none;font-weight:700}.snowflake-title-v2.lowercase .heading-2-v2{font-size:32px;line-height:1;text-transform:none;font-weight:700}.content-chip-new{border:1px solid rgba(204,204,204,.5);border-radius:16px;overflow:hidden}.content-chip-new .snowflake-image-container{border-radius:0;display:none}.content-chip-new .snowflake-content-chip-image{margin-right:0;max-width:180px;flex-shrink:0}.content-chip-new .snowflake-content-chip-content{padding:24px}.content-chip-new .black-blue-text-color .snowflake-title-v2-line:first-child{font-size:24px;line-height:1.1}.content-chip-new .black-blue-text-color .snowflake-title-v2-line:not(:first-child){font-family:'Lato',sans-serif;font-size:17px;color:#535862 !important;font-weight:500;line-height:1.45;margin-top:8px;display:none}div.snowflake-text a{font-weight:normal;color:var(--ui-01);text-decoration:underline;text-underline-offset:4px;text-decoration-style:dotted !important;text-decoration-color:transparent;transition:300ms ease text-decoration-color}div.snowflake-text a:hover{text-decoration-color:var(--ui-01);transition:300ms ease text-decoration-color}.footer-nav__link-group .snowflake-button-container,.subnav__item--button,.snowflake-card-v2-advanced-button .snowflake-button-container{justify-content:flex-start}.button-container\u003E.container\u003E.cmp-container\u003E.aem-container{align-items:center}.button-container\u003E.container\u003E.cmp-container\u003E.aem-container .snowflake-button-primary+.snowflake-button-link{margin-left:12px !important}.snowflake-button-regular.snowflake-button-link .snowflake-button-container{font-size:18px !important;text-align:left;justify-content:flex-start;line-height:1.4 !important}body .snowflake-card-v2-advanced{border:1px solid rgba(204,204,204,.5);border-radius:var(--spacing-02);transition:300ms ease all}body .snowflake-card-v2-advanced:hover{transform:translateY(-10px);box-shadow:rgba(152,162,179,.1) 0 10px 20px 0;transition:300ms ease all}body .snowflake-card-v2-advanced-inner{border-bottom:none}body .snowflake-card-v2-advanced-image{line-height:0}body .snowflake-card-v2-advanced-image__image{aspect-ratio:16 / 9}body .snowflake-card-v2-advanced-content{position:relative}body .snowflake-card-v2-advanced-content::after{content:'';display:block;position:absolute;bottom:0;left:0;transition:300ms ease all;width:20%;height:4px;background-color:var(--ui-01);opacity:0}body .snowflake-card-v2-advanced:hover .snowflake-card-v2-advanced-content::after{width:100%;opacity:1;transition:300ms ease all}body .snowflake-card-v2-advanced .snowflake-button-link.snowflake-button-blue .snowflake-button-container\u003E.link-icon{transition:300ms ease transform}body .snowflake-card-v2-advanced:hover .snowflake-button-link.snowflake-button-blue .snowflake-button-container\u003E.link-icon{transform:translateX(4px);transition:300ms ease transform}.six-columns\u003E.container\u003E.cmp-container\u003E.aem-container,.three-columns\u003E.container\u003E.cmp-container\u003E.aem-container,.four-columns\u003E.container\u003E.cmp-container\u003E.aem-container,.five-columns\u003E.container\u003E.cmp-container\u003E.aem-container{display:flex;flex-wrap:wrap;gap:24px}.six-columns.align-center\u003E.container\u003E.cmp-container\u003E.aem-container,.three-columns.align-center\u003E.container\u003E.cmp-container\u003E.aem-container,.four-columns.align-center\u003E.container\u003E.cmp-container\u003E.aem-container,.five-columns.align-center\u003E.container\u003E.cmp-container\u003E.aem-container{justify-content:center}.three-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:100%;margin:0 !important}.six-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv,.four-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv,.five-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(50% - 12px);margin:0 !important}@media screen and (min-width:768px){.three-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(50% - 12px)}.six-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv,.four-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv,.five-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(33.333% - 16px)}}@media screen and (min-width:1024px){.snowflake-title-v2.lowercase .heading-3-v2{font-size:34px}.snowflake-title-v2.lowercase.larger .heading-2-v2{font-size:44px;line-height:.95}.three-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(33.333% - 16px)}.four-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(25% - 18px)}.five-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(20% - 19.2px)}.six-columns\u003E.container\u003E.cmp-container\u003E.aem-container\u003Ediv{width:calc(16.6666% - 20px)}.snowflake-title-v2.lowercase .heading-3-v2{font-size:28px !important}}@media screen and (min-width:1200px){.snowflake-title-v2.lowercase .heading-2-v2{font-size:40px}.content-chip-new .snowflake-content-chip-content{padding:32px}.content-chip-new .snowflake-image-container,.content-chip-new .black-blue-text-color .snowflake-title-v2-line:not(:first-child){display:block}}.promo-banner-25{border-radius:16px;overflow:hidden}.promo-banner-25 .snowflake-premium-content-banner-image-container{position:relative;max-width:380px}.promo-banner-25 .snowflake-text{color:#535862}.promo-banner-25 .snowflake-premium-content-banner-image__image{transform:translateY(8px);transition:300ms ease transform;border-radius:0;width:85%;margin:0 auto;display:block;position:relative;z-index:1}.promo-banner-25 .snowflake-premium-content-banner-image__link:hover .snowflake-premium-content-banner-image__image{transform:translateY(0);transition:300ms ease transform}.promo-banner-25 .snowflake-premium-content-banner-image__inner{height:auto;padding-top:24px}.promo-banner-25 .snowflake-premium-content-banner-image__link{position:relative;z-index:1;height:auto}.promo-banner-25 .snowflake-premium-content-banner-image__link::after{content:'';display:block;position:absolute;clip-path:polygon(0 0,66% 0,100% 100%,0 100%);bottom:0;left:0;width:100%;height:100%;background:var(--ui-01);transition:300ms ease width}.promo-banner-25 .snowflake-premium-content-banner-image__link:hover::after{width:110%;transition:300ms ease width}.sf-footer .snowflake-marketo-form .mktoFormRow .mktoFieldWrap select{background-position:95% 50%}.sf-footer__disclaimers .text-size-small .snowflake-text p{color:#fff !important;font-size:10px !important;opacity:.8}@media screen and (min-width:768px){.sf-footer__disclaimers .text-size-small .snowflake-text p{font-size:12px !important}}@media screen and (max-width:1023px){.mobile-top-padding{padding-top:64px}}@media (max-width:799px){.sf-footer .snowflake-marketo-form .mktoButtonWrap.mktoNative .mktoButton{width:100% !important}.sf-footer__logo{text-align:center;display:block;margin:0 auto}}.customer-card .snowflake-card-v2-advanced-image{aspect-ratio:4.35 / 1}.customer-card .snowflake-card-v2-advanced-image__image{width:100%;height:100%;padding-left:8px;object-fit:contain;object-position:left center;margin:0 !important;aspect-ratio:initial}.customer-card .snowflake-card-v2-advanced-image__inner{height:110px}.customer-card .snowflake-card-v2-advanced-tag-indicator{display:none}.pc-hero .snowflake-container-arrow-small-gray-image{top:-34% !important;width:18% !important}.pc-hero .snowflake-container-arrow-small-gray-image path{fill:var(--ui-01);opacity:1}@media screen and (max-width:767px){.mobile-padding-top{padding-top:64px}.hide-mobile{display:none !important}.pc-hero{padding-top:52px}.pc-hero .snowflake-text p,.pc-hero .left-alignment .snowflake-title-v2-line,.pc-hero h1 span{text-align:center !important}}div.snowflake-pushdown-banner-button{margin-top:0}.button-group.align-center\u003E.container\u003E.cmp-container\u003E.aem-container{align-items:center;justify-content:center !important}.text-center .snowflake-breadcrumb-swiper .swiper-wrapper{justify-content:center}div.snowflake-breadcrumb a.snowflake-breadcrumb-item,.snowflake-breadcrumb div.snowflake-breadcrumb-item{text-transform:none;font-weight:500}.snowflake-breadcrumb svg{display:none !important}.snowflake-breadcrumb a:has(svg)::after{content:'/';margin:0 12px;color:#666}.hide-filters .snowflake-filterable-and-searchable-grid-top-part{display:none !important}.page-section{padding-left:24px;padding-right:24px}@media screen and (min-width:768px){.page-section{padding-left:48px;padding-right:48px}}.download-card pre[class*=language-]{overflow-x:scroll !important}",":type":"snowflake-site/components/markup-editor","isGSAPEnabled":false}},":itemsOrder":["container_copy","container_573483281_","markup_editor_copy"]}},":itemsOrder":["root"],"classNames":"aem-xf"},"markup_editor":{"id":"markup-editor-cd49203432","title":"Quickstarts Overrides","cssContent":".snowflake-markdown blockquote{padding:24px 32px;background:#f6f9fa;border:1px solid #29b5e8;border-radius:16px}.snowflake-markdown .snowflake-image-container img{width:auto !important;max-width:100%}.snowflake-markdown .snowflake-text ol{padding-left:20px !important}.snowflake-markdown .snowflake-text li{margin:0 0 12px 0 !important}.snowflake-markdown h3.snowflake-markdown-h3{font-size:20px !important;font-family:Texta,sans-serif !important}@media (min-width:768px){.snowflake-markdown h3.snowflake-markdown-h3{font-size:28px !important}}",":type":"snowflake-site/components/markup-editor","isGSAPEnabled":false}},":itemsOrder":["experiencefragment-banner","experiencefragment-header","markup_editor_1950346551","responsivegrid","modal_container","experiencefragment-footer","markup_editor"],":type":"wcm/foundation/components/responsivegrid"}},":itemsOrder":["root"],":hierarchyType":"page",":path":"/content/snowflake-site/global/en/developers/guides/end2endmigration","coveoConfig":{"pipeline":"snowflake.com","apiKey":"xx335921a6-2a0a-40f2-a167-e390b4766c3d","organizationId":"snowflakecomputingproduction8neljofn","searchHub":"snowflake.com"},"analyticsData":{"excludeFromAnalytics":false,"subCategory":"","pageType":"quickstart-page-template","templateName":"quickstart-page-template","siteName":"snowflake","pageUrl":"/content/snowflake-site/global/en/developers/guides/end2endmigration","language":"en","category":"general","pageName":"End-to-End Migration to Snowflake in Action: Data and Pipelines","contentTags":["snowflake-site:taxonomy/product/data-engineering","snowflake-site:taxonomy/snowflake-feature/transformation","snowflake-site:taxonomy/solution-center/certification/quickstart"]},"isPasswordProtected":false,"locale":"en"}
  