@nextblock-cms/ecom 0.9.95 → 0.9.99

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,5 +1,5 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const V=require("@supabase/supabase-js"),Z=require("crypto"),oe=require("@freemius/sdk"),ce=require("../customer.cjs.js"),Q=require("../coupon-server.cjs.js"),G=require("../customer-addresses.cjs.js"),j=require("../currency.cjs.js");function i(e){const r=process.env[e];if(!r)return null;const t=r.trim();return t.startsWith('"')&&t.endsWith('"')||t.startsWith("'")&&t.endsWith("'")?t.slice(1,-1).trim():t}function ae(e){const r=e?.trim();if(!r)return{firstName:null,lastName:null};const[t,...s]=r.split(/\s+/);return{firstName:t||null,lastName:s.length>0?s.join(" "):null}}function ee(e){if(e==null||e==="")return 0;const r=Number(e);return!Number.isFinite(r)||r<=0?0:Math.round(r)}function ue(e){return e===!0||e===1||e==="1"||e==="true"}function le(e,r){const t=e?.[r];return Array.isArray(t)?t:t&&typeof t=="object"?[t]:[]}function H(e,r){for(const t of r){const s=le(e,t);if(s.length>0)return s}return Array.isArray(e)?e:[]}function A(e){if(e==null||e==="")return null;const r=Number(e);return Number.isFinite(r)?r:null}function de(e){return e===null?0:e>5e3?(console.warn(`[Freemius Sync] Suspiciously high price detected: ${e}. Assuming it is already in cents.`),Math.round(e)):Math.round(e*100)}function pe(e){const r=Number(e);return!Number.isFinite(r)||r<1?1:Math.round(r)}function re(){const e=i("FREEMIUS_CHECKOUT_PRODUCTS_JSON");if(!e)return null;try{const r=JSON.parse(e);return r&&typeof r=="object"?r:null}catch(r){return console.error("[Freemius Checkout] Failed to parse FREEMIUS_CHECKOUT_PRODUCTS_JSON:",r),null}}function te(e){const r=re(),t=String(e),s=r?.[t],c=i("FREEMIUS_PRODUCT_ID"),d=i("FREEMIUS_ECOMMERCE_SANDBOX_PUBLIC_KEY"),f=i("FREEMIUS_ECOMMERCE_SANDBOX_SECRET_KEY");return s?.publicKey?{publicKey:s.publicKey,secretKey:s.secretKey??null,apiKey:s.apiKey??null,source:"product-map"}:process.env.FREEMIUS_SANDBOX_ENABLED==="true"&&c&&c===t&&d?{publicKey:d,secretKey:f,apiKey:i("FREEMIUS_API_KEY"),source:"single-product-sandbox-env"}:c&&c===t&&i("FREEMIUS_PUBLIC_KEY")?{publicKey:i("FREEMIUS_PUBLIC_KEY"),secretKey:i("FREEMIUS_SECRET_KEY"),apiKey:i("FREEMIUS_API_KEY"),source:"single-product-env"}:{publicKey:i("FREEMIUS_PUBLIC_KEY"),secretKey:i("FREEMIUS_SECRET_KEY"),apiKey:i("FREEMIUS_API_KEY"),source:"legacy-env"}}async function _e(e){if(!e.apiKey)throw new Error("Missing Freemius API key for SDK sandbox generation.");return new oe.Freemius({productId:Number(e.productId),apiKey:e.apiKey,secretKey:e.secretKey,publicKey:e.publicKey}).checkout.getSandboxParams()}class me{getProviderName(){return"Freemius"}async createCheckoutSession({items:r,customerEmail:t,customerPhone:s,userId:c,billingAddress:d,shippingAddress:f,currencyCode:m,couponCode:b,couponContextItems:g}){const a=process.env.NEXT_PUBLIC_SUPABASE_URL,u=process.env.SUPABASE_SERVICE_ROLE_KEY;if(!a||!u)return{error:"Missing Supabase credentials for checkout (Service Key required).",url:null};const p=V.createClient(a,u);if(!r||r.length===0)return{error:"Cart is empty",url:null};if(r.length!==1)return{error:"Freemius items must be checked out one at a time.",url:null};const{data:E,error:w}=await p.from("currencies").select("code, symbol, exchange_rate, is_default, is_active, auto_sync_product_prices, auto_update_exchange_rate, exchange_rate_source, exchange_rate_updated_at, rounding_mode, rounding_increment, rounding_charm_amount").eq("is_active",!0).order("code",{ascending:!0}),v=E??[];if(w||v.length===0)return{error:"Failed to resolve store currencies",url:null};const X=j.getDefaultCurrency(v),K=v.find(l=>l.code===(m||"").toUpperCase())??X,y=r[0],{data:o,error:M}=await p.from("products").select("id, title, price, prices, sale_price, sale_prices, sale_start_at, sale_end_at, scheduled_price, scheduled_prices, scheduled_price_at, freemius_plan_id, freemius_product_id, trial_period_days, trial_requires_payment_method").eq("id",y.product_id).single();if(M||!o)return{error:"Product not found",url:null};const U=o.freemius_plan_id,F=o.freemius_product_id;if(!U||!F)return{error:"Product is not configured for Freemius checkout (missing Plan ID or Product ID)",url:null};const D=j.resolveEffectivePriceForCurrency({prices:o.prices||{},salePrices:o.sale_prices||{},fallbackPrice:o.price,fallbackSalePrice:o.sale_price,saleStartAt:o.sale_start_at,saleEndAt:o.sale_end_at,scheduledPrice:o.scheduled_price,scheduledPrices:o.scheduled_prices||{},scheduledPriceAt:o.scheduled_price_at,currencyCode:K.code,currencies:v}),q=D.sale_price??D.price,N=j.isSaleWindowActive({saleStartAt:o.sale_start_at,saleEndAt:o.sale_end_at}),S=q*y.quantity;let n=null,_=0;if(b){const l=await Q.getCouponQuote({client:p,code:b,items:g&&g.length>0?g:r,currencyCode:K.code});if(!l.success)return{error:l.error,errorKey:l.errorKey,errorStatus:400,url:null};n=l.quote,_=Math.min(S,n.lineDiscounts.filter($=>$.product_id===o.id).reduce(($,L)=>$+L.discount,0))}const P=ee(o.trial_period_days),R=P>0?y.trial_preference?y.trial_preference:o.trial_requires_payment_method?"paid":"free":null,O="pending",h=ae(d?.recipient_name??null),{data:Y,error:J}=await p.from("orders").insert({status:O,total:Math.max(0,S-_),currency:K.code,exchange_rate_at_purchase:K.exchange_rate,subtotal:S,discount_total:_,coupon_id:n?.couponId??null,coupon_code:n?.code??null,discount_details:n?{code:n.code,discount_type:n.discountType,discount_amount:n.discountAmount,provider:"freemius",provider_discounts:n.providerDiscounts,line_discounts:n.lineDiscounts,final_amount_owned_by:"freemius"}:null,provider:"freemius",freemius_product_id:String(F),freemius_plan_id:String(U),user_id:c||null,customer_details:ce.normalizeOrderCustomerDetails({email:t,phone:s,name:d?.recipient_name,billing:d,shipping:f})}).select("id").single();if(J||!Y)return console.error("Failed to create pending order:",J),{error:"Failed to initiate order",url:null};const{error:W}=await p.from("order_items").insert([{order_id:Y.id,product_id:o.id,quantity:y.quantity,price_at_purchase:q}]);if(W&&console.error("Failed to insert order items:",W),c)try{await G.upsertDefaultUserAddresses({userId:c,billingAddress:d,shippingAddress:f,client:p}),await G.fillMissingUserProfileCheckoutDetails({userId:c,fullName:d?.recipient_name??f?.recipient_name??null,phone:s,client:p})}catch(l){console.error("Failed to sync checkout profile defaults before checkout:",l)}n&&await Q.recordCouponRedemption({client:p,quote:n,orderId:Y.id,provider:"freemius",discountTotal:_,userId:c,customerEmail:t,metadata:{currency:K.code,subtotal:S,final_amount_owned_by:"freemius"}});const x=process.env.FREEMIUS_SANDBOX_ENABLED==="true",I=te(F),k=I.publicKey,T=I.secretKey,z=I.apiKey;if(!k||x&&!T)return{error:"Missing FREEMIUS credentials (PUBLIC_KEY or SECRET_KEY) in environment variables.",url:null};if(x&&I.source==="legacy-env"){const l=i("FREEMIUS_PRODUCT_ID"),$=!!i("FREEMIUS_ECOMMERCE_SANDBOX_PUBLIC_KEY"),L=!!i("FREEMIUS_ECOMMERCE_SANDBOX_SECRET_KEY");console.warn(`[Freemius Checkout] Sandbox is enabled for product ${F}, but no product-scoped checkout credentials were selected. Falling back to legacy FREEMIUS_PUBLIC_KEY/FREEMIUS_SECRET_KEY may open live checkout instead of sandbox.`,{configuredProductId:l,productIdsMatch:l===String(F),hasSandboxOverridePublicKey:$,hasSandboxOverrideSecretKey:L,hasCheckoutProductsJson:!!i("FREEMIUS_CHECKOUT_PRODUCTS_JSON")})}let B=!1;if(x&&T&&k)try{B=await _e({productId:F,publicKey:k,secretKey:T,apiKey:z})}catch(l){console.warn("Freemius Checkout - SDK sandbox generation failed. Falling back to manual token generation.",l,{credentialSource:I.source,hasApiKey:!!z});const $=Math.floor(Date.now()/1e3).toString(),L=`${$}${F}${T}${k}checkout`,ie=Z.createHash("md5").update(L).digest("hex");B={ctx:$,token:ie}}const C=new URL(`https://checkout.freemius.com/app/${F}/plan/${U}/`);if(x&&T&&k?(C.searchParams.append("sandbox",B.token),C.searchParams.append("s_ctx_ts",B.ctx)):x&&C.searchParams.append("sandbox","true"),t&&C.searchParams.append("user_email",t),h.firstName&&C.searchParams.append("user_firstname",h.firstName),h.lastName&&C.searchParams.append("user_lastname",h.lastName),C.searchParams.append("currency",K.code.toLowerCase()),y.billing_cycle&&C.searchParams.append("billing_cycle",y.billing_cycle),R&&C.searchParams.append("trial",R),n)C.searchParams.append("coupon",n.code);else if(N){const{data:l}=await p.from("product_freemius_sale_coupons").select("freemius_coupon_code, is_active, starts_at, ends_at, sync_status").eq("product_id",o.id).maybeSingle();l?.is_active&&l.sync_status==="synced"&&l.freemius_coupon_code&&j.isSaleWindowActive({saleStartAt:l.starts_at,saleEndAt:l.ends_at})&&C.searchParams.append("coupon",l.freemius_coupon_code)}return{url:C.toString(),customProps:{provider:"freemius",plugin_id:F,plan_id:U,public_key:k,user_email:t,user_firstname:h.firstName,user_lastname:h.lastName,credential_source:I.source,sandbox:B,billing_cycle:y.billing_cycle,trial:R,trial_period_days:P,trial_requires_payment_method:o.trial_requires_payment_method,initial_order_status:O,coupon:n?.code??null,order_id:Y.id}}}}async function ne(e,r,t,s){const d=new Date().toUTCString().replace("GMT","+0000"),f=`GET
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const V=require("@supabase/supabase-js"),Z=require("crypto"),oe=require("@freemius/sdk"),ce=require("../customer.cjs.js"),Q=require("../coupon-server.cjs.js"),G=require("../customer-addresses.cjs.js"),j=require("../currency.cjs.js");function i(e){const r=process.env[e];if(!r)return null;const n=r.trim();return n.startsWith('"')&&n.endsWith('"')||n.startsWith("'")&&n.endsWith("'")?n.slice(1,-1).trim():n}function ae(e){const r=e?.trim();if(!r)return{firstName:null,lastName:null};const[n,...s]=r.split(/\s+/);return{firstName:n||null,lastName:s.length>0?s.join(" "):null}}function ee(e){if(e==null||e==="")return 0;const r=Number(e);return!Number.isFinite(r)||r<=0?0:Math.round(r)}function ue(e){return e===!0||e===1||e==="1"||e==="true"}function le(e,r){const n=e?.[r];return Array.isArray(n)?n:n&&typeof n=="object"?[n]:[]}function H(e,r){for(const n of r){const s=le(e,n);if(s.length>0)return s}return Array.isArray(e)?e:[]}function k(e){if(e==null||e==="")return null;const r=Number(e);return Number.isFinite(r)?r:null}function de(e){return e===null?0:e>5e3?(console.warn(`[Freemius Sync] Suspiciously high price detected: ${e}. Assuming it is already in cents.`),Math.round(e)):Math.round(e*100)}function pe(e){const r=Number(e);return!Number.isFinite(r)||r<1?1:Math.round(r)}function re(){const e=i("FREEMIUS_CHECKOUT_PRODUCTS_JSON");if(!e)return null;try{const r=JSON.parse(e);return r&&typeof r=="object"?r:null}catch(r){return console.error("[Freemius Checkout] Failed to parse FREEMIUS_CHECKOUT_PRODUCTS_JSON:",r),null}}function ne(e){const r=re(),n=String(e),s=r?.[n],c=i("FREEMIUS_PRODUCT_ID"),d=i("FREEMIUS_ECOMMERCE_SANDBOX_PUBLIC_KEY"),f=i("FREEMIUS_ECOMMERCE_SANDBOX_SECRET_KEY");return s?.publicKey?{publicKey:s.publicKey,secretKey:s.secretKey??null,apiKey:s.apiKey??null,source:"product-map"}:process.env.FREEMIUS_SANDBOX_ENABLED==="true"&&c&&c===n&&d?{publicKey:d,secretKey:f,apiKey:i("FREEMIUS_API_KEY"),source:"single-product-sandbox-env"}:c&&c===n&&i("FREEMIUS_PUBLIC_KEY")?{publicKey:i("FREEMIUS_PUBLIC_KEY"),secretKey:i("FREEMIUS_SECRET_KEY"),apiKey:i("FREEMIUS_API_KEY"),source:"single-product-env"}:{publicKey:i("FREEMIUS_PUBLIC_KEY"),secretKey:i("FREEMIUS_SECRET_KEY"),apiKey:i("FREEMIUS_API_KEY"),source:"legacy-env"}}async function _e(e){if(!e.apiKey)throw new Error("Missing Freemius API key for SDK sandbox generation.");return new oe.Freemius({productId:Number(e.productId),apiKey:e.apiKey,secretKey:e.secretKey,publicKey:e.publicKey}).checkout.getSandboxParams()}class me{getProviderName(){return"Freemius"}async createCheckoutSession({items:r,customerEmail:n,customerPhone:s,userId:c,billingAddress:d,shippingAddress:f,currencyCode:m,couponCode:b,couponContextItems:E}){const a=process.env.NEXT_PUBLIC_SUPABASE_URL||process.env.SUPABASE_URL,u=process.env.SUPABASE_SERVICE_ROLE_KEY||process.env.SUPABASE_SECRET_KEY;if(!a||!u)return{error:"Missing Supabase credentials for checkout (Service Key required).",url:null};const p=V.createClient(a,u);if(!r||r.length===0)return{error:"Cart is empty",url:null};if(r.length!==1)return{error:"Freemius items must be checked out one at a time.",url:null};const{data:g,error:K}=await p.from("currencies").select("code, symbol, exchange_rate, is_default, is_active, auto_sync_product_prices, auto_update_exchange_rate, exchange_rate_source, exchange_rate_updated_at, rounding_mode, rounding_increment, rounding_charm_amount").eq("is_active",!0).order("code",{ascending:!0}),$=g??[];if(K||$.length===0)return{error:"Failed to resolve store currencies",url:null};const X=j.getDefaultCurrency($),U=$.find(l=>l.code===(m||"").toUpperCase())??X,S=r[0],{data:o,error:M}=await p.from("products").select("id, title, price, prices, sale_price, sale_prices, sale_start_at, sale_end_at, scheduled_price, scheduled_prices, scheduled_price_at, freemius_plan_id, freemius_product_id, trial_period_days, trial_requires_payment_method").eq("id",S.product_id).single();if(M||!o)return{error:"Product not found",url:null};const R=o.freemius_plan_id,P=o.freemius_product_id;if(!R||!P)return{error:"Product is not configured for Freemius checkout (missing Plan ID or Product ID)",url:null};const D=j.resolveEffectivePriceForCurrency({prices:o.prices||{},salePrices:o.sale_prices||{},fallbackPrice:o.price,fallbackSalePrice:o.sale_price,saleStartAt:o.sale_start_at,saleEndAt:o.sale_end_at,scheduledPrice:o.scheduled_price,scheduledPrices:o.scheduled_prices||{},scheduledPriceAt:o.scheduled_price_at,currencyCode:U.code,currencies:$}),q=D.sale_price??D.price,N=j.isSaleWindowActive({saleStartAt:o.sale_start_at,saleEndAt:o.sale_end_at}),y=q*S.quantity;let t=null,_=0;if(b){const l=await Q.getCouponQuote({client:p,code:b,items:E&&E.length>0?E:r,currencyCode:U.code});if(!l.success)return{error:l.error,errorKey:l.errorKey,errorStatus:400,url:null};t=l.quote,_=Math.min(y,t.lineDiscounts.filter(v=>v.product_id===o.id).reduce((v,L)=>v+L.discount,0))}const F=ee(o.trial_period_days),w=F>0?S.trial_preference?S.trial_preference:o.trial_requires_payment_method?"paid":"free":null,O="pending",h=ae(d?.recipient_name??null),{data:Y,error:J}=await p.from("orders").insert({status:O,total:Math.max(0,y-_),currency:U.code,exchange_rate_at_purchase:U.exchange_rate,subtotal:y,discount_total:_,coupon_id:t?.couponId??null,coupon_code:t?.code??null,discount_details:t?{code:t.code,discount_type:t.discountType,discount_amount:t.discountAmount,provider:"freemius",provider_discounts:t.providerDiscounts,line_discounts:t.lineDiscounts,final_amount_owned_by:"freemius"}:null,provider:"freemius",freemius_product_id:String(P),freemius_plan_id:String(R),user_id:c||null,customer_details:ce.normalizeOrderCustomerDetails({email:n,phone:s,name:d?.recipient_name,billing:d,shipping:f})}).select("id").single();if(J||!Y)return console.error("Failed to create pending order:",J),{error:"Failed to initiate order",url:null};const{error:W}=await p.from("order_items").insert([{order_id:Y.id,product_id:o.id,quantity:S.quantity,price_at_purchase:q}]);if(W&&console.error("Failed to insert order items:",W),c)try{await G.upsertDefaultUserAddresses({userId:c,billingAddress:d,shippingAddress:f,client:p}),await G.fillMissingUserProfileCheckoutDetails({userId:c,fullName:d?.recipient_name??f?.recipient_name??null,phone:s,client:p})}catch(l){console.error("Failed to sync checkout profile defaults before checkout:",l)}t&&await Q.recordCouponRedemption({client:p,quote:t,orderId:Y.id,provider:"freemius",discountTotal:_,userId:c,customerEmail:n,metadata:{currency:U.code,subtotal:y,final_amount_owned_by:"freemius"}});const B=process.env.FREEMIUS_SANDBOX_ENABLED==="true",A=ne(P),I=A.publicKey,x=A.secretKey,z=A.apiKey;if(!I||B&&!x)return{error:"Missing FREEMIUS credentials (PUBLIC_KEY or SECRET_KEY) in environment variables.",url:null};if(B&&A.source==="legacy-env"){const l=i("FREEMIUS_PRODUCT_ID"),v=!!i("FREEMIUS_ECOMMERCE_SANDBOX_PUBLIC_KEY"),L=!!i("FREEMIUS_ECOMMERCE_SANDBOX_SECRET_KEY");console.warn(`[Freemius Checkout] Sandbox is enabled for product ${P}, but no product-scoped checkout credentials were selected. Falling back to legacy FREEMIUS_PUBLIC_KEY/FREEMIUS_SECRET_KEY may open live checkout instead of sandbox.`,{configuredProductId:l,productIdsMatch:l===String(P),hasSandboxOverridePublicKey:v,hasSandboxOverrideSecretKey:L,hasCheckoutProductsJson:!!i("FREEMIUS_CHECKOUT_PRODUCTS_JSON")})}let T=!1;if(B&&x&&I)try{T=await _e({productId:P,publicKey:I,secretKey:x,apiKey:z})}catch(l){console.warn("Freemius Checkout - SDK sandbox generation failed. Falling back to manual token generation.",l,{credentialSource:A.source,hasApiKey:!!z});const v=Math.floor(Date.now()/1e3).toString(),L=`${v}${P}${x}${I}checkout`,ie=Z.createHash("md5").update(L).digest("hex");T={ctx:v,token:ie}}const C=new URL(`https://checkout.freemius.com/app/${P}/plan/${R}/`);if(B&&x&&I?(C.searchParams.append("sandbox",T.token),C.searchParams.append("s_ctx_ts",T.ctx)):B&&C.searchParams.append("sandbox","true"),n&&C.searchParams.append("user_email",n),h.firstName&&C.searchParams.append("user_firstname",h.firstName),h.lastName&&C.searchParams.append("user_lastname",h.lastName),C.searchParams.append("currency",U.code.toLowerCase()),S.billing_cycle&&C.searchParams.append("billing_cycle",S.billing_cycle),w&&C.searchParams.append("trial",w),t)C.searchParams.append("coupon",t.code);else if(N){const{data:l}=await p.from("product_freemius_sale_coupons").select("freemius_coupon_code, is_active, starts_at, ends_at, sync_status").eq("product_id",o.id).maybeSingle();l?.is_active&&l.sync_status==="synced"&&l.freemius_coupon_code&&j.isSaleWindowActive({saleStartAt:l.starts_at,saleEndAt:l.ends_at})&&C.searchParams.append("coupon",l.freemius_coupon_code)}return{url:C.toString(),customProps:{provider:"freemius",plugin_id:P,plan_id:R,public_key:I,user_email:n,user_firstname:h.firstName,user_lastname:h.lastName,credential_source:A.source,sandbox:T,billing_cycle:S.billing_cycle,trial:w,trial_period_days:F,trial_requires_payment_method:o.trial_requires_payment_method,initial_order_status:O,coupon:t?.code??null,order_id:Y.id}}}}async function te(e,r,n,s){const d=new Date().toUTCString().replace("GMT","+0000"),f=`GET
2
2
 
3
3
 
4
4
  ${d}
5
- ${e}`,m=Z.createHmac("sha256",s).update(f).digest("hex"),b=Buffer.from(m).toString("base64").replace(/\+/g,"-").replace(/\//g,"_").replace(/=/g,""),g=`FS ${r}:${t}:${b}`,a=await fetch(`https://api.freemius.com${e}`,{headers:{Authorization:g,Date:d,Accept:"application/json"}});if(!a.ok){const u=await a.text();throw console.error(`[Freemius API] [ERROR] ${e} returned ${a.status}: ${u}`),new Error(`Freemius API failed on ${e}: ${a.status} - ${u}`)}return a.json()}async function fe(){const e=i("FREEMIUS_DEVELOPER_ID"),r=i("FREEMIUS_PUBLIC_KEY"),t=i("FREEMIUS_SECRET_KEY"),s=process.env.NEXT_PUBLIC_SUPABASE_URL,c=process.env.SUPABASE_SERVICE_ROLE_KEY;if(!e||!r||!t||!s||!c)throw new Error("Missing necessary environment variables for Freemius Sync.");const d=V.createClient(s,c,{auth:{autoRefreshToken:!1,persistSession:!1}}),f=m=>ne(m,e,r,t);try{console.log(`[Freemius Sync] Fetching all plugins for developer ${e}...`);const m=await f(`/v1/developers/${e}/plugins.json`),b=H(m,["plugins","plugin"]);console.log(`[Freemius Sync] Found ${b.length} plugins. Syncing plans...`);let g=0;const{data:a}=await d.from("languages").select("id").eq("code","en").single(),u=a?.id;if(!u)throw new Error("English language not found in database. Cannot sync products.");for(const p of b){const E=p.id?.toString();if(!E){console.warn("[Freemius Sync] Skipping plugin without an id:",p);continue}const w=await se(d,e,E,p.title||p.name||`Freemius Product ${E}`,f,u);g+=w}return{success:!0,count:g}}catch(m){throw console.error("[Freemius Sync] Global Error:",m),m}}async function ge(e){const r=i("FREEMIUS_DEVELOPER_ID"),t=i("FREEMIUS_PUBLIC_KEY"),s=i("FREEMIUS_SECRET_KEY"),c=process.env.NEXT_PUBLIC_SUPABASE_URL,d=process.env.SUPABASE_SERVICE_ROLE_KEY;if(!r||!t||!s||!c||!d)throw new Error("Missing environment variables for Freemius Sync.");const f=V.createClient(c,d),m=E=>ne(E,r,t,s),{data:b}=await f.from("languages").select("id").eq("code","en").single(),g=b?.id;if(!g)throw new Error("English language not found in database. Cannot sync products.");const a=await m(`/v1/developers/${r}/plugins/${e}.json`),u=a.plugin??a;return{success:!0,count:await se(f,r,e,u.title||u.name||`Freemius Product ${e}`,m,g)}}async function se(e,r,t,s,c,d){console.log(`[Freemius Sync] Fetching plans for plugin: ${s} (${t})...`);let f=0;try{const m=`/v1/developers/${r}/plugins/${t}/plans.json`,b=await c(m),g=H(b,["plans","plan"]);console.log(`[Freemius Sync] Received ${g.length} plans for plugin ${t}.`);for(const a of g){const u=a.id?.toString();if(!u){console.warn("[Freemius Sync] Skipping plan without an id:",a);continue}const p=a.name||a.title||u,E=a.title||p;console.log(`[Freemius Sync] Processing plan: ${E} (${u})...`);let w=a;if(w.trial_period===void 0||w.is_require_subscription===void 0)try{const n=`/v1/developers/${r}/plugins/${t}/plans/${u}.json`,_=await c(n),P=_.plan??_;w={...w,...P}}catch(n){console.warn(`[Freemius Sync] Could not fetch trial details for plan ${u}:`,n instanceof Error?n.message:n)}const v=ee(w.trial_period),X=v>0&&ue(w.is_require_subscription);let K=0,y=[];try{const n=`/v1/developers/${r}/plugins/${t}/plans/${u}/pricing.json`,_=await c(n);if(y=H(_,["pricing","prices","pricings"]),y.length>0){const P=y[0],R=A(P.annual_price)??A(P.monthly_price)??A(P.lifetime_price);K=de(R)}console.log(`[Freemius Sync] Plan: ${E} -> Resolved Price (cents): ${K}`)}catch(n){console.warn(`[Freemius Sync] Could not fetch pricing for plan ${u}:`,n instanceof Error?n.message:n)}const o=`${s}-${E}`.toLowerCase().replace(/[^\w\s-]/g,"").replace(/[\s_]+/g,"-").replace(/^-+|-+$/g,""),M={title:`${s} - ${E}`,slug:o,short_description:a.description||"",price:K,product_type:"digital",payment_provider:"freemius",freemius_plan_id:u,freemius_product_id:t,trial_period_days:v,trial_requires_payment_method:X,status:"active",stock:999,sku:`FM-${t}-${u}`,language_id:d},{data:U,error:F}=await e.from("products").upsert(M,{onConflict:"language_id, sku"}).select();if(F||!U||U.length===0){console.error(`[Freemius Sync] Error upserting product ${M.sku}:`,F);continue}const D=U[0].id,{data:q,error:N}=await e.from("freemius_plans").select("id").eq("product_id",D).eq("name",p).maybeSingle();N&&console.warn(`[Freemius Sync] Could not check existing local plan for ${M.sku}:`,N.message||N);let S="";if(q){S=q.id;const{error:n}=await e.from("freemius_plans").update({title:E,updated_at:new Date().toISOString()}).eq("id",S);n&&console.warn(`[Freemius Sync] Could not update local plan ${S}:`,n.message||n)}else{const{data:n,error:_}=await e.from("freemius_plans").insert({product_id:D,name:p,title:E}).select("id").single();_&&console.warn(`[Freemius Sync] Could not insert local plan for ${M.sku}:`,_.message||_),n&&(S=n.id)}if(S&&y.length>0)for(const n of y){const _=pe(n.licenses??n.license_quota??n.quota),{data:P,error:R}=await e.from("freemius_pricing").select("id").eq("plan_id",S).eq("license_quota",_).maybeSingle();R&&console.warn(`[Freemius Sync] Could not check pricing for plan ${S}, quota ${_}:`,R.message||R);const O={api_monthly_price:A(n.monthly_price),api_annual_price:A(n.annual_price),api_lifetime_price:A(n.lifetime_price),updated_at:new Date().toISOString()};if(P){const{error:h}=await e.from("freemius_pricing").update(O).eq("id",P.id);h&&console.warn(`[Freemius Sync] Could not update pricing ${P.id}:`,h.message||h)}else{const{error:h}=await e.from("freemius_pricing").insert({plan_id:S,license_quota:_,...O});h&&console.warn(`[Freemius Sync] Could not insert pricing for plan ${S}, quota ${_}:`,h.message||h)}}console.log(`[Freemius Sync] Successfully fully synced product ${M.sku}.`),f++}}catch(m){console.error(`[Freemius Sync] Failed sync for plugin ${t}:`,m.message)}return f}exports.FreemiusProvider=me;exports.parseFreemiusCheckoutCredentialsMap=re;exports.readFreemiusEnvValue=i;exports.resolveFreemiusCheckoutCredentials=te;exports.syncFreemiusProductsToSupabase=fe;exports.syncSingleFreemiusProduct=ge;
5
+ ${e}`,m=Z.createHmac("sha256",s).update(f).digest("hex"),b=Buffer.from(m).toString("base64").replace(/\+/g,"-").replace(/\//g,"_").replace(/=/g,""),E=`FS ${r}:${n}:${b}`,a=await fetch(`https://api.freemius.com${e}`,{headers:{Authorization:E,Date:d,Accept:"application/json"}});if(!a.ok){const u=await a.text();throw console.error(`[Freemius API] [ERROR] ${e} returned ${a.status}: ${u}`),new Error(`Freemius API failed on ${e}: ${a.status} - ${u}`)}return a.json()}async function fe(){const e=i("FREEMIUS_DEVELOPER_ID"),r=i("FREEMIUS_PUBLIC_KEY"),n=i("FREEMIUS_SECRET_KEY"),s=process.env.NEXT_PUBLIC_SUPABASE_URL||process.env.SUPABASE_URL,c=process.env.SUPABASE_SERVICE_ROLE_KEY||process.env.SUPABASE_SECRET_KEY;if(!e||!r||!n||!s||!c)throw new Error("Missing necessary environment variables for Freemius Sync.");const d=V.createClient(s,c,{auth:{autoRefreshToken:!1,persistSession:!1}}),f=m=>te(m,e,r,n);try{console.log(`[Freemius Sync] Fetching all plugins for developer ${e}...`);const m=await f(`/v1/developers/${e}/plugins.json`),b=H(m,["plugins","plugin"]);console.log(`[Freemius Sync] Found ${b.length} plugins. Syncing plans...`);let E=0;const{data:a}=await d.from("languages").select("id").eq("code","en").single(),u=a?.id;if(!u)throw new Error("English language not found in database. Cannot sync products.");for(const p of b){const g=p.id?.toString();if(!g){console.warn("[Freemius Sync] Skipping plugin without an id:",p);continue}const K=await se(d,e,g,p.title||p.name||`Freemius Product ${g}`,f,u);E+=K}return{success:!0,count:E}}catch(m){throw console.error("[Freemius Sync] Global Error:",m),m}}async function Ee(e){const r=i("FREEMIUS_DEVELOPER_ID"),n=i("FREEMIUS_PUBLIC_KEY"),s=i("FREEMIUS_SECRET_KEY"),c=process.env.NEXT_PUBLIC_SUPABASE_URL||process.env.SUPABASE_URL,d=process.env.SUPABASE_SERVICE_ROLE_KEY||process.env.SUPABASE_SECRET_KEY;if(!r||!n||!s||!c||!d)throw new Error("Missing environment variables for Freemius Sync.");const f=V.createClient(c,d),m=g=>te(g,r,n,s),{data:b}=await f.from("languages").select("id").eq("code","en").single(),E=b?.id;if(!E)throw new Error("English language not found in database. Cannot sync products.");const a=await m(`/v1/developers/${r}/plugins/${e}.json`),u=a.plugin??a;return{success:!0,count:await se(f,r,e,u.title||u.name||`Freemius Product ${e}`,m,E)}}async function se(e,r,n,s,c,d){console.log(`[Freemius Sync] Fetching plans for plugin: ${s} (${n})...`);let f=0;try{const m=`/v1/developers/${r}/plugins/${n}/plans.json`,b=await c(m),E=H(b,["plans","plan"]);console.log(`[Freemius Sync] Received ${E.length} plans for plugin ${n}.`);for(const a of E){const u=a.id?.toString();if(!u){console.warn("[Freemius Sync] Skipping plan without an id:",a);continue}const p=a.name||a.title||u,g=a.title||p;console.log(`[Freemius Sync] Processing plan: ${g} (${u})...`);let K=a;if(K.trial_period===void 0||K.is_require_subscription===void 0)try{const t=`/v1/developers/${r}/plugins/${n}/plans/${u}.json`,_=await c(t),F=_.plan??_;K={...K,...F}}catch(t){console.warn(`[Freemius Sync] Could not fetch trial details for plan ${u}:`,t instanceof Error?t.message:t)}const $=ee(K.trial_period),X=$>0&&ue(K.is_require_subscription);let U=0,S=[];try{const t=`/v1/developers/${r}/plugins/${n}/plans/${u}/pricing.json`,_=await c(t);if(S=H(_,["pricing","prices","pricings"]),S.length>0){const F=S[0],w=k(F.annual_price)??k(F.monthly_price)??k(F.lifetime_price);U=de(w)}console.log(`[Freemius Sync] Plan: ${g} -> Resolved Price (cents): ${U}`)}catch(t){console.warn(`[Freemius Sync] Could not fetch pricing for plan ${u}:`,t instanceof Error?t.message:t)}const o=`${s}-${g}`.toLowerCase().replace(/[^\w\s-]/g,"").replace(/[\s_]+/g,"-").replace(/^-+|-+$/g,""),M={title:`${s} - ${g}`,slug:o,short_description:a.description||"",price:U,product_type:"digital",payment_provider:"freemius",freemius_plan_id:u,freemius_product_id:n,trial_period_days:$,trial_requires_payment_method:X,status:"active",stock:999,sku:`FM-${n}-${u}`,language_id:d},{data:R,error:P}=await e.from("products").upsert(M,{onConflict:"language_id, sku"}).select();if(P||!R||R.length===0){console.error(`[Freemius Sync] Error upserting product ${M.sku}:`,P);continue}const D=R[0].id,{data:q,error:N}=await e.from("freemius_plans").select("id").eq("product_id",D).eq("name",p).maybeSingle();N&&console.warn(`[Freemius Sync] Could not check existing local plan for ${M.sku}:`,N.message||N);let y="";if(q){y=q.id;const{error:t}=await e.from("freemius_plans").update({title:g,updated_at:new Date().toISOString()}).eq("id",y);t&&console.warn(`[Freemius Sync] Could not update local plan ${y}:`,t.message||t)}else{const{data:t,error:_}=await e.from("freemius_plans").insert({product_id:D,name:p,title:g}).select("id").single();_&&console.warn(`[Freemius Sync] Could not insert local plan for ${M.sku}:`,_.message||_),t&&(y=t.id)}if(y&&S.length>0)for(const t of S){const _=pe(t.licenses??t.license_quota??t.quota),{data:F,error:w}=await e.from("freemius_pricing").select("id").eq("plan_id",y).eq("license_quota",_).maybeSingle();w&&console.warn(`[Freemius Sync] Could not check pricing for plan ${y}, quota ${_}:`,w.message||w);const O={api_monthly_price:k(t.monthly_price),api_annual_price:k(t.annual_price),api_lifetime_price:k(t.lifetime_price),updated_at:new Date().toISOString()};if(F){const{error:h}=await e.from("freemius_pricing").update(O).eq("id",F.id);h&&console.warn(`[Freemius Sync] Could not update pricing ${F.id}:`,h.message||h)}else{const{error:h}=await e.from("freemius_pricing").insert({plan_id:y,license_quota:_,...O});h&&console.warn(`[Freemius Sync] Could not insert pricing for plan ${y}, quota ${_}:`,h.message||h)}}console.log(`[Freemius Sync] Successfully fully synced product ${M.sku}.`),f++}}catch(m){console.error(`[Freemius Sync] Failed sync for plugin ${n}:`,m.message)}return f}exports.FreemiusProvider=me;exports.parseFreemiusCheckoutCredentialsMap=re;exports.readFreemiusEnvValue=i;exports.resolveFreemiusCheckoutCredentials=ne;exports.syncFreemiusProductsToSupabase=fe;exports.syncSingleFreemiusProduct=Ee;