@timeback/sdk 0.2.4-beta.20260401223329 → 0.2.4-beta.20260403024308

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.
@@ -0,0 +1,2 @@
1
+ import{R as E,U as O}from"./chunk-1s709w5e.js";var H=null,Y=!1,k=!1;function B(){try{let z=process.env.DEBUG?.trim();if(!z)return!1;return z.split(",").some((Q)=>Q.trim().length>0)}catch{return!1}}function D(z){let Z=z.replace(/[.+?^${}()|[\]\\]/g,"\\$&").replace(/\*/g,".*");return new RegExp(`^${Z}$`)}function m(){if(H!==null)return;if(H=[],!B()){k=!1;return}k=!0;let z=process.env.DEBUG.trim();if(z==="1"||z==="true"||z==="*"){Y=!0;return}let Q=z.split(",").map(($)=>$.trim()).filter(Boolean);for(let $ of Q)if($.startsWith("-"))H.push({regex:D($.slice(1)),exclude:!0});else H.push({regex:D($),exclude:!1});if(!H.some(($)=>!$.exclude)&&H.length>0)Y=!0}function I(z){if(m(),!k)return!0;if(Y){if(z){for(let Q of H)if(Q.exclude&&Q.regex.test(z))return!1}return!0}if(!z)return!1;for(let Q of H)if(Q.exclude&&Q.regex.test(z))return!1;for(let Q of H)if(!Q.exclude&&Q.regex.test(z))return!0;return!1}function R(){return typeof globalThis<"u"&&"window"in globalThis}function A(){if(R())return"browser";if(typeof process<"u"&&process.env){if(process.env.CI||process.env.GITHUB_ACTIONS||process.env.GITLAB_CI||process.env.CIRCLECI||process.env.JENKINS_URL||process.env.BUILDKITE)return"ci"}return"terminal"}var F;if(!R())try{F=(await import("./chunk-zpbvsyr1.js")).inspect}catch{}var W={reset:"\x1B[0m",bold:"\x1B[1m",dim:"\x1B[2m",red:"\x1B[31m",yellow:"\x1B[33m",blue:"\x1B[34m",cyan:"\x1B[36m"};function d(z){switch(z){case"debug":return W.blue;case"info":return W.cyan;case"warn":return W.yellow;case"error":return W.red}}function v(z){switch(z){case"debug":return console.debug;case"info":return console.info;case"warn":return console.warn;case"error":return console.error}}function y(z){if(F)return F(z,{depth:null,colors:!0,breakLength:80,compact:!1});return JSON.stringify(z,null,2)}var J=(z)=>{let Q=d(z.level),Z=v(z.level),$=z.level.toUpperCase().padEnd(5),q=z.timestamp.toISOString().replace(/\.\d{3}Z$/,""),f=`${W.dim}[${q}]${W.reset}`,b=`${Q}${$}${W.reset}`,L=z.scope?`${W.bold}[${z.scope}]${W.reset} `:"",C=`${f} ${b} ${L}${z.message}`;if(z.context&&Object.keys(z.context).length>0)Z(C,y(z.context));else Z(C)};var i={debug:"[DEBUG]",info:"[INFO]",warn:"[WARN]",error:"[ERROR]"};function u(z){return Object.entries(z).map(([Q,Z])=>`${Q}=${c(Z)}`).join(" ")}function c(z){if(typeof z==="string")return z;if(typeof z==="number")return String(z);if(typeof z==="boolean")return String(z);if(z===null)return"null";if(z===void 0)return"undefined";return JSON.stringify(z)}var P=(z)=>{let Q=[];if(Q.push(z.timestamp.toISOString()),Q.push(i[z.level]),z.scope)Q.push(`[${z.scope}]`);if(Q.push(z.message),z.context&&Object.keys(z.context).length>0)Q.push(u(z.context));console.log(Q.join(" "))};var K=(z)=>{let Q={timestamp:z.timestamp.toISOString(),level:z.level,...z.scope&&{scope:z.scope},msg:z.message};if(z.context&&Object.keys(z.context).length>0)Object.assign(Q,z.context);console.log(JSON.stringify(Q))};var p={debug:"color: gray",info:"color: #0ea5e9",warn:"color: #f59e0b",error:"color: #ef4444; font-weight: bold"},g={debug:"log",info:"info",warn:"warn",error:"error"},U=(z)=>{let Q=g[z.level],Z=p[z.level],q=`%c${z.scope?`[${z.scope}]`:""} ${z.message}`;if(z.context&&Object.keys(z.context).length>0)console[Q](q,Z,z.context);else console[Q](q,Z)};var G=["debug","info","warn","error"],S=Symbol.for("@timeback/internal-logger/config");function l(){try{return process.env.TIMEBACK_INTERNAL_LOGGER==="silent"}catch{return!1}}function a(){let z=globalThis;return z[S]??={},z[S]}function n(z){switch(z){case"terminal":return J;case"ci":return P;case"production":return K;case"browser":return U;case"test":return()=>{}}}function o(){if(B())return"debug";return"info"}function r(z,Q){return Q??a().formatter??(l()?()=>{}:n(z))}function s(z,Q){return G.indexOf(z)>=G.indexOf(Q)}class X{scope;minLevel;environment;explicitFormatter;defaultContext;constructor(z={}){this.scope=z.scope,this.minLevel=z.minLevel??o(),this.defaultContext=z.defaultContext??{},this.environment=z.environment??A(),this.explicitFormatter=z.formatter}child(z){let Q=this.scope?`${this.scope}:${z}`:z;return new X({scope:Q,minLevel:this.minLevel,environment:this.environment,defaultContext:{...this.defaultContext},formatter:this.explicitFormatter})}withContext(z){return new X({scope:this.scope,minLevel:this.minLevel,environment:this.environment,defaultContext:{...this.defaultContext,...z},formatter:this.explicitFormatter})}debug(z,Q){this.log("debug",z,Q)}info(z,Q){this.log("info",z,Q)}warn(z,Q){this.log("warn",z,Q)}error(z,Q){this.log("error",z,Q)}log(z,Q,Z){if(z==="debug"&&!I(this.scope))return;if(!s(z,this.minLevel))return;let $={level:z,message:Q,scope:this.scope,context:Z||Object.keys(this.defaultContext).length>0?{...this.defaultContext,...Z}:void 0,timestamp:new Date};r(this.environment,this.explicitFormatter)($)}}function T(z={}){return new X(z)}var w,_,t={debug(){},info(){},warn(){},error(){}};function e(){try{let z=typeof process>"u"?void 0:process.env.DEBUG;return z==="1"||z==="true"}catch{return!1}}var N=new Map;function zz(z,Q){let Z=`[${Q}]`;return{debug:(...$)=>z.debug(`${Z} ${$[0]}`,...$.slice(1)),info:(...$)=>z.info(`${Z} ${$[0]}`,...$.slice(1)),warn:(...$)=>z.warn(`${Z} ${$[0]}`,...$.slice(1)),error:(...$)=>z.error(`${Z} ${$[0]}`,...$.slice(1))}}function M(z){let Q=N.get(z);if(Q)return Q;if(w)Q=zz(w,z);else{let Z=_??(e()?"debug":"warn");if(Z==="silent")Q=t;else Q=T({scope:z,minLevel:Z})}return N.set(z,Q),Q}function Iz(z){w=z.logger,_=z.logLevel,N.clear()}function h(z){let Q=`timeback:${z}`;return{debug(Z,...$){M(Q).debug(Z,...$)},info(Z,...$){M(Q).info(Z,...$)},warn(Z,...$){M(Q).warn(Z,...$)},error(Z,...$){M(Q).error(Z,...$)}}}var Ez=h("sso"),j=h("oidc");var x=new Map;async function V(z){let Q=x.get(z);if(Q)return Q;let Z=`${z}/.well-known/openid-configuration`,$=await fetch(Z);if(!$.ok)throw j.error("Discovery fetch failed",{status:$.status}),Error(`Failed to fetch OIDC discovery: ${$.statusText}`);let q=await $.json();return j.debug("Fetched OIDC discovery document",{authEndpoint:q.authorization_endpoint,tokenEndpoint:q.token_endpoint}),x.set(z,q),q}function Sz(z){switch(z){case"production":return"https://cognito-idp.us-east-1.amazonaws.com/us-east-1_3uhuoRM3R";case"staging":return"https://cognito-idp.us-east-1.amazonaws.com/us-east-1_5EUwTP9XD";case"local":throw Error("Local environment is not yet supported for OIDC")}}async function _z(z){let Q=await V(z.issuer),Z=new URL(Q.authorization_endpoint);return Z.searchParams.set("response_type","code"),Z.searchParams.set("client_id",z.clientId),Z.searchParams.set("redirect_uri",z.redirectUri),Z.searchParams.set("scope","openid profile email"),Z.searchParams.set("state",z.state),Z.toString()}async function hz(z){let Q=await V(z.issuer),Z=await fetch(Q.token_endpoint,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:new URLSearchParams({grant_type:"authorization_code",client_id:z.clientId,client_secret:z.clientSecret,code:z.code,redirect_uri:z.redirectUri})});if(!Z.ok){let q=await Z.text();throw j.error("Token exchange failed",{status:Z.status,body:q}),Error(`Token exchange failed: ${Z.status} ${q}`)}let $=await Z.json();return j.debug("Received tokens from IdP",{expiresIn:$.expires_in}),$}async function xz(z){let Q=await V(z.issuer),Z=await fetch(Q.userinfo_endpoint,{headers:{Authorization:`Bearer ${z.accessToken}`}});if(!Z.ok)throw Error(`UserInfo request failed: ${Z.statusText}`);return Z.json()}
2
+ export{Iz as t,h as u,Ez as v,Sz as w,_z as x,hz as y,xz as z};
@@ -1 +1 @@
1
- import{u as B}from"./chunk-nw6brg34.js";import{J as Z}from"./chunk-dbcsf5tk.js";import{TimebackClient as O}from"@timeback/core";function P(...x){return x.map((G)=>G?.trim()).filter((G)=>Boolean(G)).join(" ")||void 0}function X(x){let $=x.lastIndexOf("@");if($<=0)return"<redacted>";let G=x.slice(0,$),K=x.slice($+1);return`${G.slice(0,1)}***@${K}`}var Y=B("resolve");class V extends Error{code;constructor(x,$){super(x);this.code=$;this.name="TimebackUserResolutionError"}}function z(x){switch(x.code){case"timeback_user_ambiguous":return 409;case"timeback_user_not_found":return 404;default:return 502}}function F(x){return{sub:x.sub,email:x.email,firstName:x.given_name,lastName:x.family_name,pictureUrl:x.picture}}async function D(x,$){let K=(await x.oneroster.users.list({where:{email:$},limit:2})).data;if(K.length===0)throw Y.warn("No Timeback user found for email",{email:X($)}),new V(`No Timeback user found with email: ${$}`,"timeback_user_not_found");if(K.length>1)throw Y.error("Multiple Timeback users found for email",{email:X($),count:K.length}),new V(`Multiple Timeback users found with email: ${$}`,"timeback_user_ambiguous");let Q=K[0];if(!Q.sourcedId)throw new V("Timeback user is missing sourcedId","timeback_user_lookup_failed");return Q}async function W(x){let{env:$,apiCredentials:G,userInfo:K}=x;if(!K.email)throw Y.error("Missing email in IdP user info",{sub:K.sub}),new V("IdP did not return an email address, which is required to resolve Timeback identity","missing_email");let Q=K.email;Y.debug("Resolving Timeback user by email",{email:X(Q),env:$});let J=x.client,_=J??new O({env:Z($),auth:{clientId:G.clientId,clientSecret:G.clientSecret}});try{let H=await D(_,Q);return{...{id:H.sourcedId,email:H.email??Q,name:P(H.givenName,H.familyName),school:H.primaryOrg?{id:H.primaryOrg.sourcedId,name:H.primaryOrg.name??H.primaryOrg.sourcedId}:void 0,grade:H.grades?.length?Math.max(...H.grades):void 0},claims:F(K)}}catch(H){if(H instanceof V)throw H;let M=H instanceof Error?H.message:"Unknown error";throw Y.error("Failed to lookup Timeback user",{email:X(Q),error:M}),new V(`Failed to lookup Timeback user: ${M}`,"timeback_user_lookup_failed")}finally{if(!J)_.close()}}async function A(x){let{email:$,client:G}=x;Y.debug("Looking up Timeback user ID by email",{email:X($)});try{return(await D(G,$)).sourcedId}catch(K){if(K instanceof V)throw K;let Q=K instanceof Error?K.message:"Unknown error";throw Y.error("Failed to lookup Timeback user",{email:X($),error:Q}),new V(`Failed to lookup Timeback user: ${Q}`,"timeback_user_lookup_failed")}}function L(x,$){let G=x.totalQuestions!==void 0,K=x.correctQuestions!==void 0;if(G!==K){$.addIssue({code:"custom",message:"totalQuestions and correctQuestions must be provided together."});return}if(G&&K){let{correctQuestions:Q,totalQuestions:J}=x;if(Q>J)$.addIssue({code:"custom",message:"correctQuestions cannot exceed totalQuestions."})}}function N(x,$,G){let K=Z(G),Q=$.overrides?.[K]?.sensor??$.sensor??x.sensor;if(!Q){let J=$.overrides?.[K]?.launchUrl??$.launchUrl??x.launchUrl;if(J)try{Q=new URL(J).origin}catch{}}return Q}function j(x,$,G){if(!$?.onRequestStart&&!$?.onRequestEnd)return G;return async(K)=>{await $.onRequestStart?.({handler:x});let Q=Date.now(),J;try{J=await G(K)}catch(_){throw await $.onRequestEnd?.({handler:x,durationMs:Date.now()-Q,status:500}),_}return await $.onRequestEnd?.({handler:x,durationMs:Date.now()-Q,status:J.status}),J}}export{V as m,z as n,W as o,A as p,L as q,N as r,j as s};
1
+ import{u as B}from"./chunk-b49f5qfw.js";import{J as Z}from"./chunk-dbcsf5tk.js";import{TimebackClient as O}from"@timeback/core";function P(...x){return x.map((G)=>G?.trim()).filter((G)=>Boolean(G)).join(" ")||void 0}function X(x){let $=x.lastIndexOf("@");if($<=0)return"<redacted>";let G=x.slice(0,$),K=x.slice($+1);return`${G.slice(0,1)}***@${K}`}var Y=B("resolve");class V extends Error{code;constructor(x,$){super(x);this.code=$;this.name="TimebackUserResolutionError"}}function z(x){switch(x.code){case"timeback_user_ambiguous":return 409;case"timeback_user_not_found":return 404;default:return 502}}function F(x){return{sub:x.sub,email:x.email,firstName:x.given_name,lastName:x.family_name,pictureUrl:x.picture}}async function D(x,$){let K=(await x.oneroster.users.list({where:{email:$},limit:2})).data;if(K.length===0)throw Y.warn("No Timeback user found for email",{email:X($)}),new V(`No Timeback user found with email: ${$}`,"timeback_user_not_found");if(K.length>1)throw Y.error("Multiple Timeback users found for email",{email:X($),count:K.length}),new V(`Multiple Timeback users found with email: ${$}`,"timeback_user_ambiguous");let Q=K[0];if(!Q.sourcedId)throw new V("Timeback user is missing sourcedId","timeback_user_lookup_failed");return Q}async function W(x){let{env:$,apiCredentials:G,userInfo:K}=x;if(!K.email)throw Y.error("Missing email in IdP user info",{sub:K.sub}),new V("IdP did not return an email address, which is required to resolve Timeback identity","missing_email");let Q=K.email;Y.debug("Resolving Timeback user by email",{email:X(Q),env:$});let J=x.client,_=J??new O({env:Z($),auth:{clientId:G.clientId,clientSecret:G.clientSecret}});try{let H=await D(_,Q);return{...{id:H.sourcedId,email:H.email??Q,name:P(H.givenName,H.familyName),school:H.primaryOrg?{id:H.primaryOrg.sourcedId,name:H.primaryOrg.name??H.primaryOrg.sourcedId}:void 0,grade:H.grades?.length?Math.max(...H.grades):void 0},claims:F(K)}}catch(H){if(H instanceof V)throw H;let M=H instanceof Error?H.message:"Unknown error";throw Y.error("Failed to lookup Timeback user",{email:X(Q),error:M}),new V(`Failed to lookup Timeback user: ${M}`,"timeback_user_lookup_failed")}finally{if(!J)_.close()}}async function A(x){let{email:$,client:G}=x;Y.debug("Looking up Timeback user ID by email",{email:X($)});try{return(await D(G,$)).sourcedId}catch(K){if(K instanceof V)throw K;let Q=K instanceof Error?K.message:"Unknown error";throw Y.error("Failed to lookup Timeback user",{email:X($),error:Q}),new V(`Failed to lookup Timeback user: ${Q}`,"timeback_user_lookup_failed")}}function L(x,$){let G=x.totalQuestions!==void 0,K=x.correctQuestions!==void 0;if(G!==K){$.addIssue({code:"custom",message:"totalQuestions and correctQuestions must be provided together."});return}if(G&&K){let{correctQuestions:Q,totalQuestions:J}=x;if(Q>J)$.addIssue({code:"custom",message:"correctQuestions cannot exceed totalQuestions."})}}function N(x,$,G){let K=Z(G),Q=$.overrides?.[K]?.sensor??$.sensor??x.sensor;if(!Q){let J=$.overrides?.[K]?.launchUrl??$.launchUrl??x.launchUrl;if(J)try{Q=new URL(J).origin}catch{}}return Q}function j(x,$,G){if(!$?.onRequestStart&&!$?.onRequestEnd)return G;return async(K)=>{await $.onRequestStart?.({handler:x});let Q=Date.now(),J;try{J=await G(K)}catch(_){throw await $.onRequestEnd?.({handler:x,durationMs:Date.now()-Q,status:500}),_}return await $.onRequestEnd?.({handler:x,durationMs:Date.now()-Q,status:J.status}),J}}export{V as m,z as n,W as o,A as p,L as q,N as r,j as s};
@@ -1,2 +1,2 @@
1
- import{v as Y,w as H,x,y as v,z as E}from"./chunk-nw6brg34.js";import{K as k,L as _,N as $,O,P as D,Q as w}from"./chunk-dbcsf5tk.js";function M(G,K,J,N){return{error:G,errorCode:K,state:J,req:N,redirect:$,json:_}}function R(G){try{return D(G)}catch{Y.warn("Failed to decode state");return}}function j(G,K,J,N,W){let Q=K.searchParams.get("error_description");Y.error("IdP returned error",{error:G,description:Q});let V=Error(Q??G);if(W)return W(M(V,G,J,N));return _({error:G},400)}function T(G,K,J){Y.error("Missing authorization code in callback");let N=Error("Missing authorization code");if(J)return J(M(N,"missing_code",G,K));return _({error:"Missing authorization code"},400)}async function L(G){let{req:K,env:J,clientId:N,buildState:W}=G,Q=G.issuer??H(J),V=new URL(K.url),X=G.redirectUri;if(!X){let A=V.pathname.replace(w.IDENTITY.SIGNIN,"");X=`${V.origin}${A}${w.IDENTITY.CALLBACK}`}Y.debug("SSO sign-in initiated",{env:J,issuer:Q,clientId:N,redirectUri:X});let F=W?W({req:K,url:V}):{},Z=O(F),z=await x({issuer:Q,clientId:N,redirectUri:X,state:Z});return $(z)}function S(G){let K=new URL(G.url),J=K.searchParams.get("code"),N=K.searchParams.get("error"),W=K.searchParams.get("state");Y.debug("Received callback from IdP",{hasCode:!!J,error:N});let Q=W?R(W):void 0;return{url:K,code:J,errorParam:N,state:Q}}function h(G,K){if(K)return K;let J=G.pathname.replace(w.IDENTITY.CALLBACK,"");return`${G.origin}${J}${w.IDENTITY.CALLBACK}`}async function P(G,K,J){return await L({req:G,env:K,clientId:J.clientId,issuer:J.issuer,redirectUri:J.redirectUri,buildState:J.buildState})}async function C(G,K,J,N,W,Q){try{let V=Q.issuer??H(W),X=h(K,Q.redirectUri);Y.debug("Exchanging auth code for tokens (identity-only)",{issuer:V,clientId:Q.clientId});let F=await v({issuer:V,clientId:Q.clientId,clientSecret:Q.clientSecret,code:G,redirectUri:X}),Z=await E({issuer:V,accessToken:F.access_token}),z=typeof Z.identities==="string"?JSON.parse(Z.identities):Z.identities;Y.debug("SSO completed (identity-only)",{user:{...Z,identities:z}});let A={tokens:F,user:Z,state:J,req:N,redirect:$,json:_};return Q.onCallbackSuccess(A)}catch(V){let X=V instanceof Error?V:Error("Unknown error");if(Y.error("Token exchange failed (identity-only)",{error:X.message}),Q.onCallbackError)return Q.onCallbackError(M(X,void 0,J,N));return _({error:X.message},500)}}async function I(G,K,J){let{url:N,code:W,errorParam:Q,state:V}=S(G);if(Q)return j(Q,N,V,G,J.onCallbackError);if(!W)return T(V,G,J.onCallbackError);return await C(W,N,V,G,K,J)}function B(G){let{env:K,identity:J}=G;if(J.mode!=="sso")throw Error('Identity-only mode requires identity.mode === "sso"');return{signIn:(N)=>P(N,K,J),callback:(N)=>I(N,K,J),signOut:()=>$("/")}}function t(G){let K=B({env:k(G.env),identity:G.identity});return{config:G,handle:{identity:K}}}
1
+ import{v as Y,w as H,x,y as v,z as E}from"./chunk-b49f5qfw.js";import{K as k,L as _,N as $,O,P as D,Q as w}from"./chunk-dbcsf5tk.js";function M(G,K,J,N){return{error:G,errorCode:K,state:J,req:N,redirect:$,json:_}}function R(G){try{return D(G)}catch{Y.warn("Failed to decode state");return}}function j(G,K,J,N,W){let Q=K.searchParams.get("error_description");Y.error("IdP returned error",{error:G,description:Q});let V=Error(Q??G);if(W)return W(M(V,G,J,N));return _({error:G},400)}function T(G,K,J){Y.error("Missing authorization code in callback");let N=Error("Missing authorization code");if(J)return J(M(N,"missing_code",G,K));return _({error:"Missing authorization code"},400)}async function L(G){let{req:K,env:J,clientId:N,buildState:W}=G,Q=G.issuer??H(J),V=new URL(K.url),X=G.redirectUri;if(!X){let A=V.pathname.replace(w.IDENTITY.SIGNIN,"");X=`${V.origin}${A}${w.IDENTITY.CALLBACK}`}Y.debug("SSO sign-in initiated",{env:J,issuer:Q,clientId:N,redirectUri:X});let F=W?W({req:K,url:V}):{},Z=O(F),z=await x({issuer:Q,clientId:N,redirectUri:X,state:Z});return $(z)}function S(G){let K=new URL(G.url),J=K.searchParams.get("code"),N=K.searchParams.get("error"),W=K.searchParams.get("state");Y.debug("Received callback from IdP",{hasCode:!!J,error:N});let Q=W?R(W):void 0;return{url:K,code:J,errorParam:N,state:Q}}function h(G,K){if(K)return K;let J=G.pathname.replace(w.IDENTITY.CALLBACK,"");return`${G.origin}${J}${w.IDENTITY.CALLBACK}`}async function P(G,K,J){return await L({req:G,env:K,clientId:J.clientId,issuer:J.issuer,redirectUri:J.redirectUri,buildState:J.buildState})}async function C(G,K,J,N,W,Q){try{let V=Q.issuer??H(W),X=h(K,Q.redirectUri);Y.debug("Exchanging auth code for tokens (identity-only)",{issuer:V,clientId:Q.clientId});let F=await v({issuer:V,clientId:Q.clientId,clientSecret:Q.clientSecret,code:G,redirectUri:X}),Z=await E({issuer:V,accessToken:F.access_token}),z=typeof Z.identities==="string"?JSON.parse(Z.identities):Z.identities;Y.debug("SSO completed (identity-only)",{user:{...Z,identities:z}});let A={tokens:F,user:Z,state:J,req:N,redirect:$,json:_};return Q.onCallbackSuccess(A)}catch(V){let X=V instanceof Error?V:Error("Unknown error");if(Y.error("Token exchange failed (identity-only)",{error:X.message}),Q.onCallbackError)return Q.onCallbackError(M(X,void 0,J,N));return _({error:X.message},500)}}async function I(G,K,J){let{url:N,code:W,errorParam:Q,state:V}=S(G);if(Q)return j(Q,N,V,G,J.onCallbackError);if(!W)return T(V,G,J.onCallbackError);return await C(W,N,V,G,K,J)}function B(G){let{env:K,identity:J}=G;if(J.mode!=="sso")throw Error('Identity-only mode requires identity.mode === "sso"');return{signIn:(N)=>P(N,K,J),callback:(N)=>I(N,K,J),signOut:()=>$("/")}}function t(G){let K=B({env:k(G.env),identity:G.identity});return{config:G,handle:{identity:K}}}
2
2
  export{M as e,j as f,T as g,L as h,S as i,h as j,t as k};
package/dist/identity.js CHANGED
@@ -1 +1 @@
1
- import{k as e}from"./chunk-6nybr8hs.js";import"./chunk-nw6brg34.js";import"./chunk-dbcsf5tk.js";import"./chunk-1s709w5e.js";export{e as createTimebackIdentity};
1
+ import{k as e}from"./chunk-qpzke7wy.js";import"./chunk-b49f5qfw.js";import"./chunk-dbcsf5tk.js";import"./chunk-1s709w5e.js";export{e as createTimebackIdentity};
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- import"./chunk-ae6bkfs5.js";import{a as E,b as F$,c as O$}from"./chunk-nrn5gjey.js";import{e as R$,f as N$,g as I$,h as b$,i as h$,j as S$,k as wJ}from"./chunk-6nybr8hs.js";import{l as H1}from"./server/adapters/native.js";import{m as d,n as F2,o as O2,p as B2,q as n2,r as S2,s as OJ}from"./chunk-bh0ewh45.js";import{t as BJ,u as R,v as j2,w as z$,y as P$,z as T$}from"./chunk-nw6brg34.js";import"./chunk-3pf3cpfk.js";import{G as g8,H as L$,I as E2,J as u,K as FJ,L as P,M,N as J8,Q as Z1}from"./chunk-dbcsf5tk.js";import{R as L8,U as j8}from"./chunk-1s709w5e.js";import{TimebackClient as e0}from"@timeback/core";import{getServiceUrlsForEnv as C0}from"@timeback/core";class s extends Error{statusCode;response;name="ApiError";constructor($,J,Z){super($);this.statusCode=J,this.response=Z}get minorCodes(){let $=this.response;if(!$?.imsx_CodeMinor?.imsx_codeMinorField)return[];return $.imsx_CodeMinor.imsx_codeMinorField.map((J)=>({field:J.imsx_codeMinorFieldName,value:J.imsx_codeMinorFieldValue}))}get details(){return this.response?.imsx_error_details??[]}}class $9 extends s{name="UnauthorizedError";constructor($="Unauthorized",J){super($,401,J)}}class J9 extends s{name="ForbiddenError";constructor($="Forbidden",J){super($,403,J)}}class Z9 extends s{name="NotFoundError";constructor($="Not Found",J){super($,404,J)}}class H9 extends s{name="ValidationError";constructor($="Validation Error",J){super($,422,J)}}class K9 extends s{name="InputValidationError";issues;constructor($,J){let Z={imsx_codeMajor:"failure",imsx_severity:"error",imsx_description:$,imsx_error_details:J.map((H)=>({path:H.path,message:H.message}))};super($,400,Z);this.issues=J}}function I2($,J){return new K9($,J)}function xJ($){if(!($ instanceof Error))return!1;return"statusCode"in $&&"response"in $}var MJ="BEYOND_AI",X9="LEARNWITH_AI",W9=MJ,LJ={staging:"https://staging-beyond-timeback-api-2-idp.auth.us-east-1.amazoncognito.com/oauth2/token",production:"https://prod-beyond-timeback-api-2-idp.auth.us-east-1.amazoncognito.com/oauth2/token"},jJ={staging:"https://api.staging.alpha-1edtech.ai",production:"https://api.alpha-1edtech.ai"},VJ={staging:"https://caliper.staging.alpha-1edtech.ai",production:"https://caliper.alpha-1edtech.ai"},kJ={staging:"https://qti.alpha-1edtech.ai/api",production:"https://qti.alpha-1edtech.ai/api"},qJ={staging:"https://platform.dev.timeback.com/auth/1.0/token",production:"https://platform.timeback.com/auth/1.0/token"},CJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},AJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},zJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},PJ={BEYOND_AI:{token:LJ,tokenScope:void 0,api:jJ,caliper:VJ,qti:kJ},LEARNWITH_AI:{token:qJ,tokenScope:"https://purl.imsglobal.org/spec/caliper/v1p2/scope/events.write",api:CJ,caliper:AJ,qti:zJ}},TJ={tsc:"tsc",nativePreview:"@typescript/native-preview"},Y1={package:TJ.nativePreview,bin:"tsgo"},e=null,V8=!1,k8=!1;function o8($){let Z=$.replace(/[.+?^${}()|[\]\\]/g,"\\$&").replace(/\*/g,".*");return new RegExp(`^${Z}$`)}function RJ(){if(e!==null)return;if(e=[],typeof process>"u"||!process.env?.DEBUG){k8=!1;return}k8=!0;let $=process.env.DEBUG.trim();if($==="1"||$==="true"||$==="*"){V8=!0;return}let J=$.split(",").map((H)=>H.trim()).filter(Boolean);for(let H of J)if(H.startsWith("-"))e.push({regex:o8(H.slice(1)),exclude:!0});else e.push({regex:o8(H),exclude:!1});if(!e.some((H)=>!H.exclude)&&e.length>0)V8=!0}function NJ($){if(RJ(),!k8)return!0;if(V8){if($){for(let J of e)if(J.exclude&&J.regex.test($))return!1}return!0}if(!$)return!1;for(let J of e)if(J.exclude&&J.regex.test($))return!1;for(let J of e)if(!J.exclude&&J.regex.test($))return!0;return!1}function Y9(){return typeof globalThis<"u"&&"window"in globalThis}function IJ(){if(Y9())return"browser";if(typeof process<"u"&&process.env){if(process.env.CI||process.env.GITHUB_ACTIONS||process.env.GITLAB_CI||process.env.CIRCLECI||process.env.JENKINS_URL||process.env.BUILDKITE)return"ci"}return"terminal"}var q8;if(!Y9())try{q8=(await import("./chunk-kkajswjx.js")).inspect}catch{}var $2={reset:"\x1B[0m",bold:"\x1B[1m",dim:"\x1B[2m",red:"\x1B[31m",yellow:"\x1B[33m",blue:"\x1B[34m",cyan:"\x1B[36m"};function bJ($){switch($){case"debug":return $2.blue;case"info":return $2.cyan;case"warn":return $2.yellow;case"error":return $2.red}}function hJ($){switch($){case"debug":return console.debug;case"info":return console.info;case"warn":return console.warn;case"error":return console.error}}function SJ($){if(q8)return q8($,{depth:null,colors:!0,breakLength:80,compact:!1});return JSON.stringify($,null,2)}var EJ=($)=>{let J=bJ($.level),Z=hJ($.level),H=$.level.toUpperCase().padEnd(5),K=$.timestamp.toISOString().replace(/\.\d{3}Z$/,""),W=`${$2.dim}[${K}]${$2.reset}`,B=`${J}${H}${$2.reset}`,F=$.scope?`${$2.bold}[${$.scope}]${$2.reset} `:"",O=`${W} ${B} ${F}${$.message}`;if($.context&&Object.keys($.context).length>0)Z(O,SJ($.context));else Z(O)},vJ={debug:"[DEBUG]",info:"[INFO]",warn:"[WARN]",error:"[ERROR]"};function yJ($){return Object.entries($).map(([J,Z])=>`${J}=${fJ(Z)}`).join(" ")}function fJ($){if(typeof $==="string")return $;if(typeof $==="number")return String($);if(typeof $==="boolean")return String($);if($===null)return"null";if($===void 0)return"undefined";return JSON.stringify($)}var gJ=($)=>{let J=[];if(J.push($.timestamp.toISOString()),J.push(vJ[$.level]),$.scope)J.push(`[${$.scope}]`);if(J.push($.message),$.context&&Object.keys($.context).length>0)J.push(yJ($.context));console.log(J.join(" "))},dJ=($)=>{let J={timestamp:$.timestamp.toISOString(),level:$.level,...$.scope&&{scope:$.scope},msg:$.message};if($.context&&Object.keys($.context).length>0)Object.assign(J,$.context);console.log(JSON.stringify(J))},mJ={debug:"color: gray",info:"color: #0ea5e9",warn:"color: #f59e0b",error:"color: #ef4444; font-weight: bold"},lJ={debug:"log",info:"info",warn:"warn",error:"error"},uJ=($)=>{let J=lJ[$.level],Z=mJ[$.level],K=`%c${$.scope?`[${$.scope}]`:""} ${$.message}`;if($.context&&Object.keys($.context).length>0)console[J](K,Z,$.context);else console[J](K,Z)},n8=["debug","info","warn","error"];function cJ($){switch($){case"terminal":return EJ;case"ci":return gJ;case"production":return dJ;case"browser":return uJ;case"test":return()=>{}}}function pJ(){if(typeof process<"u"&&process.env?.DEBUG)return"debug";return"info"}function aJ($,J){return n8.indexOf($)>=n8.indexOf(J)}class d2{scope;minLevel;environment;formatter;defaultContext;constructor($={}){this.scope=$.scope,this.minLevel=$.minLevel??pJ(),this.defaultContext=$.defaultContext??{},this.environment=$.environment??IJ(),this.formatter=cJ(this.environment)}child($){let J=this.scope?`${this.scope}:${$}`:$;return new d2({scope:J,minLevel:this.minLevel,environment:this.environment,defaultContext:{...this.defaultContext}})}withContext($){return new d2({scope:this.scope,minLevel:this.minLevel,environment:this.environment,defaultContext:{...this.defaultContext,...$}})}debug($,J){this.log("debug",$,J)}info($,J){this.log("info",$,J)}warn($,J){this.log("warn",$,J)}error($,J){this.log("error",$,J)}log($,J,Z){if($==="debug"&&!NJ(this.scope))return;if(!aJ($,this.minLevel))return;let H={level:$,message:J,scope:this.scope,context:Z||Object.keys(this.defaultContext).length>0?{...this.defaultContext,...Z}:void 0,timestamp:new Date};this.formatter(H)}}function Q9($={}){return new d2($)}function iJ(){try{let $=typeof process>"u"?void 0:process.env.DEBUG;return $==="1"||$==="true"}catch{return!1}}var k2=Q9({scope:"auth",minLevel:iJ()?"debug":"warn"});class m2{config;accessToken=null;tokenExpiry=0;pendingRequest=null;fetchFn;constructor($){this.config=$,this.fetchFn=$.fetch??globalThis.fetch.bind(globalThis)}async getToken(){if(this.accessToken&&Date.now()<this.tokenExpiry)return k2.debug("Using cached token"),this.accessToken;if(this.pendingRequest)return k2.debug("Waiting for in-flight token request"),this.pendingRequest;this.pendingRequest=this.fetchToken();try{return await this.pendingRequest}finally{this.pendingRequest=null}}async fetchToken(){k2.debug("Fetching new access token...");let{clientId:$,clientSecret:J}=this.config.credentials,Z=btoa(`${$}:${J}`),H=performance.now(),K=await this.fetchFn(this.config.tokenUrl,{method:"POST",headers:{Authorization:`Basic ${Z}`,"Content-Type":"application/x-www-form-urlencoded"},body:this.config.scope?`grant_type=client_credentials&scope=${encodeURIComponent(this.config.scope)}`:"grant_type=client_credentials"}),W=Math.round(performance.now()-H);if(!K.ok)throw k2.error(`Token request failed: ${K.status} ${K.statusText}`),Error(`Failed to obtain access token: ${K.status} ${K.statusText}`);let B=await K.json();return this.accessToken=B.access_token,this.tokenExpiry=Date.now()+(B.expires_in-60)*1000,k2.debug(`Token acquired (${W}ms, expires in ${B.expires_in}s)`),this.accessToken}invalidate(){k2.debug("Token invalidated"),this.accessToken=null,this.tokenExpiry=0}}var l2={caliper:{send:"/caliper/event",validate:"/caliper/event/validate",list:"/caliper/events",get:"/caliper/events/{id}",jobStatus:"/jobs/{id}/status"},oneroster:{rostering:"/ims/oneroster/rostering/v1p2",gradebook:"/ims/oneroster/gradebook/v1p2",resources:"/ims/oneroster/resources/v1p2"},edubridge:{base:"/edubridge"},powerpath:{base:"/powerpath"},clr:{credentials:"/ims/clr/v2p0/credentials",discovery:"/ims/clr/v2p0/discovery"},case:{base:"/ims/case/v1p1"},webhooks:{webhookList:"/webhooks/",webhookGet:"/webhooks/{id}",webhookCreate:"/webhooks/",webhookUpdate:"/webhooks/{id}",webhookDelete:"/webhooks/{id}",webhookActivate:"/webhooks/{id}/activate",webhookDeactivate:"/webhooks/{id}/deactivate",webhookFilterList:"/webhook-filters/",webhookFilterGet:"/webhook-filters/{id}",webhookFilterCreate:"/webhook-filters/",webhookFilterUpdate:"/webhook-filters/{id}",webhookFilterDelete:"/webhook-filters/{id}",webhookFiltersByWebhook:"/webhook-filters/webhook/{webhookId}"},reporting:{mcp:"/mcp/reporting",savedQueryExecute:"/reporting/saved-queries/{id}",adminGroupCheck:"/mcp/admin/users/{email}/group",adminGroupAdd:"/mcp/admin/users/{email}/group",adminGroupRemove:"/mcp/admin/users/{email}/group"}},sJ={caliper:{send:"/caliper/v1p2",validate:null,list:null,get:null,jobStatus:null},oneroster:{rostering:"/rostering/1.0",gradebook:"/gradebook/1.0",resources:"/resources/1.0"},webhooks:null,reporting:null,edubridge:null,powerpath:null,clr:null,case:{base:"/case/1.1"}},G9={BEYOND_AI:l2,LEARNWITH_AI:sJ};function oJ($){return"env"in $&&!("baseUrl"in $)&&!("services"in $)}function nJ($){return"baseUrl"in $&&!("services"in $)}function rJ($){return"services"in $}function r8($,J){let Z=$?G9[$]??l2:l2;return{caliper:J?.caliper??Z.caliper,oneroster:J?.oneroster??Z.oneroster,webhooks:J?.webhooks??Z.webhooks,reporting:J?.reporting??Z.reporting,edubridge:J?.edubridge??Z.edubridge,powerpath:J?.powerpath??Z.powerpath,clr:J?.clr??Z.clr,case:J?.case??Z.case}}class b2{platform;env;auth;timeout;_endpoints;_authUrl;_tokenScope;_pathProfiles;_tokenManagers=new Map;constructor($){if(this.timeout=$.timeout??30000,oJ($)){this.auth=$.auth;let J=$.platform??W9,Z=$.env;this.platform=J,this.env=Z;let H=PJ[J];if(!H)throw Error(`Unknown platform: ${J}`);this._authUrl=H.token[Z],this._tokenScope=H.tokenScope??void 0,this._pathProfiles=G9[J]??l2,this._endpoints={oneroster:{baseUrl:H.api[Z],authUrl:this._authUrl},edubridge:{baseUrl:H.api[Z],authUrl:this._authUrl},powerpath:{baseUrl:H.api[Z],authUrl:this._authUrl},clr:{baseUrl:H.api[Z],authUrl:this._authUrl},case:{baseUrl:H.api[Z],authUrl:this._authUrl},caliper:{baseUrl:H.caliper[Z],authUrl:this._authUrl},webhooks:{baseUrl:H.caliper[Z],authUrl:this._authUrl},reporting:{baseUrl:H.api[Z],authUrl:this._authUrl},qti:{baseUrl:H.qti[Z],authUrl:this._authUrl}}}else if(nJ($))this.auth=$.auth,this._authUrl=$.authUrl,this._pathProfiles=r8($.pathProfile,$.paths),this._endpoints={oneroster:{baseUrl:$.baseUrl,authUrl:this._authUrl},edubridge:{baseUrl:$.baseUrl,authUrl:this._authUrl},powerpath:{baseUrl:$.baseUrl,authUrl:this._authUrl},clr:{baseUrl:$.baseUrl,authUrl:this._authUrl},case:{baseUrl:$.baseUrl,authUrl:this._authUrl},caliper:{baseUrl:$.baseUrl,authUrl:this._authUrl},webhooks:{baseUrl:$.baseUrl,authUrl:this._authUrl},reporting:{baseUrl:$.baseUrl,authUrl:this._authUrl},qti:{baseUrl:$.baseUrl,authUrl:this._authUrl}};else if(rJ($)){this.auth=$.auth,this._authUrl=$.authUrl,this._pathProfiles=r8($.pathProfile,$.paths),this._endpoints={};for(let[J,Z]of Object.entries($.services))if(Z)this._endpoints[J]={baseUrl:Z,authUrl:this._authUrl}}else throw Error("Invalid provider configuration");for(let J of Object.keys(this._pathProfiles))if(this._pathProfiles[J]===null)delete this._endpoints[J]}getEndpoint($){let J=this._endpoints[$];if(!J){let Z=$;if(Z in this._pathProfiles&&this._pathProfiles[Z]===null)throw Error(`Service "${$}" is not supported on ${this.platform??"this"} platform.`);throw Error(`Service "${$}" is not configured in this provider`)}return J}hasService($){return $ in this._endpoints}getAvailableServices(){return Object.keys(this._endpoints)}getTokenUrl(){return this._authUrl}getEndpointWithPaths($){let J=this.getEndpoint($),Z=this.getServicePaths($);return{...J,paths:Z}}getPaths(){return this._pathProfiles}getServicePaths($){let J=this._pathProfiles[$];if(!J)throw Error(`Service "${$}" is not supported on ${this.platform??"this"} platform.`);return J}hasServiceSupport($){return this._pathProfiles[$]!==null}getTokenProvider($){let J=this.getEndpoint($),{authUrl:Z}=J;if(!Z)return;if(!this.auth)throw Error(`Service "${$}" requires authentication but no credentials were provided`);let H=this._tokenManagers.get(Z);if(!H)H=new m2({tokenUrl:Z,credentials:{clientId:this.auth.clientId,clientSecret:this.auth.clientSecret},scope:this._tokenScope}),this._tokenManagers.set(Z,H);return H}async checkAuth(){if(!this._authUrl||!this.auth)throw Error("No auth configured on this provider");let $=Date.now(),J=this._tokenManagers.get(this._authUrl);if(!J)J=new m2({tokenUrl:this._authUrl,credentials:{clientId:this.auth.clientId,clientSecret:this.auth.clientSecret},scope:this._tokenScope}),this._tokenManagers.set(this._authUrl,J);try{return await J.getToken(),{ok:!0,latencyMs:Date.now()-$,checks:{tokenAcquisition:!0}}}catch(Z){return{ok:!1,latencyMs:Date.now()-$,error:Z instanceof Error?Z.message:String(Z),checks:{tokenAcquisition:!1}}}}invalidateTokens(){for(let $ of this._tokenManagers.values())$.invalidate?.();this._tokenManagers.clear()}}function b($){try{if(typeof process>"u")return;if(typeof $==="string")return process.env[$];for(let J of $){let Z=process.env[J];if(Z!==void 0)return Z}return}catch{return}}function D2(){let $=b("DEBUG");return $==="1"||$==="true"}function tJ(){return Math.random().toString(16).slice(2,10)}function C8($){return Q9({scope:$,minLevel:D2()?"debug":"warn"})}var u2={defaultPlatform:W9,templates:{BEYOND_AI:{staging:{platform:"BEYOND_AI",env:"staging"},production:{platform:"BEYOND_AI",env:"production"}},LEARNWITH_AI:{staging:{platform:"LEARNWITH_AI",env:"staging"},production:{platform:"LEARNWITH_AI",env:"production"}}}};function eJ($){if(typeof $==="string")return $;if($.length===0)throw Error(`Missing env var key: ${$}`);return $[0]}function w2($){return eJ($)}function _9($){if($!=="staging"&&$!=="production")throw Error(`Invalid env "${$}": must be "staging" or "production"`);return $}function B9($,J){let Z=$?.clientId??b(J.clientId),H=$?.clientSecret??b(J.clientSecret);if(!Z)throw Error(`Missing clientId: provide in config or set ${w2(J.clientId)}`);if(!H)throw Error(`Missing clientSecret: provide in config or set ${w2(J.clientSecret)}`);return{clientId:Z,clientSecret:H}}function $5($,J,Z){if(!Z)return;return B9($,J)}function J5($){let J=b($.baseUrl),Z=b($.authUrl),H=b($.clientId),K=b($.clientSecret);if(J===void 0&&H===void 0)return`Missing env: provide in config or set ${w2($.env??$.baseUrl)}`;let W=[];if(J===void 0)W.push(w2($.env??$.baseUrl));if(J!==void 0&&Z===void 0)W.push(w2($.authUrl));if(H===void 0)W.push(w2($.clientId));if(K===void 0)W.push(w2($.clientSecret));return`Missing environment variables: ${W.join(", ")}`}var Z5={name:"transport",matches($){return"transport"in $&&!!$.transport},resolve($){return{mode:"transport",transport:$.transport}}},H5={name:"provider",matches($){return"provider"in $&&!!$.provider},resolve($){return{mode:"provider",provider:$.provider}}},K5={name:"env-config",matches($){return"env"in $},resolve($,J,Z){let H=$,K=_9(H.env),W=B9(H.auth,J),B=H.platform??Z.defaultPlatform,F=Z.templates[B];if(!F){let U=Object.keys(Z.templates).join(", ");throw Error(`Unknown platform "${B}": available platforms are ${U}`)}let O=F[K];if(!O){let U=Object.keys(F).join(", ");throw Error(`Unknown env "${K}" for platform "${B}": available environments are ${U}`)}return{mode:"provider",provider:new b2({platform:O.platform,env:O.env,auth:W,timeout:H.timeout})}}},X5={name:"explicit-config",matches($){return"baseUrl"in $},resolve($,J){let Z=$,H=Z.authUrl??Z.auth?.authUrl,K=!!H,W=$5(Z.auth,J,K);return{mode:"provider",provider:new b2({baseUrl:Z.baseUrl,authUrl:H,auth:W,timeout:Z.timeout,pathProfile:Z.pathProfile,paths:Z.paths})}}},W5={name:"env-fallback-platform",matches($,J){if(Object.keys($).length>0)return!1;let Z=J.env?b(J.env):void 0,H=b(J.clientId),K=b(J.clientSecret);return Z!==void 0&&H!==void 0&&K!==void 0},resolve($,J,Z){let H=_9(b(J.env)),K=b(J.clientId),W=b(J.clientSecret),B=Z.defaultPlatform,F=Z.templates[B]?.[H];if(!F)throw Error(`Unknown env "${H}" for platform "${B}"`);return{mode:"provider",provider:new b2({platform:F.platform,env:F.env,auth:{clientId:K,clientSecret:W}})}}},Y5={name:"env-fallback-explicit",matches($,J){if(Object.keys($).length>0)return!1;let Z=b(J.baseUrl),H=b(J.authUrl),K=b(J.clientId),W=b(J.clientSecret);return Z!==void 0&&H!==void 0&&K!==void 0&&W!==void 0},resolve($,J){let Z=b(J.baseUrl),H=b(J.authUrl),K=b(J.clientId),W=b(J.clientSecret);return{mode:"provider",provider:new b2({baseUrl:Z,authUrl:H,auth:{clientId:K,clientSecret:W}})}}},Q5={name:"token-provider",matches($){return"tokenProvider"in $},resolve(){throw Error("TokenProvider mode is not supported with provider pattern. Use { provider: TimebackProvider } or { env, auth } instead.")}},G5=[Z5,H5,K5,Q5,X5,W5,Y5];function F9($,J,Z=u2){for(let H of G5)if(H.matches($,J))return H.resolve($,J,Z);throw Error(J5(J))}var M8=3,_5=[429,503],t8=1000;class A8{config;log;constructor($){let{config:J,logger:Z}=$,H=J.fetch??globalThis.fetch.bind(globalThis),K;if("tokenProvider"in J&&J.tokenProvider)K=J.tokenProvider;else if("auth"in J&&J.auth)K=new m2({tokenUrl:J.auth.authUrl,credentials:{clientId:J.auth.clientId,clientSecret:J.auth.clientSecret},fetch:H});let W=J.baseUrl.endsWith("/")?J.baseUrl:`${J.baseUrl}/`;this.config={baseUrl:W,timeout:J.timeout??30000,fetch:H,tokenProvider:K},this.log=Z.child("http")}get baseUrl(){return this.config.baseUrl}async request($,J={}){let Z=await this.requestRaw($,J);return this.handleResponse(Z)}async exists($,J={}){try{return await this.requestRaw($,J),!0}catch(Z){if(xJ(Z)&&Z.statusCode===404)return!1;throw Z}}async requestRaw($,J={}){let{method:Z="GET",params:H,body:K,headers:W={}}=J,B=this.buildUrl($,H),F=J.requestId??tJ(),U=Date.now()+this.config.timeout;for(let V=0;V<M8;V++){let A=U-Date.now();if(A<=0)throw this.log.error("Request timeout before attempt",D2()?{requestId:F,path:$}:{path:$}),new s("Request timeout",408);let C=await this.getAccessToken(),q=V===M8-1,z=performance.now();this.log.debug(`→ ${Z} ${B}`,{requestId:F,attempt:V>0?V+1:void 0});let N={"Content-Type":"application/json",Accept:"application/json","X-Request-ID":F,...W};if(C)N.Authorization=`Bearer ${C}`;let I=await this.config.fetch(B,{method:Z,headers:N,body:K?JSON.stringify(K):void 0,signal:AbortSignal.timeout(Math.min(A,this.config.timeout))}),r=Math.round(performance.now()-z);if(this.log.debug(`← ${I.status} ${I.statusText} (${r}ms)`,{requestId:F}),_5.includes(I.status)&&!q){let y2=I.headers.get("Retry-After"),p=this.parseRetryAfter(y2,V),s8=U-Date.now();if(p>=s8)throw this.log.error("Request timeout during retry backoff",D2()?{requestId:F,path:$}:{path:$}),new s("Request timeout",408);this.log.warn(`Retrying in ${p}ms (attempt ${V+1}/${M8})`,{...D2()&&{requestId:F},status:I.status}),await this.sleep(p);continue}if(!I.ok)return this.handleErrorResponse(I,F);return I}throw this.log.error("Max retries exceeded",D2()?{requestId:F,path:$}:{path:$}),new s("Max retries exceeded")}getAccessToken(){if(!this.config.tokenProvider)return Promise.resolve(void 0);return this.config.tokenProvider.getToken()}buildUrl($,J){if(/^[a-z][a-z0-9+.-]*:/i.test($))throw Error(`Absolute URLs are not allowed in path: ${$}. Use relative paths only.`);let Z=$.startsWith("/")?$.slice(1):$,H=new URL(Z,this.config.baseUrl);if(J){for(let[K,W]of Object.entries(J))if(W!==void 0)H.searchParams.set(K,String(W))}return H.toString()}async handleResponse($){if(!$.ok)await this.handleErrorResponse($);if($.status===204)return;if($.headers.get("content-length")==="0")return;return this.parseJsonResponse($)}async parseJsonResponse($){let J=await $.text();if(!J||J.trim()==="")return;try{return JSON.parse(J)}catch(Z){let H=J.length>200?J.slice(0,200)+"...":J,K=Z instanceof Error?Z.message:String(Z),W=$.url||"unknown";throw this.log.error("Failed to parse JSON response",{url:W,status:$.status,contentType:$.headers.get("content-type"),bodyPreview:H,error:K}),new s(`Invalid JSON response from ${W}`,$.status,{parseError:K,body:H})}}async handleErrorResponse($,J){let Z,H=await $.text().catch(()=>"");if(H)try{Z=JSON.parse(H)}catch(W){let B=W instanceof Error?W.message:"Unknown parse error";Z={rawBody:H.slice(0,500),parseError:B},this.log.warn("Failed to parse error response as JSON",{...D2()&&{requestId:J},url:$.url,status:$.status,parseError:B,bodyPreview:H.slice(0,200)})}let K=this.extractErrorMessage(Z,$.statusText);if($.status!==404)this.log.error(`Request failed: ${$.status} ${K}`,D2()?{requestId:J}:void 0);switch($.status){case 401:throw this.config.tokenProvider?.invalidate?.(),new $9(K,Z);case 403:throw new J9(K,Z);case 404:throw new Z9(K,Z);case 422:throw new H9(K,Z);default:throw new s(K,$.status,Z)}}extractErrorMessage($,J){if(typeof $==="object"&&$!==null){let Z=$;if(typeof Z.message==="string")return Z.message;if(typeof Z.error==="string")return Z.error;if(typeof Z.imsx_description==="string")return Z.imsx_description}return J}sleep($){return new Promise((J)=>{setTimeout(J,$)})}parseRetryAfter($,J){if(!$)return t8*Math.pow(2,J);let Z=parseInt($,10);if(!isNaN(Z))return Z*1000;let H=Date.parse($);if(!isNaN(H)){let K=H-Date.now();return Math.max(0,K)}return t8*Math.pow(2,J)}}function B5($){if($ instanceof Date)return $.toISOString();if(typeof $==="boolean")return $?"true":"false";if(typeof $==="number")return String($);return $.replaceAll("'","''")}function t($){let J=B5($);return typeof $==="string"||$ instanceof Date?`'${J}'`:J}function F5($,J){if(typeof J==="string"||typeof J==="number"||typeof J==="boolean"||J instanceof Date)return[`${$}=${t(J)}`];if(typeof J==="object"&&J!==null){let Z=J,H=[];if(Z.ne!==void 0)H.push(`${$}!=${t(Z.ne)}`);if(Z.gt!==void 0)H.push(`${$}>${t(Z.gt)}`);if(Z.gte!==void 0)H.push(`${$}>=${t(Z.gte)}`);if(Z.lt!==void 0)H.push(`${$}<${t(Z.lt)}`);if(Z.lte!==void 0)H.push(`${$}<=${t(Z.lte)}`);if(Z.contains!==void 0)H.push(`${$}~${t(Z.contains)}`);if(Z.in!==void 0&&Z.in.length>0){let K=Z.in.map((B)=>`${$}=${t(B)}`),W=K.join(" OR ");H.push(K.length>1?`(${W})`:W)}if(Z.notIn!==void 0&&Z.notIn.length>0){let K=Z.notIn.map((W)=>`${$}!=${t(W)}`);H.push(K.join(" AND "))}return H}return[]}function O5($){return"OR"in $&&Array.isArray($.OR)}function O9($){if(O5($)){let Z=$.OR.map((H)=>O9(H)).filter((H)=>H!==void 0);return Z.length>0?Z.join(" OR "):void 0}let J=[];for(let[Z,H]of Object.entries($))if(H!==void 0)J.push(...F5(Z,H));return J.length>0?J.join(" AND "):void 0}var e8=100,D5=1e4;class z8{fetcher;path;params;max;unwrapKey;log;transform;paginationStyle;constructor($){this.fetcher=$.fetcher,this.path=$.path,this.params=$.params??{},this.max=$.max,this.unwrapKey=$.unwrapKey,this.log=$.logger?.child("pagination")??C8("pagination"),this.transform=$.transform,this.paginationStyle=$.paginationStyle??"offset"}buildRequestParams($,J){let{where:Z,fields:H,offset:K,limit:W,...B}=this.params,F={...B,filter:Z?O9(Z):void 0,fields:H?.join(","),limit:$};if(this.paginationStyle==="page"){let O=Math.floor(J/$)+1;return{...F,page:O}}return{...F,offset:J}}extractItems($,J){let Z;if(this.unwrapKey)Z=$[this.unwrapKey]??[];else Z=$;return this.validateItems(Z,J)}validateItems($,J){if($===null||$===void 0)return this.log.warn(`Page ${J}: response data is ${$}, treating as empty`),[];if(!Array.isArray($))throw Error(`Expected array for page ${J}, got ${typeof $}. `+(this.unwrapKey?`Check unwrapKey '${this.unwrapKey}' is correct.`:""));return $}hasMorePages($,J,Z,H){if(J===0)return!1;let K=$.hasMore,W=$.total!==void 0&&Z+J<$.total;return K||W||J===H&&$.total===void 0}async*[Symbol.asyncIterator](){let $=this.params.offset??0,J=this.params.limit??e8,Z=!0,H=1,K=0;while(Z){this.log.debug(`Fetching page ${H} (offset: ${$}, limit: ${J})`);let W=await this.fetcher(this.path,{params:this.buildRequestParams(J,$)}),B=this.extractItems(W.data,H);this.log.debug(`Page ${H}: ${B.length} items`);for(let F of B)if(yield this.transform?this.transform(F):F,K++,this.max!==void 0&&K>=this.max){this.log.debug(`Pagination complete: reached max of ${this.max} items`);return}Z=this.hasMorePages(W,B.length,$,J),$+=B.length,H++}this.log.debug(`Pagination complete: ${K} total items`)}async toArray($={}){let J=$.maxItems??D5,Z=[];for await(let H of this){if(Z.length>=J)throw Error(`toArray() exceeded maxItems limit of ${J}. Use 'for await...of' to stream large datasets, or increase maxItems.`);Z.push(H)}return Z}async firstPage(){let $=this.params.limit??e8,J=this.params.offset??0,Z=await this.fetcher(this.path,{params:this.buildRequestParams($,J)}),H=this.extractItems(Z.data,1),K=this.transform?H.map(this.transform):H,W=this.hasMorePages(Z,K.length,J,$);return this.log.debug(`First page: ${K.length} items, total: ${Z.total}, hasMore: ${W}`),{data:K,hasMore:W,total:Z.total,nextOffset:W?J+K.length:void 0}}}function g2($,J){return{path:$,message:J}}function o($,J,Z){let H=$.safeParse(J);if(H.success)return H.data;let K=H.error.issues.map((W)=>({path:W.path.join(".")||"(root)",message:W.message}));throw I2(`Invalid ${Z} data`,K)}function c2($,J){if(typeof $!=="string"||$.trim()==="")throw I2(`Invalid ${J}`,[g2(J,"Must be a non-empty string")])}function P8($){if(!$)return;let J=[];if($.limit!==void 0){if(!Number.isInteger($.limit)||$.limit<=0)J.push(g2("limit","Must be a positive integer"))}if($.offset!==void 0){if(!Number.isInteger($.offset)||$.offset<0)J.push(g2("offset","Must be a non-negative integer"))}if($.max!==void 0){if(!Number.isInteger($.max)||$.max<=0)J.push(g2("max","Must be a positive integer"))}if(J.length>0)throw I2("Invalid list parameters",J)}var D9={baseUrl:["TIMEBACK_API_BASE_URL","TIMEBACK_BASE_URL","CALIPER_BASE_URL"],authUrl:["TIMEBACK_API_AUTH_URL","TIMEBACK_AUTH_URL","CALIPER_TOKEN_URL"],clientId:["TIMEBACK_API_CLIENT_ID","TIMEBACK_CLIENT_ID","CALIPER_CLIENT_ID"],clientSecret:["TIMEBACK_API_CLIENT_SECRET","TIMEBACK_CLIENT_SECRET","CALIPER_CLIENT_SECRET"]},q2="http://purl.imsglobal.org/ctx/caliper/v1p2",w9="QUESTION_RESULT";import{z as L}from"zod/v4";import{z as Y}from"zod/v4";import{z as Z2}from"zod/v4";import{z as x}from"zod/v4";import{z as w}from"zod/v4";import{z as k}from"zod/v4";import{z as D}from"zod/v4";import{z as g}from"zod/v4";import{z as Q}from"zod/v4";import{z as _}from"zod/v4";import{z as G}from"zod/v4";import{z as z2}from"zod/v4";var S=C8("caliper");class k9 extends A8{paths;constructor($){super({config:$,logger:S});this.paths=$.paths}async requestPaginated($,J={}){let Z=await this.request($,J),{events:H,pagination:K}=Z,B=(J.params?.offset??0)+H.length<K.total;return{data:H,hasMore:B,total:K.total}}}class q9 extends z8{constructor($,J,Z={}){P8(Z);let{max:H,...K}=Z;super({fetcher:(W,B)=>$.requestPaginated(W,B),path:J,params:K,max:H,logger:S})}}function T8($){let J=$.id??`urn:uuid:${crypto.randomUUID()}`,Z=$.metricsId??`urn:uuid:${crypto.randomUUID()}`;return{"@context":q2,id:J,type:"ActivityEvent",action:"Completed",actor:$.actor,object:$.object,eventTime:$.eventTime??new Date().toISOString(),profile:"TimebackProfile",generated:{id:Z,type:"TimebackActivityMetricsCollection",items:$.metrics,...$.attempt===void 0?{}:{attempt:$.attempt},...$.generatedExtensions?{extensions:$.generatedExtensions}:{}},extensions:$.extensions,...$.edApp===void 0?{}:{edApp:$.edApp},...$.session===void 0?{}:{session:$.session}}}function R8($){let J=$.id??`urn:uuid:${crypto.randomUUID()}`,Z=$.metricsId??`urn:uuid:${crypto.randomUUID()}`;return{"@context":q2,id:J,type:"TimeSpentEvent",action:"SpentTime",actor:$.actor,object:$.object,eventTime:$.eventTime??new Date().toISOString(),profile:"TimebackProfile",generated:{id:Z,type:"TimebackTimeSpentMetricsCollection",items:$.metrics},extensions:$.extensions,...$.edApp===void 0?{}:{edApp:$.edApp},...$.session===void 0?{}:{session:$.session}}}function w5($){return{"@context":q2,id:$.id??`urn:uuid:${crypto.randomUUID()}`,type:"AssessmentItemEvent",action:"Started",profile:"AssessmentProfile",actor:$.actor,object:{...$.object,type:"AssessmentItem"},eventTime:$.eventTime??new Date().toISOString(),edApp:$.edApp,...$.session===void 0?{}:{session:$.session},...$.extensions===void 0?{}:{extensions:$.extensions}}}function U5($){return{"@context":q2,id:$.id??`urn:uuid:${crypto.randomUUID()}`,type:"AssessmentItemEvent",action:"Completed",profile:"AssessmentProfile",actor:$.actor,object:{...$.object,type:"AssessmentItem"},eventTime:$.eventTime??new Date().toISOString(),edApp:$.edApp,...$.generated===void 0?{}:{generated:{...$.generated,type:"Response"}},...$.session===void 0?{}:{session:$.session},...$.extensions===void 0?{}:{extensions:$.extensions}}}function x5($){let{id:J,...Z}=$.generated;return{"@context":q2,id:$.id??`urn:uuid:${crypto.randomUUID()}`,type:"GradeEvent",action:"Graded",profile:"GradingProfile",actor:$.actor,object:$.object,eventTime:$.eventTime??new Date().toISOString(),edApp:$.edApp,generated:{...Z,id:J??`urn:uuid:${crypto.randomUUID()}`,type:"Score",scoreType:w9},...$.session===void 0?{}:{session:$.session},...$.extensions===void 0?{}:{extensions:$.extensions}}}var M5="http://purl.imsglobal.org/ctx/caliper/v1p2",L5="urn:tag:auto-attach",Q2={PERSON:"Person",ASSIGNABLE_DIGITAL_RESOURCE:"AssignableDigitalResource",ACTIVITY_METRIC:"ActivityMetric",TIME_SPENT_METRIC:"TimeSpentMetric",ACTIVITY_METRICS_COLLECTION:"ActivityMetricsCollection",TIME_SPENT_METRICS_COLLECTION:"TimeSpentMetricsCollection",COURSE_OFFERING:"CourseOffering",SOFTWARE_APPLICATION:"SoftwareApplication"},j5=new Set(["ActivityEvent","TimeSpentEvent","AssignableEvent","ViewEvent"]);class C9{transformEnvelope($){let{data:J}=$;if(!Array.isArray(J))return $;let Z=J.map((H)=>U2(H)?V5(H):H);return{...$,data:Z}}}function V5($){let J={...$};if(!("@context"in J))J["@context"]=M5;if(J.profile==="TimebackProfile")J.profile="AggregationProfile";if(U2(J.actor)&&J.actor.type==="TimebackUser")J.actor=k5(J.actor);if(U2(J.object))J.object=q5(J.object,J.type);if(U2(J.generated)){let Z=J.type;if(Z==="ActivityEvent")J.generated=C5(J.generated);else if(Z==="TimeSpentEvent")J.generated=A5(J.generated)}if(!J.session)J.session=L5;return J}function k5($){return{type:Q2.PERSON,id:$.id}}function q5($,J){if(!($.type==="TimebackActivityContext"&&J!==void 0&&j5.has(J)))return{...$};let H={type:Q2.ASSIGNABLE_DIGITAL_RESOURCE,id:$.id};if($.name)H.name=$.name;if(U2($.course)){let K={type:Q2.COURSE_OFFERING,id:$.course.id};if($.course.name)K.name=$.course.name;H.isPartOf=K}if(U2($.app))H.extensions={app:{name:$.app.name,type:$.app.type??Q2.SOFTWARE_APPLICATION}};return H}function A9($,J,Z,H){let W={id:typeof $?.id==="string"?$.id:`urn:uuid:${crypto.randomUUID()}`,type:J};if(Array.isArray($.items))W.items=$.items.map((B)=>{if(!U2(B))return B;let F=typeof B.type==="string"?B.type:"",O=F?F[0].toUpperCase()+F.slice(1):H;return{id:`urn:uuid:${crypto.randomUUID()}`,type:Z,metricType:O,metricValue:B.value}});for(let[B,F]of Object.entries($))if(B!=="id"&&B!=="type"&&B!=="items")W[B]=F;return W}function C5($){return A9($,Q2.ACTIVITY_METRICS_COLLECTION,Q2.ACTIVITY_METRIC,"")}function A5($){return A9($,Q2.TIME_SPENT_METRICS_COLLECTION,Q2.TIME_SPENT_METRIC,"Active")}function U2($){return typeof $==="object"&&$!==null&&!Array.isArray($)}function z5($,J=u2){return F9($,D9,J)}var P5=/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(?:\.\d+)?(?:Z|[+-]\d{2}:\d{2})$/,y=L.string().min(1).regex(P5,"must be a valid ISO 8601 datetime"),A2=L.string().min(1).regex(/^\d{4}-\d{2}-\d{2}$/,"must be a valid ISO 8601 date (YYYY-MM-DD)"),X=L.string().trim().min(1),X2=L.enum(["Reading","Language","Vocabulary","Social Studies","Writing","Science","FastMath","Math","None","Other"]).meta({id:"TimebackSubject",description:"Subject area"}),T5=L.union([L.literal(-1),L.literal(0),L.literal(1),L.literal(2),L.literal(3),L.literal(4),L.literal(5),L.literal(6),L.literal(7),L.literal(8),L.literal(9),L.literal(10),L.literal(11),L.literal(12),L.literal(13)]),R5=L.string().transform(($,J)=>{let Z=$.toLowerCase().trim().replaceAll("_"," ");if(Z==="")return J.addIssue({code:"custom",message:"must be a valid Timeback grade"}),L.NEVER;let H=Z.replace(/\bgrade\b/g,"").replace(/(\d+)(st|nd|rd|th)\b/g,"$1").trim();if(H==="")return J.addIssue({code:"custom",message:"must be a valid Timeback grade"}),L.NEVER;if(H==="pre-k"||H==="pk")return-1;if(H==="k")return 0;if(H==="middle school")return 7;let W=H.replace(/\s+/g,"").replace(/^0+/,"")||"0",B=Number(W);if(!Number.isInteger(B))return J.addIssue({code:"custom",message:"must be a valid Timeback grade"}),L.NEVER;return B}),n=L.union([L.number(),R5]).pipe(T5).meta({id:"TimebackGrade",description:"Grade level (-1 = Pre-K, 0 = K, 1-12 = grades, 13 = AP)"}),N5=L.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),I5=L.enum(["department","school","district","local","state","national"]),z9=L.enum(["administrator","aide","guardian","parent","proctor","relative","student","teacher"]),b5=L.enum(["administrator","proctor","student","teacher"]),h5=L.enum(["qti","text","audio","video","interactive","visual","course-material","assessment-bank"]),S5=L.enum(["qti-test","qti-question","qti-stimulus","qti-test-bank"]),E5=L.enum(["unit","course","resource-collection"]),v5=L.enum(["choice","order","associate","match","hotspot","hottext","select-point","graphic-order","graphic-associate","graphic-gap-match","text-entry","extended-text","inline-choice","upload","slider","drawing","media","custom"]),y5=L.enum(["easy","medium","hard"]),P9=L.array(L.object({source:L.string(),learningObjectiveIds:L.array(L.string())})),T9=L.object({consecutive_failures:L.number().int().min(1).optional(),stagnation_limit:L.number().int().min(1).optional()}).optional(),f5=L.enum(["powerpath-100","quiz","test-out","placement","unit-test","alpha-read-article"]).nullable(),A1=L.object({imsx_codeMajor:L.enum(["failure","success"]),imsx_severity:L.enum(["error","warning","status"]),imsx_description:L.string(),imsx_CodeMinor:L.object({imsx_codeMinorField:L.array(L.object({imsx_codeMinorFieldName:L.string(),imsx_codeMinorFieldValue:L.string()}))}).optional()}),g5=Y.union([Y.url(),Y.string().regex(/^urn:/,"Must be a URL or URN")]),x2=Y.object({id:Y.url(),type:Y.literal("TimebackUser"),email:Y.string()}),N8=Y.object({id:Y.url(),type:Y.literal("TimebackActivityContext"),subject:X2,app:Y.object({name:Y.string()}),activity:Y.object({id:Y.url().optional(),name:Y.string()}).strict().optional(),course:Y.object({id:Y.url().optional(),name:Y.string().optional()}).strict(),process:Y.boolean().optional()}),R9=Y.object({"@context":Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),id:Y.string().regex(/^urn:uuid:/,"Event id must start with urn:uuid:"),type:Y.string(),eventTime:y,profile:Y.literal("TimebackProfile"),actor:x2,action:Y.string(),object:N8,edApp:Y.object({id:Y.string(),name:Y.string().optional()}).optional()}),N9=Y.object({type:Y.enum(["totalQuestions","correctQuestions","xpEarned","masteredUnits"]),value:Y.number()}),d5=Y.object({id:Y.url(),type:Y.literal("TimebackActivityMetricsCollection"),attempt:Y.number().optional(),items:Y.array(N9),extensions:Y.record(Y.string(),Y.unknown()).optional()}),m5=R9.extend({type:Y.literal("ActivityEvent"),action:Y.literal("Completed"),generated:d5}),I9=Y.object({type:Y.enum(["active","inactive","waste","unknown","anti-pattern"]),value:Y.number(),subType:Y.string().optional()}),l5=Y.object({id:Y.url(),type:Y.literal("TimebackTimeSpentMetricsCollection"),items:Y.array(I9)}),u5=R9.extend({type:Y.literal("TimeSpentEvent"),action:Y.literal("SpentTime"),generated:l5}),c5=Y.object({actor:x2,object:N8,metrics:Y.array(N9).min(1,"metrics must contain at least one metric"),eventTime:y.optional(),metricsId:Y.string().optional(),id:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional(),edApp:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),session:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),attempt:Y.number().int().min(1).optional(),generatedExtensions:Y.object({pctCompleteApp:Y.number().optional()}).loose().optional()}).strict(),p5=Y.object({actor:x2,object:N8,metrics:Y.array(I9).min(1,"metrics must contain at least one metric"),eventTime:y.optional(),metricsId:Y.string().optional(),id:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional(),edApp:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),session:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional()}).strict(),a5=Y.union([m5,u5]),z1=Y.object({sensor:Y.string(),sendTime:y,dataVersion:Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),data:Y.array(a5)}),i5=Y.enum(["AnnotationProfile","AssessmentProfile","ToolUseProfile","GeneralProfile","FeedbackProfile","MediaProfile","SurveyProfile","ResourceManagementProfile","ForumProfile","AssignableProfile","GradingProfile","ReadingProfile","SessionProfile","SearchProfile","ToolLaunchProfile","TimebackProfile"]),l=Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]),p2=Y.object({id:g5,type:Y.string(),name:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),b9=Y.object({"@context":Y.string().optional(),id:Y.string().regex(/^urn:uuid:/,"Event id must start with urn:uuid:"),type:Y.string(),actor:Y.union([Y.string(),p2,x2]),action:Y.string(),object:l,eventTime:y,profile:i5,edApp:l.optional(),generated:l.optional(),target:l.optional(),referrer:l.optional(),group:l.optional(),membership:l.optional(),session:l.optional(),federatedSession:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),U9=Y.object({sensor:X,sendTime:y,dataVersion:Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),data:Y.array(b9).min(1,"data must contain at least one event")}),s5=Y.object({sensor:X,events:Y.array(b9).min(1,"events must contain at least one event")}),x9=Y.object({limit:Y.number().int().positive().optional(),offset:Y.number().int().min(0).optional(),sensor:X.optional(),startDate:y.optional(),endDate:y.optional(),actorId:X.optional(),actorEmail:Y.email().optional()}).strict(),h9=Y.object({id:Y.string(),name:Y.string().optional(),isPartOf:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),o5=Y.object({id:Y.string(),attempt:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),startedAtTime:y.optional(),endedAtTime:y.optional(),duration:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),n5=Y.object({id:Y.string().optional(),scoreGiven:Y.number(),maxScore:Y.number().optional(),attempt:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),comment:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),r5=Y.object({actor:Y.union([Y.string(),p2,x2]),object:h9,edApp:l,id:Y.string().optional(),eventTime:y.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),t5=Y.object({actor:Y.union([Y.string(),p2,x2]),object:h9,edApp:l,generated:o5.optional(),id:Y.string().optional(),eventTime:y.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),e5=Y.object({actor:Y.union([Y.string(),p2,x2]),object:Y.string(),generated:n5,edApp:l,id:Y.string().optional(),eventTime:y.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),P1=Z2.object({name:X,targetUrl:Z2.url("targetUrl must be a valid URL"),secret:X,active:Z2.boolean(),sensor:Z2.string().nullable().optional(),description:Z2.string().nullable().optional()}).strict(),$Z=Z2.enum(["string","number","boolean"]),JZ=Z2.enum(["eq","neq","gt","gte","lt","lte","contains","notContains","in","notIn","startsWith","endsWith","regexp"]),T1=Z2.object({webhookId:X,filterKey:X,filterValue:X,filterType:$Z,filterOperator:JZ,active:Z2.boolean()}).strict(),a2=x.string().uuid(),M9=x.object({title:X,identifier:a2,uri:X}),ZZ=x.object({identifier:a2,uri:X,lastChangeDateTime:X,title:X,creator:X,officialSourceURL:x.string().optional(),publisher:x.string().optional(),description:x.string().optional(),language:x.string().optional(),version:x.string().optional(),caseVersion:x.string().optional(),adoptionStatus:x.string().optional(),statusStartDate:x.string().optional(),statusEndDate:x.string().optional(),licenseUri:x.string().optional(),notes:x.string().optional(),subject:x.array(x.string()).optional(),extensions:x.unknown().optional()}),HZ=x.object({identifier:a2,uri:X,lastChangeDateTime:X,fullStatement:X,alternativeLabel:x.string().optional(),CFItemType:x.string().optional(),cfItemType:x.string().optional(),humanCodingScheme:x.string().optional(),listEnumeration:x.string().optional(),abbreviatedStatement:x.string().optional(),conceptKeywords:x.array(x.string()).optional(),notes:x.string().optional(),subject:x.array(x.string()).optional(),language:x.string().optional(),educationLevel:x.array(x.string()).optional(),CFItemTypeURI:x.unknown().optional(),licenseURI:x.unknown().optional(),statusStartDate:x.string().optional(),statusEndDate:x.string().optional(),extensions:x.unknown().optional()}),KZ=x.object({identifier:a2,uri:X,lastChangeDateTime:X,associationType:X,originNodeURI:M9,destinationNodeURI:M9,sequenceNumber:x.number().optional(),extensions:x.unknown().optional()}),XZ=x.object({CFItemTypes:x.array(x.unknown()).optional(),CFSubjects:x.array(x.unknown()).optional(),CFConcepts:x.array(x.unknown()).optional(),CFLicenses:x.array(x.unknown()).optional(),CFAssociationGroupings:x.array(x.unknown()).optional(),extensions:x.unknown().optional()}),R1=x.object({CFDocument:ZZ,CFItems:x.array(HZ),CFAssociations:x.array(KZ),CFDefinitions:XZ.optional(),extensions:x.unknown().optional()}),L9="https://www.w3.org/ns/credentials/v2",WZ=w.array(w.url()).min(3,"@context must include W3C, CLR, and OB context URLs").refine(($)=>$[0]===L9,`First @context entry must be "${L9}"`),S9=w.object({id:w.url(),type:w.literal("Image"),caption:w.string().optional()}),I8=w.object({id:w.url(),type:w.array(w.string()).min(1).refine(($)=>$.includes("Profile"),'type must include "Profile"'),name:w.string().optional(),url:w.string().url().optional(),phone:w.string().optional(),description:w.string().optional(),image:S9.optional(),email:w.string().email().optional()}),E9=w.object({type:X,proofPurpose:X,verificationMethod:X,created:w.iso.datetime(),proofValue:X,cryptosuite:w.string().optional()}),YZ=w.object({id:X,type:w.literal("1EdTechJsonSchemaValidator2019")}),QZ=w.object({id:w.string().url().optional(),narrative:w.string().optional()}),GZ=w.object({id:w.url(),type:w.array(w.string()).min(1).refine(($)=>$.includes("Achievement"),'type must include "Achievement"'),name:X,description:X,criteria:QZ,image:S9.optional(),achievementType:w.string().optional(),creator:I8.optional()}),_Z=w.object({type:w.literal("IdentityObject"),identityHash:X,identityType:X,hashed:w.boolean(),salt:w.string().optional()}),BZ=w.enum(["exactMatchOf","isChildOf","isParentOf","isPartOf","isPeerOf","isRelatedTo","precedes","replacedBy"]),FZ=w.object({type:w.literal("Association"),associationType:BZ,sourceId:w.url(),targetId:w.url()}),OZ=w.object({"@context":w.array(w.string()).min(1),id:w.url(),type:w.array(w.string()).min(1).refine(($)=>$.includes("VerifiableCredential"),'type must include "VerifiableCredential"'),issuer:I8,validFrom:w.iso.datetime(),validUntil:w.string().datetime().optional(),credentialSubject:w.object({id:w.string().optional()}),proof:w.array(E9).optional()}),DZ=w.object({id:w.string().url().optional(),type:w.array(w.string()).min(1).refine(($)=>$.includes("ClrSubject"),'type must include "ClrSubject"'),identifier:w.array(_Z).optional(),achievement:w.array(GZ).optional(),verifiableCredential:w.array(OZ).min(1),association:w.array(FZ).optional()}),N1=w.object({"@context":WZ,id:w.url(),type:w.array(w.string()).min(1).refine(($)=>$.includes("VerifiableCredential")&&$.includes("ClrCredential"),'type must include "VerifiableCredential" and "ClrCredential"'),issuer:I8,name:X,description:w.string().optional(),validFrom:w.iso.datetime(),validUntil:w.iso.datetime().optional(),credentialSubject:DZ,proof:w.array(E9).optional(),credentialSchema:w.array(YZ).optional()}),wZ=k.object({staging:k.string().meta({description:"Course ID in staging environment"}).optional(),production:k.string().meta({description:"Course ID in production environment"}).optional()}).meta({id:"CourseIds",description:"Environment-specific course IDs (populated by sync)"}),UZ=k.enum(["base","hole-filling","optional"]).meta({id:"CourseType",description:"Course classification type"}),xZ=k.enum(["draft","testing","published","deactivated"]).meta({id:"PublishStatus",description:"Course publication status"}),MZ=k.object({dailyXp:k.number().int().positive().meta({description:"Target XP to earn per day"}).optional(),dailyLessons:k.number().int().positive().meta({description:"Target lessons to complete per day"}).optional(),dailyActiveMinutes:k.number().int().positive().meta({description:"Target active learning minutes per day"}).optional(),dailyAccuracy:k.number().int().min(0).max(100).meta({description:"Target accuracy percentage (0-100)"}).optional(),dailyMasteredUnits:k.number().int().positive().meta({description:"Target units to master per day"}).optional()}).meta({id:"CourseGoals",description:"Daily learning goals for a course"}),LZ=k.object({totalXp:k.number().int().positive().meta({description:"Total XP available in the course"}).optional(),totalLessons:k.number().int().positive().meta({description:"Total number of lessons/activities"}).optional(),totalGrades:k.number().int().positive().meta({description:"Total grade levels covered"}).optional()}).meta({id:"CourseMetrics",description:"Aggregate metrics for a course"}),v9=k.object({courseType:UZ.optional(),isSupplemental:k.boolean().meta({description:"Whether this is supplemental to a base course"}).optional(),isCustom:k.boolean().meta({description:"Whether this is a custom course for an individual student"}).optional(),publishStatus:xZ.optional(),contactEmail:k.email().meta({description:"Contact email for course issues"}).optional(),primaryApp:k.string().meta({description:"Primary application identifier"}).optional(),goals:MZ.optional(),metrics:LZ.optional()}).meta({id:"CourseMetadata",description:"Course metadata (matches API metadata object)"}),y9=k.object({courseCode:k.string().meta({description:"Course code (e.g., 'MATH101')"}).optional(),level:k.string().meta({description:"Course level (e.g., 'AP', 'Honors')"}).optional(),metadata:v9.optional()}).meta({id:"CourseDefaults",description:"Default properties that apply to all courses unless overridden"}),j9=k.object({level:k.string().meta({description:"Course level for this environment"}).optional(),sensor:k.url().meta({description:"Caliper sensor endpoint URL for this environment"}).optional(),launchUrl:k.url().meta({description:"LTI launch URL for this environment"}).optional(),metadata:v9.optional()}).meta({id:"CourseEnvOverrides",description:"Environment-specific course overrides (non-identity fields)"}),jZ=k.object({staging:j9.meta({description:"Overrides for staging environment"}).optional(),production:j9.meta({description:"Overrides for production environment"}).optional()}).meta({id:"CourseOverrides",description:"Per-environment course overrides"}),VZ=y9.extend({subject:X2.meta({description:"Subject area for this course"}),grade:n.meta({description:"Grade level (-1 = Pre-K, 0 = K, 1-12 = grades, 13 = AP)"}).optional(),ids:wZ.nullable().optional(),sensor:k.url().meta({description:"Caliper sensor endpoint URL for this course"}).optional(),launchUrl:k.url().meta({description:"LTI launch URL for this course"}).optional(),overrides:jZ.optional()}).meta({id:"CourseConfig",description:"Configuration for a single course. Must have either grade or courseCode (or both)."}),I1=k.object({$schema:k.string().meta({description:"JSON Schema reference for editor support"}).optional(),name:k.string().min(1,"App name is required").meta({description:"Display name for your app"}),defaults:y9.meta({description:"Default properties applied to all courses"}).optional(),courses:k.array(VZ).min(1,"At least one course is required").meta({description:"Courses available in this app"}),sensor:k.url().meta({description:"Default Caliper sensor endpoint URL for all courses"}).optional(),launchUrl:k.url().meta({description:"Default LTI launch URL for all courses"}).optional(),studio:k.object({telemetry:k.boolean().meta({description:"Enable anonymous usage telemetry for Studio (default: true)"}).optional().default(!0)}).meta({description:"Studio-specific configuration"}).optional()}).meta({id:"TimebackConfig",title:"Timeback Config",description:"Configuration schema for timeback.config.json files"}).refine(($)=>{return $.courses.every((J)=>J.grade!==void 0||J.courseCode!==void 0)},{message:"Each course must have either a grade or a courseCode",path:["courses"]}).refine(($)=>{let Z=$.courses.filter((H)=>H.grade!==void 0).map((H)=>`${H.subject}:${H.grade}`);return new Set(Z).size===Z.length},{message:"Duplicate (subject, grade) pair found; each must be unique",path:["courses"]}).refine(($)=>{let Z=$.courses.filter((H)=>H.courseCode!==void 0).map((H)=>H.courseCode);return new Set(Z).size===Z.length},{message:"Duplicate courseCode found; each must be unique",path:["courses"]}).refine(($)=>{return $.courses.every((J)=>{if(J.sensor!==void 0||$.sensor!==void 0)return!0;return[J.launchUrl,$.launchUrl,J.overrides?.staging?.launchUrl,J.overrides?.production?.launchUrl].filter(Boolean).length>0})},{message:"Each course must have an effective sensor. Either set `sensor` explicitly (top-level or per-course), or provide a `launchUrl` so sensor can be derived from its origin.",path:["courses"]}),H2=D.union([y,A2]),b1=D.object({id:D.string(),role:D.string(),beginDate:H2.nullable(),endDate:H2.nullable(),metadata:D.object({goals:D.object({dailyXp:D.number().optional()}).optional(),metrics:D.object({totalXp:D.number().optional(),totalLessons:D.number().optional()}).optional()}).optional(),course:D.object({id:D.string(),title:D.string(),subjects:D.array(D.string()).nullable(),grades:D.array(D.string()).nullable()}),school:D.object({id:D.string(),name:D.string()}),testOutSupported:D.boolean(),testOutEligible:D.boolean()}),kZ=D.object({activityMetrics:D.object({xpEarned:D.number(),totalQuestions:D.number(),correctQuestions:D.number(),masteredUnits:D.number()}),timeSpentMetrics:D.object({activeSeconds:D.number(),inactiveSeconds:D.number(),wasteSeconds:D.number()}),apps:D.array(D.string())}),qZ=D.record(D.string(),D.record(D.string(),kZ)),h1=D.object({message:D.string(),enrollmentId:D.string(),startDate:H2,endDate:H2,facts:qZ,factsByApp:D.unknown()}),f9=D.object({email:D.email().optional(),studentId:X.optional()}).superRefine(($,J)=>{if(!$.email&&!$.studentId)J.addIssue({code:D.ZodIssueCode.custom,message:"must provide either email or studentId",path:["email"]}),J.addIssue({code:D.ZodIssueCode.custom,message:"must provide either email or studentId",path:["studentId"]})}),S1=D.object({subject:X,grade:X,courseId:X,orgSourcedId:X.optional()}),E1=D.object({userId:X}),v1=D.object({sourcedId:X.optional(),role:b5.optional(),beginDate:y.optional(),metadata:D.record(D.string(),D.unknown()).optional()}),y1=D.object({fields:D.string().optional(),limit:D.number().int().positive().optional(),offset:D.number().int().nonnegative().optional(),sort:D.string().optional(),orderBy:D.enum(["asc","desc"]).optional(),filter:D.string().optional(),search:D.string().optional(),roles:D.array(z9).min(1),orgSourcedIds:D.array(X).optional()}),f1=f9.extend({startDate:H2,endDate:H2,timezone:D.string().optional()}),g1=f9.extend({weekDate:H2,timezone:D.string().optional()}),d1=D.object({enrollmentId:X,startDate:H2.optional(),endDate:H2.optional(),timezone:D.string().optional()}),m1=g.object({email:g.email()}),l1=g.object({name:X.optional(),timeback_id:X.optional(),grade:X.optional(),subject:X.optional(),all:g.boolean().optional()}),u1=g.object({student_email:g.email(),timeback_id:X.optional(),subject:X.optional(),grade_rank:g.number().int().min(0).max(12).optional(),assessment_line_item_sourced_id:X.optional(),assessment_result_sourced_id:X.optional()}).superRefine(($,J)=>{let Z=!!$.timeback_id,H=!!$.subject,K=$.grade_rank!==void 0;if(!Z&&!H)J.addIssue({code:g.ZodIssueCode.custom,message:"must provide either timeback_id or subject",path:["timeback_id"]}),J.addIssue({code:g.ZodIssueCode.custom,message:"must provide either timeback_id or subject",path:["subject"]});if(K&&!H)J.addIssue({code:g.ZodIssueCode.custom,message:"grade_rank requires subject",path:["grade_rank"]});let W=!!$.assessment_line_item_sourced_id,B=!!$.assessment_result_sourced_id;if(W!==B)J.addIssue({code:g.ZodIssueCode.custom,message:"assessment_line_item_sourced_id and assessment_result_sourced_id must be provided together",path:W?["assessment_result_sourced_id"]:["assessment_line_item_sourced_id"]})}),c1=g.object({student_email:g.email(),assignment_id:g.number().int().positive().optional(),timeback_id:X.optional(),subject:X.optional(),grade_rank:g.number().int().min(0).max(12).optional()}).superRefine(($,J)=>{let Z=$.assignment_id!==void 0,H=!!$.timeback_id,K=!!$.subject&&$.grade_rank!==void 0;if(!Z&&!H&&!K)J.addIssue({code:g.ZodIssueCode.custom,message:"Either assignment_id, timeback_id, or (subject and grade_rank) is required",path:["assignment_id"]});if($.grade_rank!==void 0&&!$.subject)J.addIssue({code:g.ZodIssueCode.custom,message:"grade_rank requires subject",path:["grade_rank"]})}),f=Q.enum(["active","tobedeleted"]),v=Q.record(Q.string(),Q.unknown()).nullable().optional(),j=Q.object({sourcedId:X,type:Q.string().optional(),href:Q.string().optional()}).strict(),CZ=Q.enum(["academicSession","assessmentLineItem","category","class","course","demographics","enrollment","gradingPeriod","lineItem","org","resource","result","scoreScale","student","teacher","term","user","componentResource","courseComponent"]),AZ=Q.object({sourcedId:X,type:CZ,href:Q.url()}).strict(),K2=Q.union([A2,y]).transform(($)=>$.includes("T")?$:`${$}T00:00:00Z`),zZ=Q.object({learningObjectiveId:Q.string(),score:Q.number().optional(),textScore:Q.string().optional()}),g9=Q.array(Q.object({source:Q.string(),learningObjectiveResults:Q.array(zZ)})),PZ=Q.object({roleType:Q.enum(["primary","secondary"]),role:z9,org:j,userProfile:Q.string().optional(),metadata:v,beginDate:K2.optional(),endDate:K2.optional()}).strict(),p1=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:f.optional(),enabledUser:Q.union([Q.boolean(),Q.enum(["true","false"]).transform(($)=>$==="true")]),givenName:X.describe("givenName must be a non-empty string"),familyName:X.describe("familyName must be a non-empty string"),middleName:X.nullable().optional(),preferredFirstName:X.nullable().optional(),preferredMiddleName:X.nullable().optional(),preferredLastName:X.nullable().optional(),username:X.nullable().optional(),email:Q.email(),userMasterIdentifier:Q.string().nullable().optional(),roles:Q.array(PZ).min(1,"roles must include at least one role"),userIds:Q.array(Q.object({type:X,identifier:X}).strict()).optional(),agents:Q.array(j).optional(),primaryOrg:j.optional(),grades:Q.array(n).optional(),sms:X.nullable().optional(),phone:X.nullable().optional(),pronouns:X.nullable().optional(),password:X.nullable().optional(),metadata:v}).strict(),a1=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:f,title:X.describe("title must be a non-empty string"),org:j,courseCode:X.nullable().optional(),subjects:Q.array(X2).nullable().optional(),subjectCodes:Q.array(Q.string()).nullable().optional(),grades:Q.array(n).nullable().optional(),level:X.nullable().optional(),academicSession:j.nullable().optional(),schoolYear:AZ.nullable().optional(),gradingScheme:X.nullable().optional(),metadata:v}).strict(),i1=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:f.optional(),title:X.describe("title must be a non-empty string"),terms:Q.array(j).min(1,"terms must have at least one item"),course:j,org:j,classCode:X.nullable().optional(),classType:Q.enum(["homeroom","scheduled"]).optional(),location:X.nullable().optional(),grades:Q.array(n).optional(),subjects:Q.array(X2).optional(),subjectCodes:Q.array(X).optional(),periods:Q.array(X).optional(),metadata:v}).strict(),J2=Q.enum(["true","false"]),s1=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:f.optional(),user:j,class:j,school:j.optional(),role:Q.enum(["administrator","proctor","student","teacher"]),primary:J2.optional(),beginDate:A2.optional(),endDate:A2.optional(),metadata:v}).strict(),o1=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),status:f,weight:Q.number().nullable().optional(),metadata:v}).strict(),n1=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),class:j,school:j,category:j,assignDate:K2,dueDate:K2,status:f.optional(),description:Q.string().nullable().optional(),resultValueMin:Q.number().nullable().optional(),resultValueMax:Q.number().nullable().optional(),gradingPeriod:j.nullable().optional(),academicSession:j.nullable().optional(),scoreScale:j.nullable().optional(),learningObjectiveSet:P9.nullable().optional(),metadata:v}).strict(),r1=Q.object({sourcedId:X.optional(),lineItem:j,student:j,class:j.nullable().optional(),scoreDate:K2,scoreStatus:Q.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),score:Q.number().nullable().optional(),textScore:Q.string().nullable().optional(),status:f,scoreScale:j.nullable().optional(),comment:Q.string().nullable().optional(),learningObjectiveSet:g9.nullable().optional(),inProgress:Q.string().optional(),incomplete:Q.string().optional(),late:Q.string().optional(),missing:Q.string().optional(),metadata:v}).strict(),t1=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),status:f,type:X,class:j,course:j.nullable().optional(),scoreScaleValue:Q.array(Q.object({itemValueLHS:X,itemValueRHS:X,value:Q.string().optional(),description:Q.string().optional()}).strict()),metadata:v}).strict(),e1=Q.object({sourcedId:X.optional(),status:f,dateLastModified:y.optional(),title:X.describe("title must be a non-empty string"),description:Q.string().nullable().optional(),class:j.nullable().optional(),parentAssessmentLineItem:j.nullable().optional(),scoreScale:j.nullable().optional(),resultValueMin:Q.number().nullable().optional(),resultValueMax:Q.number().nullable().optional(),component:j.nullable().optional(),componentResource:j.nullable().optional(),learningObjectiveSet:Q.array(Q.object({source:Q.string(),learningObjectiveIds:Q.array(Q.string())})).optional().nullable(),course:j.nullable().optional(),metadata:v}).strict(),$6=Q.object({sourcedId:X.optional(),status:f,dateLastModified:y.optional(),metadata:v,assessmentLineItem:j,student:j,score:Q.number().nullable().optional(),textScore:Q.string().nullable().optional(),scoreDate:K2,scoreScale:j.nullable().optional(),scorePercentile:Q.number().nullable().optional(),scoreStatus:Q.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),comment:Q.string().nullable().optional(),learningObjectiveSet:g9.nullable().optional(),inProgress:Q.string().nullable().optional(),incomplete:Q.string().nullable().optional(),late:Q.string().nullable().optional(),missing:Q.string().nullable().optional()}).strict(),J6=Q.object({sourcedId:X.optional(),status:f.optional(),name:X.describe("name must be a non-empty string"),type:I5,identifier:Q.string().nullish(),parent:j.optional(),metadata:v}).strict(),Z6=Q.object({sourcedId:X.optional(),status:f.optional(),name:X.describe("name must be a non-empty string"),type:Q.literal("school").default("school"),identifier:Q.string().nullish(),parent:j.optional(),metadata:v}).strict(),H6=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:f,title:X.describe("title must be a non-empty string"),startDate:K2,endDate:K2,type:Q.enum(["gradingPeriod","semester","schoolYear","term"]),schoolYear:X.describe("schoolYear must be a non-empty string"),org:j,parent:j.optional(),children:Q.array(j).optional(),metadata:v}).strict(),K6=Q.object({sourcedId:X,title:X.describe("title must be a non-empty string"),courseComponent:j,resource:j,status:f,sortOrder:Q.number().optional(),lessonType:f5.optional(),metadata:v}).strict(),X6=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),course:j,status:f,sortOrder:Q.number().optional(),parent:j.nullable().optional(),courseComponent:j.nullable().optional(),prerequisites:Q.array(Q.string()).nullable().optional(),prerequisiteCriteria:Q.string().nullable().optional(),unlockDate:K2.nullable().optional(),metadata:v}).strict().transform(({courseComponent:$,parent:J,...Z})=>({...Z,parent:J===void 0?$??void 0:J})),W6=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string"),role:Q.enum(["student","teacher"]),primary:J2.optional(),beginDate:A2.optional(),endDate:A2.optional(),metadata:v}).strict(),Y6=Q.object({agentSourcedId:X.describe("agentSourcedId must be a non-empty string")}).strict(),Q6=Q.object({applicationName:X.describe("applicationName must be a non-empty string"),credentials:Q.object({username:X.describe("username must be a non-empty string"),password:X.describe("password must be a non-empty string")})}).strict(),G6=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:f.optional(),metadata:v,birthDate:Q.string().regex(/^\d{4}-\d{2}-\d{2}$/).nullable().optional(),sex:Q.enum(["male","female"]).nullable().optional(),americanIndianOrAlaskaNative:J2.nullable().optional(),asian:J2.nullable().optional(),blackOrAfricanAmerican:J2.nullable().optional(),nativeHawaiianOrOtherPacificIslander:J2.nullable().optional(),white:J2.nullable().optional(),demographicRaceTwoOrMoreRaces:J2.nullable().optional(),hispanicOrLatinoEthnicity:J2.nullable().optional(),countryOfBirthCode:Q.string().nullable().optional(),stateOfBirthAbbreviation:Q.string().nullable().optional(),cityOfBirth:Q.string().nullable().optional(),publicSchoolResidenceStatus:Q.string().nullable().optional()}).strict(),G2=Q.object({type:h5,subject:X2.nullish(),grades:Q.array(n).nullish(),language:Q.string().nullish(),xp:Q.number().nullish(),url:Q.url().nullish(),keywords:Q.array(Q.string()).nullish(),learningObjectiveSet:P9.nullish(),lessonType:Q.string().nullish()}).passthrough(),TZ=G2.extend({type:Q.literal("qti"),subType:S5,questionType:v5.optional(),difficulty:y5.optional()}),RZ=G2.extend({type:Q.literal("text"),format:Q.string(),author:Q.string().optional(),pageCount:Q.number().optional()}),NZ=G2.extend({type:Q.literal("audio"),duration:Q.string().regex(/^\d{2}:\d{2}:\d{2}(\.\d{2})?$/).optional(),format:Q.string(),speaker:Q.string().optional()}),IZ=G2.extend({type:Q.literal("video"),duration:Q.string().regex(/^\d{2}:\d{2}:\d{2}(\.\d{2})?$/).optional(),captionsAvailable:Q.boolean().optional(),format:Q.string()}),bZ=G2.extend({type:Q.literal("interactive"),launchUrl:Q.url().optional(),toolProvider:Q.string().optional(),instructionalMethod:Q.string().optional(),courseIdOnFail:Q.string().nullable().optional(),fail_fast:T9}),hZ=G2.extend({type:Q.literal("visual"),format:Q.string(),resolution:Q.string().optional()}),SZ=G2.extend({type:Q.literal("course-material"),subType:E5,author:Q.string().optional(),format:Q.string(),instructionalMethod:Q.string().optional()}),EZ=G2.extend({type:Q.literal("assessment-bank"),resources:Q.array(Q.string())}),vZ=Q.discriminatedUnion("type",[TZ,RZ,NZ,IZ,bZ,hZ,SZ,EZ]),yZ=new Set(["type","subject","grades","language","xp","url","keywords","learningObjectiveSet","lessonType","subType","questionType","difficulty","format","author","pageCount","duration","speaker","captionsAvailable","launchUrl","toolProvider","instructionalMethod","courseIdOnFail","resolution","resources"]),fZ=Q.record(Q.string(),Q.unknown()).superRefine(($,J)=>{if("fail_fast"in $){let H=T9.safeParse($.fail_fast);if(!H.success)for(let K of H.error.issues)J.addIssue({...K,path:["fail_fast",...K.path]})}let Z=Object.keys($).filter((H)=>yZ.has(H));if(Z.length===0)return;J.addIssue({code:"custom",message:`metadata keys ${Z.map((H)=>`\`${H}\``).join(", ")} require a valid \`type\` discriminator`})}),_6=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),vendorResourceId:X.describe("vendorResourceId must be a non-empty string"),roles:Q.array(Q.enum(["primary","secondary"])).optional(),importance:Q.enum(["primary","secondary"]).optional(),vendorId:X.nullable().optional(),applicationId:X.nullable().optional(),status:f.optional(),metadata:Q.union([vZ,fZ]).nullable().optional()}).strict(),gZ=Q.object({url:X.describe("courseStructure.url must be a non-empty string"),skillCode:X.describe("courseStructure.skillCode must be a non-empty string"),lessonCode:X.describe("courseStructure.lessonCode must be a non-empty string"),title:X.describe("courseStructure.title must be a non-empty string"),"unit-title":X.describe("courseStructure.unit-title must be a non-empty string"),status:X.describe("courseStructure.status must be a non-empty string"),xp:Q.number()}).loose(),B6=Q.object({course:Q.object({sourcedId:X.describe("course.sourcedId must be a non-empty string").optional(),title:X.describe("course.title must be a non-empty string"),org:j,status:f,metadata:v}).strict(),courseStructure:Q.record(Q.string(),gZ)}).strict(),dZ=Q.object({student:j}).loose(),F6=Q.array(dZ).min(1,"results must have at least one item"),mZ=_.enum(["edulastic","mastery-track"]),lZ=_.enum(["powerpath-100","quiz","test-out","placement","unit-test","alpha-read-article"]),d9=_.array(n),m9=_.record(_.string(),_.unknown()).optional(),l9=_.object({courseId:X,lessonTitle:X.optional(),launchUrl:X.optional(),toolProvider:mZ,unitTitle:X.optional(),courseComponentSourcedId:X.optional(),vendorId:X.optional(),description:X.optional(),resourceMetadata:m9.nullable().optional(),grades:d9}),O6=l9.extend({lessonType:_.literal("test-out"),xp:_.number()}),D6=l9.extend({lessonType:_.literal("placement"),courseIdOnFail:X.nullable().optional(),xp:_.number().optional()}),V9=_.object({courseId:X,lessonType:lZ,lessonTitle:X.optional(),unitTitle:X.optional(),courseComponentSourcedId:X.optional(),resourceMetadata:m9.nullable().optional(),xp:_.number().optional(),grades:d9.optional(),courseIdOnFail:X.nullable().optional()}),w6=_.discriminatedUnion("testType",[V9.extend({testType:_.literal("qti"),qti:_.object({url:_.url(),title:X.optional(),metadata:_.record(_.string(),_.unknown()).optional()})}),V9.extend({testType:_.literal("assessment-bank"),assessmentBank:_.object({resources:_.array(_.object({url:_.url(),title:X.optional(),metadata:_.record(_.string(),_.unknown()).optional()})).min(1)})})]),U6=_.object({student:X,lesson:X}),x6=_.object({student:X,lesson:X}),M6=_.object({courseId:X,userId:X,classId:X.optional()}),C2=_.object({type:_.enum(["component","resource"]),id:X}),uZ=_.union([_.object({type:_.literal("set-skipped"),payload:_.object({target:C2,value:_.boolean()})}),_.object({type:_.literal("add-custom-resource"),payload:_.object({resource_id:X,parent_component_id:X,skipped:_.boolean().optional()})}),_.object({type:_.literal("move-item-before"),payload:_.object({target:C2,reference_id:X})}),_.object({type:_.literal("move-item-after"),payload:_.object({target:C2,reference_id:X})}),_.object({type:_.literal("move-item-to-start"),payload:_.object({target:C2})}),_.object({type:_.literal("move-item-to-end"),payload:_.object({target:C2})}),_.object({type:_.literal("change-item-parent"),payload:_.object({target:C2,new_parent_id:X,position:_.enum(["start","end"]).optional()})})]),L6=_.object({operation:uZ,reason:X.optional()}),j6=_.object({studentId:X,componentResourceId:X,result:_.object({status:_.enum(["active","tobedeleted"]),metadata:_.record(_.string(),_.unknown()).optional(),score:_.number().optional(),textScore:X.optional(),scoreDate:_.string().datetime(),scorePercentile:_.number().optional(),scoreStatus:N5,comment:X.optional(),learningObjectiveSet:_.array(_.object({source:X,learningObjectiveResults:_.array(_.object({learningObjectiveId:X,score:_.number().optional(),textScore:X.optional()}))})).optional(),inProgress:X.optional(),incomplete:X.optional(),late:X.optional(),missing:X.optional()})}),V6=_.object({student:X,lesson:X,applicationName:X.optional(),testId:X.optional(),skipCourseEnrollment:_.boolean().optional()}),k6=_.object({student:X,subject:X2}),q6=_.object({student:X,lesson:X}),C6=_.object({userId:X}),A6=_.object({userId:X,subject:_.enum(["Math","Reading","Language","Science"])}),z6=_.object({student:X,subject:X2,grade:n,testName:X.optional()}),P6=_.object({testName:X}),cZ=_.object({student:X,subject:X2,grade:n,testName:X.optional()}),T6=_.object({items:_.array(cZ)}),R6=_.object({spreadsheetUrl:_.url(),sheet:X}),N6=_.object({student:X,status:_.enum(["assigned","in_progress","completed","failed","expired","cancelled"]).optional(),subject:X.optional(),grade:n.optional(),limit:_.number().int().positive().max(3000).optional(),offset:_.number().int().nonnegative().optional()}),I6=_.object({student:X.optional(),status:_.enum(["assigned","in_progress","completed","failed","expired","cancelled"]).optional(),subject:X.optional(),grade:n.optional(),limit:_.number().int().positive().max(3000).optional(),offset:_.number().int().nonnegative().optional()}),b6=_.object({student:X,question:X,response:_.union([X,_.array(X)]).optional(),responses:_.record(_.string(),_.union([X,_.array(X)])).optional(),lesson:X,rendererOutcomes:_.object({score:_.number(),maxScore:_.number().min(0),isCorrect:_.boolean()}).optional(),playerState:_.string().optional()}).refine(($)=>$.response!==void 0||$.responses!==void 0,{message:"Either 'response' or 'responses' must be provided",path:["response","responses"]}).transform(($)=>{if($.response!==void 0&&$.responses===void 0)return{...$,responses:{RESPONSE:$.response}};return $}),h6=_.object({student:X,lesson:X,attempt:_.number().int().positive().optional()}),S6=_.object({student:X,lesson:X}),E6=_.object({student:X,lesson:X}),v6=_.object({student:X,course:X}),y6=_.object({student:X,lesson:X,applicationName:X.optional()}),f6=_.object({student:X,subject:X2}),g6=_.object({oneRosterSourcedId:X,subject:X}),d6=_.object({courseIds:_.array(X).min(1),rendererId:X,rendererUrl:_.url(),rendererVersion:X.optional(),suppressFeedback:_.boolean().optional(),suppressCorrectResponse:_.boolean().optional()}),m6=_.object({status:_.enum(["active","tobedeleted"]).optional()}),u9=G.enum(["choice","text-entry","extended-text","inline-choice","match","order","associate","select-point","graphic-order","graphic-associate","graphic-gap-match","hotspot","hottext","slider","drawing","media","upload"]),b8=G.enum(["single","multiple","ordered","record"]),h8=G.enum(["identifier","boolean","integer","float","string","point","pair","directedPair","duration","file","uri"]),pZ=G.enum(["easy","medium","hard"]),aZ=G.enum(["linear","nonlinear"]),iZ=G.enum(["individual","simultaneous"]),S8=G.enum(["show","hide"]),c9=G.enum(["test","item","stimulus"]),l6=G.enum(["QUESTION","LESSON"]),sZ=G.object({value:G.array(G.string())}).strict(),p9=G.object({identifier:X,cardinality:b8,baseType:h8.optional(),correctResponse:sZ}).strict(),a9=G.object({identifier:X,cardinality:b8,baseType:h8.optional()}).strict(),i9=G.object({identifier:X,cardinality:b8.optional(),baseType:h8,normalMaximum:G.number().optional(),normalMinimum:G.number().optional(),defaultValue:G.object({value:G.unknown().optional()}).strict().optional()}).strict(),oZ=G.object({outcomeIdentifier:X,variableIdentifier:X}).strict(),s9=G.object({templateType:G.enum(["match_correct","map_response"]),responseDeclarationIdentifier:X,outcomeIdentifier:X,correctResponseIdentifier:X,incorrectResponseIdentifier:X,inlineFeedback:oZ.optional()}).strict(),nZ=G.object({source:X,learningObjectiveIds:G.array(G.string())}).strict(),E8=G.object({subject:G.string().optional(),grade:n.optional(),difficulty:pZ.optional(),learningObjectiveSet:G.array(nZ).optional()}).loose(),o9=G.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:G.string(),title:G.string()}).strict(),n9=G.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:G.string(),class:G.array(G.string())}).strict(),r9=G.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:G.string(),class:G.array(G.string())}).strict(),t9=G.object({href:X,type:X}).strict(),e9=G.object({id:X,support:G.string(),content:G.string()}).strict(),u6=G.object({page:G.number().int().positive().optional(),limit:G.number().int().positive().optional(),sort:G.string().optional(),order:G.enum(["asc","desc"]).optional()}).strict(),rZ=G.object({format:G.string().pipe(G.literal("xml")),xml:X,metadata:E8.optional()}).strict(),tZ=G.object({identifier:X,title:X,type:u9,qtiVersion:G.string().optional(),timeDependent:G.boolean().optional(),adaptive:G.boolean().optional(),responseDeclarations:G.array(p9).optional(),outcomeDeclarations:G.array(a9).optional(),responseProcessing:s9.optional(),metadata:E8.optional(),modalFeedback:G.array(o9).optional(),feedbackInline:G.array(n9).optional(),feedbackBlock:G.array(r9).optional()}).strict(),c6=G.union([rZ,tZ]),p6=G.object({identifier:X.optional(),title:X,type:u9,qtiVersion:G.string().optional(),timeDependent:G.boolean().optional(),adaptive:G.boolean().optional(),responseDeclarations:G.array(p9).optional(),outcomeDeclarations:G.array(a9).optional(),responseProcessing:s9.optional(),metadata:E8.optional(),modalFeedback:G.array(o9).optional(),feedbackInline:G.array(n9).optional(),feedbackBlock:G.array(r9).optional(),rawXml:G.string(),content:G.record(G.string(),G.unknown())}).strict(),a6=G.object({identifier:X,response:G.union([G.string(),G.array(G.string())])}).strict(),$$=G.object({identifier:X,href:X,sequence:G.number().int().positive().optional()}).strict(),eZ=G.object({identifier:X,title:X,visible:G.boolean(),required:G.boolean().optional(),fixed:G.boolean().optional(),sequence:G.number().int().positive(),"qti-assessment-item-ref":G.array($$).optional()}).strict(),J$=G.object({identifier:X,navigationMode:G.string().pipe(aZ),submissionMode:G.string().pipe(iZ),"qti-assessment-section":G.array(eZ)}).strict(),i6=G.object({items:G.array($$).min(1)}).strict(),s6=G.object({metadata:G.record(G.string(),G.unknown()).optional()}).strict(),o6=G.object({identifier:X,title:X,qtiVersion:G.string().optional(),toolName:G.string().optional(),toolVersion:G.string().optional(),timeLimit:G.number().optional(),maxAttempts:G.number().optional(),toolsEnabled:G.record(G.string(),G.boolean()).optional(),metadata:G.record(G.string(),G.unknown()).optional(),"qti-test-part":G.array(J$),"qti-outcome-declaration":G.array(i9).optional()}).strict(),n6=G.object({identifier:X.optional(),title:X,qtiVersion:G.string().optional(),toolName:G.string().optional(),toolVersion:G.string().optional(),timeLimit:G.number().optional(),maxAttempts:G.number().optional(),toolsEnabled:G.record(G.string(),G.boolean()).optional(),metadata:G.record(G.string(),G.unknown()).optional(),"qti-test-part":G.array(J$),"qti-outcome-declaration":G.array(i9).optional()}).strict(),r6=G.object({identifier:X,title:X,label:G.string().optional(),language:G.string().optional(),stylesheet:t9.optional(),content:G.string(),catalogInfo:G.array(e9).optional(),toolName:G.string().optional(),toolVersion:G.string().optional(),metadata:G.record(G.string(),G.unknown()).optional()}).strict(),t6=G.object({identifier:X.optional(),title:X,label:G.string().optional(),language:G.string().optional(),stylesheet:t9.optional(),content:G.string(),catalogInfo:G.array(e9).optional(),toolName:G.string().optional(),toolVersion:G.string().optional(),metadata:G.record(G.string(),G.unknown()).optional()}).strict(),e6=G.object({xml:G.string().optional(),schema:c9,entityId:G.string().optional()}).strict(),$7=G.object({xml:G.array(G.string()),schema:c9,entityIds:G.array(G.string())}).strict(),J7=G.object({questionId:G.string().optional(),userId:X,feedback:X,lessonId:X,humanApproved:G.boolean().optional()}).strict(),$0=z2.enum(["creator_only","same_org","all_orgs"]),J0=z2.string().uuid(),Z7=z2.email(),H7=z2.object({sql:X}),K7=z2.object({endpointName:X,sql:X,authMode:$0,description:X.optional()}),X7=z2.object({endpointId:J0});class Z${transport;eventTransformer;constructor($,J){this.transport=$,this.eventTransformer=J?.eventTransformer}send($,J){o(s5,{sensor:$,events:J},"send events");let Z={sensor:$,sendTime:new Date().toISOString(),dataVersion:"http://purl.imsglobal.org/ctx/caliper/v1p2",data:J};return this.sendEnvelope(Z)}async sendEnvelope($){o(U9,$,"caliper envelope"),S.debug("Sending Caliper envelope",{sensor:$.sensor,eventCount:$.data.length});let J=this.eventTransformer?this.eventTransformer.transformEnvelope($):$,H=(await this.transport.request(this.transport.paths.send,{method:"POST",body:J}))?.jobId;return S.debug("Events sent",{jobId:H??"(none)"}),{jobId:H}}async validate($){let J=this.transport.paths.validate;if(!J)throw Error("validate() is not supported on this platform");o(U9,$,"caliper envelope"),S.debug("Validating Caliper envelope",{sensor:$.sensor,eventCount:$.data.length});let Z=await this.transport.request(J,{method:"POST",body:$});return S.debug("Validation complete",{status:Z.status}),Z}async list($={}){let J=this.transport.paths.list;if(!J)throw Error("list() is not supported on this platform");o(x9,$,"list events params"),S.debug("Listing events",{params:$});let Z={limit:$.limit??100,offset:$.offset??0};if($.sensor)Z.sensor=$.sensor;if($.startDate)Z.startDate=$.startDate;if($.endDate)Z.endDate=$.endDate;if($.actorId)Z.actorId=$.actorId;if($.actorEmail)Z.actorEmail=$.actorEmail;let H=await this.transport.request(J,{params:Z});return{events:H.events??[],pagination:H.pagination}}stream($={}){let J=this.transport.paths.list;if(!J)throw Error("stream() is not supported on this platform");let{max:Z,...H}=$;o(x9,H,"list events params"),P8($),S.debug("Streaming events",{params:$});let K={};if(H.limit!==void 0)K.limit=H.limit;if(H.offset!==void 0)K.offset=H.offset;if(H.sensor)K.sensor=H.sensor;if(H.startDate)K.startDate=H.startDate;if(H.endDate)K.endDate=H.endDate;if(H.actorId)K.actorId=H.actorId;if(H.actorEmail)K.actorEmail=H.actorEmail;return new q9(this.transport,J,{...K,max:Z})}async get($){let J=this.transport.paths.get;if(!J)throw Error("get() is not supported on this platform");c2($,"externalId"),S.debug("Getting event",{externalId:$});let Z=J.replace("{id}",encodeURIComponent($)),H=await this.transport.request(Z);if(!H.event)throw Error(`Event not found: ${$}`);return H.event}sendActivity($,J){o(c5,J,"activity event");let Z=T8(J);return S.debug("Sending ActivityCompletedEvent",{eventId:Z.id,actor:J.actor.email,subject:J.object.subject,metricsCount:J.metrics.length}),this.send($,[Z])}sendTimeSpent($,J){o(p5,J,"time spent event");let Z=R8(J);return S.debug("Sending TimeSpentEvent",{eventId:Z.id,actor:J.actor.email,subject:J.object.subject,metricsCount:J.metrics.length}),this.send($,[Z])}sendQuestionSeen($,J){o(r5,J,"question seen event");let Z=w5(J);return S.debug("Sending AssessmentItemEvent.Started",{eventId:Z.id,actor:J.actor,object:J.object.id}),this.send($,[Z])}sendQuestionAnswered($,J){o(t5,J,"question answered event");let Z=U5(J);return S.debug("Sending AssessmentItemEvent.Completed",{eventId:Z.id,actor:J.actor,object:J.object.id}),this.send($,[Z])}sendQuestionGraded($,J){o(e5,J,"question graded event");let Z=x5(J);return S.debug("Sending GradeEvent.Graded",{eventId:Z.id,actor:J.actor,object:J.object,scoreGiven:J.generated.scoreGiven}),this.send($,[Z])}}class H${transport;constructor($){this.transport=$}async getStatus($){let J=this.transport.paths.jobStatus;if(!J)throw Error("getStatus() is not supported on this platform");c2($,"jobId"),S.debug("Getting job status",{jobId:$});let Z=J.replace("{id}",encodeURIComponent($));return(await this.transport.request(Z)).job}async waitForCompletion($,J={}){c2($,"jobId"),Z0(J);let{timeoutMs:Z=30000,pollIntervalMs:H=1000}=J,K=Date.now();S.debug("Waiting for job completion",{jobId:$,timeoutMs:Z,pollIntervalMs:H});while(Date.now()-K<Z){let W=await this.getStatus($);if(W.state==="completed")return S.info("Job completed",{jobId:$}),W;if(W.state==="failed")throw S.error("Job failed",{jobId:$}),Error(`Job ${$} failed`);await new Promise((B)=>{setTimeout(B,H)})}throw Error(`Job ${$} timed out after ${Z}ms`)}}function Z0($){if(!$)return;let J=[];if($.timeoutMs!==void 0){if(!Number.isInteger($.timeoutMs)||$.timeoutMs<=0)J.push({path:"timeoutMs",message:"Must be a positive integer"})}if($.pollIntervalMs!==void 0){if(!Number.isInteger($.pollIntervalMs)||$.pollIntervalMs<=0)J.push({path:"pollIntervalMs",message:"Must be a positive integer"})}if(J.length>0)throw I2("Invalid polling options",J)}function H0($=u2){return class{transport;_provider;events;jobs;constructor(Z={}){let H=z5(Z,$),K;if(H.mode==="transport")this.transport=H.transport,S.info("Client initialized with custom transport");else{let{provider:W}=H,{baseUrl:B,paths:F}=W.getEndpointWithPaths("caliper"),O=W.getTokenProvider("caliper");if(!O)throw Error("Caliper API requires authentication");if(this._provider=W,this.transport=new k9({baseUrl:B,tokenProvider:O,timeout:W.timeout,paths:F}),W.platform===X9)K=new C9;S.info("Client initialized",{platform:W.platform,env:W.env,baseUrl:B})}this.events=new Z$(this.transport,{eventTransformer:K}),this.jobs=new H$(this.transport)}getTransport(){return this.transport}checkAuth(){if(!this._provider)throw Error("Cannot check auth: client initialized with custom transport");return this._provider.checkAuth()}}}var W7=H0();class W2 extends Error{course;env;constructor($,J){let Z=$.grade===void 0?$.courseCode??$.subject:`${$.subject} grade ${$.grade}`;super(`Course "${Z}" is missing a synced ID for ${J}. Run \`timeback resources push\` first.`);this.name="MissingSyncedCourseIdError",this.course=$,this.env=J}}class i2 extends Error{sensor;constructor($){super(`Invalid sensor URL "${$}". Sensor must be a valid absolute URL (e.g., "https://sensor.example.com") to support slug-based activity IDs.`);this.name="InvalidSensorUrlError",this.sensor=$}}var K$="/ims/oneroster/rostering/v1p2";function X$($,J){return`${$.replace(/\/+$/,"")}/${J.replace(/^\/+/,"")}`}function K0($,J){return X$($,`${K$}/courses/${J}`)}function X0($,J){return X$($,`${K$}/users/${J}`)}function W$($,J){let Z=$.ids?.[J];if(!Z)throw new W2($,J);return Z}function W0($,J,Z){let H=W$(J,Z);return K0($,H)}function Y0($){if($.courseCode)return $.courseCode;if($.grade!==void 0)return`${$.subject} G${String($.grade)}`;return $.subject}function v8($,J,Z){let H;try{H=new URL($)}catch{throw new i2($)}let K="grade"in J?`${J.subject}/g${String(J.grade)}`:J.code,W=H.pathname.replace(/\/+$/,"");return H.pathname=`${W}/activities/${K}/${encodeURIComponent(Z)}`,H.toString()}function Y$($,J,Z,H,K,W,B){return{id:v8(K,$.course,$.id),type:"TimebackActivityContext",subject:J.subject,app:{name:Z},activity:{name:$.name},course:{id:W0(W,J,H),name:Y0(J)},process:B??!0}}function Q0($){let J=[];if($.totalQuestions!==void 0)J.push({type:"totalQuestions",value:$.totalQuestions});if($.correctQuestions!==void 0)J.push({type:"correctQuestions",value:$.correctQuestions});if($.xpEarned!==void 0)J.push({type:"xpEarned",value:$.xpEarned});if($.masteredUnits!==void 0)J.push({type:"masteredUnits",value:$.masteredUnits});return J}function G0($,J){let Z=[{type:"active",value:Math.max(0,$)/1000}];if(J>0)Z.push({type:"inactive",value:Math.max(0,J)/1000});return Z}function _0($){return $===void 0?void 0:{pctCompleteApp:$}}function Q$($,J){return{courseId:W$($,J)}}function G$($){if(!$)return;return`urn:uuid:${$}`}function _$($,J,Z){return{id:X0($,J),type:"TimebackUser",email:Z}}function B$($){let{sensor:J,timebackId:Z,email:H,payload:K,process:W,course:B,appName:F,apiEnv:O,onerosterBaseUrl:U,attempt:V,runId:A}=$,C=_$(U,Z,H),q=Y$(K,B,F,O,J,U,W??K.process),z=Q0(K.metrics),N=_0(K.pctComplete),I=G$(A),r=Q$(B,O),Y2=T8({actor:C,object:q,metrics:z,eventTime:K.endedAt,generatedExtensions:N,extensions:r,attempt:V,session:I});return{sensor:J,actor:C,object:q,event:Y2,payload:K,course:B,appName:F,apiEnv:O,email:H,timebackId:Z}}function s2($){let{sensor:J,timebackId:Z,email:H,payload:K,course:W,appName:B,apiEnv:F,onerosterBaseUrl:O,runId:U}=$,V=_$(O,Z,H),A=Y$(K,W,B,F,J,O),C=G0(K.elapsedMs,K.pausedMs),q=Q$(W,F),z=G$(U),N=R8({actor:V,object:A,metrics:C,eventTime:K.endedAt,extensions:q,session:z});return{sensor:J,actor:V,object:A,event:N,payload:{id:K.id,name:K.name,course:K.course,startedAt:K.startedAt,endedAt:K.endedAt,elapsedMs:K.elapsedMs,pausedMs:K.pausedMs},course:W,appName:B,apiEnv:F,email:H,timebackId:Z,runId:U}}import*as h from"zod";import*as h2 from"zod";var B0=h2.object({subject:F$,grade:O$}),F0=h2.object({code:E}),M2=h2.union([B0,F0]);class _2 extends Error{code;selector;count;constructor($,J,Z){super($);this.code=$,this.selector=J,this.count=Z}get selectorDescription(){if("grade"in this.selector)return`${this.selector.subject} grade ${this.selector.grade}`;return`code "${this.selector.code}"`}}function L2($,J){let Z;if("grade"in J)Z=$.filter((H)=>H.subject===J.subject&&H.grade===J.grade);else Z=$.filter((H)=>H.courseCode===J.code);if(Z.length===0)throw new _2("unknown_course",J);if(Z.length>1)throw new _2("ambiguous_course",J,Z.length);return Z[0]}var P2=R("handlers:activity:schema"),O0=h.object({totalQuestions:h.number().int().nonnegative().optional(),correctQuestions:h.number().int().nonnegative().optional(),xpEarned:h.number(),masteredUnits:h.number().int().nonnegative().optional()}).superRefine(n2);function D0($){return Math.min(100,Math.max(0,$))}function o2($){if("grade"in $)return`${$.subject} grade ${$.grade}`;return`code "${$.code}"`}var w0=h.object({id:E,name:E,course:M2,runId:h.string().uuid(),startedAt:h.iso.datetime(),endedAt:h.iso.datetime(),elapsedMs:h.number().int().nonnegative(),pausedMs:h.number().int().nonnegative()});function D$($,J,Z){let H=w0.safeParse($);if(!H.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:H.error.flatten()})};let K=H.data,W;try{W=L2(J.courses,K.course)}catch(F){if(F instanceof _2){let O=o2(K.course);if(F.code==="unknown_course")return P2.warn("Unknown course selector",{selector:K.course}),{ok:!1,response:M("UNKNOWN_COURSE",`Unknown course: ${O}`,400)};return P2.error("Ambiguous course selector",{selector:K.course}),{ok:!1,response:M("AMBIGUOUS_COURSE_SELECTOR","Ambiguous course selector in timeback.config.json",500)}}throw F}let B=S2(J,W,Z);if(!B){let F=o2(K.course);return P2.error("Missing sensor for course",{selector:K.course}),{ok:!1,response:M("MISSING_SENSOR",`Course "${F}" has no sensor configured.`,500)}}return{ok:!0,payload:K,course:W,sensor:B}}var U0=h.object({id:E,name:E,course:M2,process:h.boolean().optional(),runId:h.string().uuid(),endedAt:h.iso.datetime(),metrics:O0,pctComplete:h.number().optional().transform(($)=>$===void 0?void 0:D0($))});function w$($,J,Z){let H=U0.safeParse($);if(!H.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:H.error.flatten()})};let K=H.data,W;try{W=L2(J.courses,K.course)}catch(F){if(F instanceof _2){let O=o2(K.course);if(F.code==="unknown_course")return P2.warn("Unknown course selector",{selector:K.course}),{ok:!1,response:M("UNKNOWN_COURSE",`Unknown course: ${O}`,400)};return P2.error("Ambiguous course selector",{selector:K.course}),{ok:!1,response:M("AMBIGUOUS_COURSE_SELECTOR","Ambiguous course selector in timeback.config.json",500)}}throw F}let B=S2(J,W,Z);if(!B){let F=o2(K.course);return P2.error("Missing sensor for course",{selector:K.course}),{ok:!1,response:M("MISSING_SENSOR",`Course "${F}" has no sensor configured.`,500)}}return{ok:!0,payload:K,course:W,sensor:B}}import{getServiceUrlsForEnv as k$}from"@timeback/core";var y8=R("handlers:activity:attempts");async function U$($,J){let Z=[$,J].join("_");return`caliper_${await E2(Z)}`}function x0($,J){return $.find((Z)=>Z.scoreDate===J)}function M0($){let J=0;for(let Z of $){let K=Z.metadata?.attempt;if(typeof K==="number"&&K>J)J=K}return J}function L0($){let J=$.metadata;return typeof J?.attempt==="number"&&J.attempt>=1?J.attempt:1}async function x$($,J,Z,H){let K;try{K=await $.oneroster.assessmentResults.listAll({where:{status:"active","assessmentLineItem.sourcedId":J,"student.sourcedId":Z}})}catch{return y8.debug("No existing results found (line item may not exist yet)",{lineItemId:J}),1}if(K.length===0)return 1;let W=x0(K,H);if(W){let O=L0(W);return y8.debug("Retry detected, reusing attempt number",{lineItemId:J,timebackId:Z,endedAt:H,attempt:O}),O}let B=M0(K),F=B+1;return y8.debug("New attempt computed",{lineItemId:J,timebackId:Z,endedAt:H,maxAttempt:B,nextAttempt:F}),F}var M$=R("handlers:activity:completion");async function j$($){let{client:J,courseId:Z,timebackId:H,pctComplete:K,appName:W}=$;if(K!==100)return;let B=L$(Z),F=`timeback_sdk_${await E2(`mastery-completion_${Z}`)}`,O=`timeback_sdk_${await E2(`mastery-completion_${Z}_${H}`)}`;try{let U=!1;try{await J.oneroster.assessmentLineItems(F).get(),U=!0}catch{}if(!U)await J.oneroster.assessmentLineItems.create({sourcedId:F,title:`${W}: Complete`,status:"active",course:{sourcedId:B.course},componentResource:{sourcedId:B.componentResource},resultValueMin:0,resultValueMax:100,metadata:{appName:W}});let V=new Date().toISOString();await J.oneroster.assessmentResults.upsert(O,{status:"active",assessmentLineItem:{sourcedId:F},student:{sourcedId:H},score:100,scoreDate:V,scoreStatus:"fully graded",metadata:{isMasteryCompletion:!0,completedAt:V,appName:W}}),M$.debug("Created mastery completion entry",{courseId:Z,timebackId:H,lineItemId:F,resultId:O})}catch(U){let V=U instanceof Error?U.message:"Unknown error";M$.error("Failed to create mastery completion entry",{courseId:Z,timebackId:H,lineItemId:F,error:V})}}import{aggregateActivityMetrics as j0}from"@timeback/core/utils";var T2=R("handlers:activity:progress");function V0($,J){let Z=u(J),H=$.overrides?.[Z]?.metadata?.metrics,K=$.metadata?.metrics;return H?.totalLessons??K?.totalLessons}function k0($,J){let Z=$/J*100;return Math.min(100,Math.max(0,Math.round(Z)))}async function V$($){let{client:J,courseId:Z,timebackId:H,payload:K,course:W,env:B}=$;if(K.pctComplete!==void 0)return;let F=K.metrics.masteredUnits;if(typeof F!=="number"||F<=0)return;let O=V0(W,B);if(!O||O<=0){T2.debug("Skipping progress computation: totalLessons not configured",{courseId:Z});return}let U;try{U=(await J.edubridge.enrollments.list({userId:H})).find((N)=>N.course.id===Z)?.id}catch(z){let N=z instanceof Error?z.message:"Unknown error";T2.warn("Failed to fetch enrollments for progress computation",{courseId:Z,timebackId:H,error:N});return}if(!U){T2.warn("Skipping progress computation: enrollment not found for student/course",{courseId:Z,timebackId:H});return}let V=0;try{let z=await J.edubridge.analytics.getEnrollmentFacts({enrollmentId:U});V=j0(z).masteredUnits}catch(z){let N=z instanceof Error?z.message:"Unknown error";T2.warn("Failed to fetch enrollment facts for progress computation",{courseId:Z,timebackId:H,enrollmentId:U,error:N});return}let A=!0;try{if((await J.edubridge.analytics.getWeeklyFacts({studentId:H,weekDate:K.endedAt})).some((I)=>{if(I.activityId!==K.id)return!1;if(I.enrollmentId&&I.enrollmentId===U)return!0;if(I.courseId&&I.courseId===Z)return!0;return!1}))A=!1}catch(z){let N=z instanceof Error?z.message:"Unknown error";T2.warn("Failed to fetch weekly facts for retry-safe progress computation",{courseId:Z,timebackId:H,enrollmentId:U,activityId:K.id,error:N}),A=!1}let C=V+(A?F:0),q=k0(C,O);return T2.debug("Computed pctComplete",{courseId:Z,timebackId:H,enrollmentId:U,historicalMasteredUnits:V,currentMasteredUnits:F,shouldIncludeCurrentMasteredUnits:A,totalMastered:C,totalLessons:O,pctComplete:q}),{pctComplete:q}}var f8=R("handlers:activity:submit"),q0={computeProgress:V$,maybeWriteCompletionEntry:j$};function q$($,J){return $.timebackId??B2({email:$.email,client:J})}async function r2($,J,Z=q0){let H=u($.env),{userInfo:K,payload:W,course:B,sensor:F,preview:O=!1,runId:U}=J,V=B.ids?.[H];if(!V)throw new W2(B,H);let A=$.getClient(),C=await q$(K,A),q=W;if(W.pctComplete===void 0){let f2=await Z.computeProgress({client:A,courseId:V,timebackId:C,payload:W,course:B,env:$.env});if(f2)q={...W,pctComplete:f2.pctComplete}}let z;if(!O){let f2=v8(F,q.course,q.id),UJ=await U$(f2,V);z=await x$(A,UJ,C,q.endedAt)}let I=k$(H).oneroster,r=B$({sensor:F,timebackId:C,email:K.email,payload:q,process:q.process,course:B,appName:$.appConfig.name,apiEnv:H,onerosterBaseUrl:I,attempt:z,runId:U}),Y2=$.hooks?.beforeActivitySend,y2=Y2?await Y2(r):r,p=y2??r;if(O||y2===null)return{success:!0,preview:!0,sensor:p.sensor,actor:p.actor,object:p.object,event:p.event};return await A.caliper.events.send(p.sensor,[p.event]),await Z.maybeWriteCompletionEntry({client:A,courseId:V,timebackId:p.timebackId,pctComplete:p.payload.pctComplete,appName:p.appName}),f8.debug("Recorded completion",{courseSelector:W.course,activityId:W.id}),{success:!0}}async function C$($,J){let Z=u($.env),{userInfo:H,payload:K,course:W,sensor:B,runId:F}=J;if(!W.ids?.[Z])throw new W2(W,Z);let U=$.getClient(),V=await q$(H,U),C=k$(Z).oneroster,q=s2({sensor:B,timebackId:V,email:H.email,payload:K,course:W,appName:$.appConfig.name,apiEnv:Z,onerosterBaseUrl:C,runId:F}),z=$.hooks?.beforeTimeSpentSend,N=z?await z(q):q,I=N??q;if(N===null){f8.debug("Hook skipped time-spent send",{activityId:K.id});return}await U.caliper.events.send(I.sensor,[I.event]),f8.debug("Sent time-spent event",{activityId:K.id,elapsedMs:K.elapsedMs})}var v2=R("handlers:activity:submit");async function R2($,J){if($.mode==="custom"){let H=await $.getEmail(J);return H?{email:H}:void 0}let Z=await $.getUser(J);return Z?{email:Z.email,timebackId:Z.id}:void 0}function t2($){return async(J)=>{let Z=await R2($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let H=w$(await J.json(),$.appConfig,$.env);if(!H.ok)return H.response;let{payload:K,course:W,sensor:B}=H,F={id:K.id,name:K.name,course:K.course,process:K.process,endedAt:K.endedAt,metrics:K.metrics,pctComplete:K.pctComplete};try{if((await r2({env:$.env,api:$.api,appConfig:$.appConfig,getClient:$.getClient,hooks:$.hooks},{userInfo:Z,payload:F,course:W,sensor:B,preview:!1,runId:K.runId})).preview)return P({preview:!0});return v2.debug("Submitted activity",{runId:K.runId,activityId:K.id}),new Response(null,{status:204})}catch(O){if(O instanceof d)return v2.warn("Failed to resolve Timeback user",{code:O.code}),M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",F2(O));if(O instanceof W2)return v2.warn("Course not synced",{course:O.course,env:O.env}),M("MISSING_SYNCED_COURSE_ID",O.message,503);if(O instanceof i2)return v2.error("Invalid sensor URL",{sensor:O.sensor}),M("INVALID_SENSOR_URL",O.message,500);let U=O instanceof Error?O.message:"Unknown error";return v2.error("Failed to submit activity",{error:U}),M("INTERNAL_ERROR",U,502)}}}var N2=R("handlers:activity:heartbeat"),e2=new Map,A0=300000;function z0(){let $=Date.now();for(let[J,Z]of e2)if($-Z>A0)e2.delete(J)}function P0($){return z0(),e2.has($)}function A$($){e2.set($,Date.now())}function $8($){return async(J)=>{let Z=await R2($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let H=D$(await J.json(),$.appConfig,$.env);if(!H.ok)return H.response;let{payload:K,course:W,sensor:B}=H,F=new Date(K.startedAt).getTime(),O=`${K.runId}:${F}`;if(P0(O))return N2.debug("Duplicate heartbeat, skipping",{runId:K.runId,windowKey:O}),P({duplicate:!0});let U=u($.env);if(!W.ids?.[U]){let C=new W2(W,U);return N2.warn("Course not synced",{course:K.course,env:U}),M("MISSING_SYNCED_COURSE_ID",C.message,503)}let A=$.getClient();try{let C=Z.timebackId??await B2({email:Z.email,client:A}),z=C0(U).oneroster,N=s2({sensor:B,timebackId:C,email:Z.email,payload:K,course:W,appName:$.appConfig.name,apiEnv:U,onerosterBaseUrl:z,runId:K.runId}),I=$.hooks?.beforeTimeSpentSend,r=I?await I(N):N,Y2=r??N;if(r===null)return A$(O),N2.debug("Hook skipped heartbeat send",{runId:K.runId}),new Response(null,{status:204});return await A.caliper.events.send(Y2.sensor,[Y2.event]),A$(O),N2.debug("Sent heartbeat",{runId:K.runId,activityId:K.id,elapsedMs:K.elapsedMs}),new Response(null,{status:204})}catch(C){if(C instanceof d)return N2.warn("Failed to resolve Timeback user",{code:C.code}),M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",F2(C));let q=C instanceof Error?C.message:"Unknown error";return N2.error("Failed to send heartbeat",{error:q}),M("INTERNAL_ERROR",q,502)}}}async function E$($,J,Z){if(Z.mode!=="sso")return j2.warn("SSO not configured"),P({error:"SSO not configured"},400);return await b$({req:$,env:J,clientId:Z.clientId,issuer:Z.issuer,redirectUri:Z.redirectUri,buildState:Z.buildState})}async function T0($,J,Z,H,K,W,B){try{let F=W.issuer??z$(K),O=S$(J,W.redirectUri);j2.debug("Exchanging auth code for tokens",{issuer:F,clientId:W.clientId});let U=await P$({issuer:F,clientId:W.clientId,clientSecret:W.clientSecret,code:$,redirectUri:O}),V=await T$({issuer:F,accessToken:U.access_token}),A=typeof V.identities==="string"?JSON.parse(V.identities):V.identities;j2.debug("SSO completed, resolving Timeback user",{user:{...V,identities:A}});let C=await O2({env:K,apiCredentials:B.credentials,userInfo:V,client:B.getClient()});j2.debug("Timeback user resolved",{timebackId:C.id});let q={user:C,idp:{tokens:U,userInfo:V},state:Z,req:H,redirect:J8,json:P};return W.onCallbackSuccess(q)}catch(F){let O=F instanceof Error?F:Error("Unknown error"),U=F instanceof d?F.code:"token_exchange_failed";if(j2.error("SSO callback failed",{error:O.message,errorCode:U}),W.onCallbackError)return W.onCallbackError(R$(O,U,Z,H));if(F instanceof d)return P({error:O.message},F2(F));return P({error:O.message},500)}}async function v$($,J,Z,H){if(Z.mode!=="sso")return j2.warn("SSO not configured"),P({error:"SSO not configured"},400);let{url:K,code:W,errorParam:B,state:F}=h$($);if(B)return N$(B,K,F,$,Z.onCallbackError);if(!W)return I$(F,$,Z.onCallbackError);return await T0(W,K,F,$,J,Z,H)}function Z8($){let{env:J,identity:Z}=$;return(H)=>E$(H,J,Z)}function H8($){let{env:J,identity:Z,api:H}=$;return(K)=>v$(K,J,Z,H)}function K8(){return()=>J8("/")}var y$=($)=>{return async(J)=>{return await $.getClient().powerpath.assessments.getAssessmentProgress({student:J.student,lesson:J.lesson,attempt:J.attempt})}};var f$=($)=>{return async(J)=>{return(await $.getClient().powerpath.assessments.getAttempts({student:J.student,lesson:J.lesson})).attempts.map((H)=>({attempt:H.attempt??void 0,score:H.score,startedAt:H.startedAt??void 0,completedAt:H.completedAt??void 0,scoreStatus:H.scoreStatus}))}};var g$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments,H=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson}),K;if(H.lessonType!=="powerpath-100"&&H.finalized!==!0){try{K=await Z.finalStudentAssessmentResponse({student:J.student,lesson:J.lesson})}catch{}H=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson})}let W=H.lessonType==="powerpath-100"?!0:H.finalized===!0;return{...K??{},score:H.score??0,totalQuestions:H.totalQuestions,correctQuestions:H.correctQuestions,accuracy:H.accuracy,finalized:W}}};function d$($){let J=$.metadata?.questionCount;return typeof J==="number"?J:void 0}function X8($){return{id:$.id,title:$.title,difficulty:$.difficulty,content:$.content?{rawXml:$.content.rawXml}:void 0,index:$.index,url:$.url,humanApproved:$.humanApproved,response:$.response,responses:$.responses,correct:$.correct,result:$.result,resultId:$.resultId,learningObjectives:$.learningObjectives}}function m$($,J){if(!J)return{lessonId:$,lessonType:"quiz",score:0,seenQuestions:0,finalized:!1};return{lessonId:$,lessonType:J.lessonType,attempt:J.attempt,score:J.score??0,questionCount:J.totalQuestions,seenQuestions:J.lessonType==="powerpath-100"?J.seenQuestions.length:0,finalized:J.lessonType==="powerpath-100"?!1:J.finalized}}function l$($){let J=$.questions,Z=J.filter((H)=>H.response!==void 0).map((H)=>H.id);return{questions:J.map(X8),answeredIds:Z,score:$.score??0,finalized:$.finalized,complete:$.finalized}}function R0($){if($.lessonType==="powerpath-100")return $.responseResult.isCorrect;return!1}function N0($){if($.lessonType==="powerpath-100")return $.powerpathScore;return 0}function u$($){let J=N0($);return{correct:R0($),score:J,complete:$.lessonType==="powerpath-100"&&J>=100,questionResult:$.questionResult}}function c$($,J){return Array.from(new Set($.map((Z)=>Z.ids?.[J]).filter((Z)=>Boolean(Z))))}function p$($){return String($.sourcedId??"")}var a$=($)=>{return async(J={})=>{let Z=$.getClient(),H=u($.env),K;if(J.course){let F=L2($.appConfig.courses,J.course).ids?.[H];K=F?[F]:[]}else K=c$($.appConfig.courses,H);let W=[];for(let B of K){let F=await Z.oneroster.courses(B).components({where:{status:"active"}}),O=g8(F);for(let U of O){let V=p$(U);if(!V)continue;let A=await Z.oneroster.courses.componentResources({where:{"courseComponent.sourcedId":V,status:"active"}}),C=g8(A);for(let q of C){if(!q.sourcedId)continue;let z=q.metadata??{},N=d$(q);W.push({id:q.sourcedId,name:q.title??q.sourcedId,type:z.lessonType??"quiz",courseId:B,questionCount:N,metadata:z})}}}return W}};class W8 extends Error{name="LessonTypeMismatchError"}var i$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments;if(J.lessonType==="powerpath-100"){let K=await Z.getNextQuestion({student:J.student,lesson:J.lesson});return X8(K.question)}let H=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson});if(H.lessonType==="powerpath-100"){if(J.lessonType!==void 0)throw new W8(`Expected quiz progress but got powerpath-100 for lesson ${J.lesson}`);let K=await Z.getNextQuestion({student:J.student,lesson:J.lesson});return X8(K.question)}return l$(H)}};var I0=R("lessons:start"),s$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments,H=J.lesson,K=J.student,W=await Z.getAssessmentProgress({student:K,lesson:H});if(J.forceNew){if(W.lessonType!=="powerpath-100"&&W.finalized!==!0)try{await Z.finalStudentAssessmentResponse({student:K,lesson:H})}catch{}try{await Z.createNewAttempt({student:K,lesson:H}),W=await Z.getAssessmentProgress({student:K,lesson:H})}catch(B){I0.warn("Failed to create new lesson attempt",{lessonId:H,error:B})}}else if(W.lessonType!=="powerpath-100"&&W.finalized!==!0&&(W.questions?.length??0)===0)try{await Z.createNewAttempt({student:K,lesson:H}),W=await Z.getAssessmentProgress({student:K,lesson:H})}catch{}return m$(H,W)}};var o$=($)=>{return async(J)=>{let H=await $.getClient().powerpath.assessments.updateStudentQuestionResponse({student:J.student,lesson:J.lesson,question:J.question,responses:{RESPONSE:J.response}});return u$(H)}};function m($){return{list:a$($),start:s$($),next:i$($),submit:o$($),complete:g$($),attempts:f$($),attemptDetails:y$($)}}async function a($,J){let Z=await R2($.identity,J);if(!Z)return null;if(Z.timebackId)return Z.timebackId;return await B2({email:Z.email,client:$.getClient()})}async function n$($,J){let Z=await R2($.identity,J);return Boolean(Z)}var b0=R("handlers:lessons");function c($,J){if($ instanceof d)return M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",F2($));if($ instanceof W8)return M("LESSON_TYPE_MISMATCH",$.message,400);let Z=$ instanceof Error?$.message:"Unknown error";return b0.error(J,{error:Z}),M("INTERNAL_ERROR",Z,500)}import*as i from"zod";var h0=i.object({course:M2.optional()}).partial(),S0=i.object({lessonId:E,forceNew:i.boolean().optional()}),E0=i.object({lessonId:E,lessonType:E.optional()}),v0=i.object({lessonId:E,questionId:E,response:E}),y0=i.object({lessonId:E}),f0=i.object({lessonId:E}),g0=i.object({lessonId:E,attempt:i.number().int().positive()});async function d0($){try{let J=await $.json();if(!J||typeof J!=="object"||Array.isArray(J))return null;return J}catch{return null}}async function V2($,J){let Z=await d0($);if(!Z)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid JSON body",400)};let H=J.safeParse(Z);if(!H.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:H.error.flatten()})};return{ok:!0,payload:H.data}}async function r$($){return await V2($,h0)}async function t$($){return await V2($,S0)}async function e$($){return await V2($,E0)}async function $J($){return await V2($,v0)}async function JJ($){return await V2($,y0)}async function ZJ($){return await V2($,f0)}async function HJ($){return await V2($,g0)}function Y8($){let J=m($);return async(Z)=>{try{let H=await HJ(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.attemptDetails({lesson:H.payload.lessonId,attempt:H.payload.attempt,student:K});return P(W)}catch(H){return c(H,"Failed to fetch lesson attempt details")}}}function Q8($){let J=m($);return async(Z)=>{try{let H=await ZJ(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.attempts({lesson:H.payload.lessonId,student:K});return P({attempts:W})}catch(H){return c(H,"Failed to list lesson attempts")}}}function G8($){let J=m($);return async(Z)=>{try{let H=await JJ(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.complete({lesson:H.payload.lessonId,student:K});return P(W)}catch(H){return c(H,"Failed to complete lesson")}}}function _8($){let J=m($);return async(Z)=>{try{if(!await n$($,Z))return M("UNAUTHORIZED","Unauthorized",401);let H=await r$(Z);if(!H.ok)return H.response;let K=await J.list({course:H.payload.course});return P({lessons:K})}catch(H){return c(H,"Failed to list lessons")}}}function B8($){let J=m($);return async(Z)=>{try{let H=await e$(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.next({lesson:H.payload.lessonId,lessonType:H.payload.lessonType,student:K});return P(W)}catch(H){return c(H,"Failed to fetch next lesson question")}}}function F8($){let J=m($);return async(Z)=>{try{let H=await t$(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.start({lesson:H.payload.lessonId,student:K,forceNew:H.payload.forceNew});return P(W)}catch(H){return c(H,"Failed to start lesson")}}}function O8($){let J=m($);return async(Z)=>{try{let H=await $J(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.submit({lesson:H.payload.lessonId,question:H.payload.questionId,response:H.payload.response,student:K});return P(W)}catch(H){return c(H,"Failed to submit lesson answer")}}}function KJ($,J){let Z=new Map;for(let H of $){let K=H.ids?.[J];if(K)Z.set(K,H)}return Z}function XJ($,J){return $.map((Z)=>{let H=J.get(Z.course.id);return{id:Z.course.id,code:H?.courseCode??Z.course.id,name:Z.course.title}})}function WJ($){return $.map((J)=>J.metadata?.goals).find(Boolean)}function YJ($){let J=new Date(Date.UTC($.getUTCFullYear(),$.getUTCMonth(),$.getUTCDate())),Z=new Date(Date.UTC($.getUTCFullYear(),$.getUTCMonth(),$.getUTCDate(),23,59,59,999));return{start:J,end:Z}}function d8($){return Object.values($).reduce((J,Z)=>{return J+Object.values(Z).reduce((H,K)=>{return H+(K.activityMetrics?.xpEarned??0)},0)},0)}async function D8($,J,Z,H){let K=await $.edubridge.enrollments.list({userId:J.id}),W=KJ(Z.courses,H),B=XJ(K,W),F=WJ(K),{start:O,end:U}=YJ(new Date),[V,A]=await Promise.all([$.edubridge.analytics.getActivity({studentId:J.id,startDate:O.toISOString(),endDate:U.toISOString()}),$.edubridge.analytics.getActivity({studentId:J.id,startDate:"2000-01-01",endDate:U.toISOString()})]);return{id:J.id,email:J.email,name:J.name,school:J.school,grade:J.grade,courses:B.length?B:void 0,goals:F,xp:{today:d8(V),all:d8(A)}}}var m8=R("handlers:user");async function m0($,J){if($.mode==="custom"){let H=await $.getEmail(J);if(!H)return;return{sub:H,email:H}}let Z=await $.getUser(J);if(!Z)return;return{sub:Z.id,email:Z.email}}function l0($){if(m8.warn("Timeback user resolution failed",{code:$.code}),$.code==="timeback_user_ambiguous")return M("USER_AMBIGUOUS","Timeback user resolution ambiguous",409);if($.code==="timeback_user_not_found")return M("USER_NOT_FOUND","Timeback user not found",404);return M("USER_RESOLUTION_FAILED",$.message,502)}function w8($){return async(J)=>{try{let Z=await m0($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let H=u($.env),K=$.getClient();try{let W=await O2({env:$.env,apiCredentials:$.api,userInfo:Z,client:K}),B=await D8(K,W,$.appConfig,H);return P(B)}catch(W){if(W instanceof d)return l0(W);let B=W instanceof Error?W.message:"Unknown error";return m8.error("Failed to build user profile",{error:B}),M("INTERNAL_ERROR",B,502)}}catch(Z){let H=Z instanceof Error?Z.message:"Unknown error";return m8.error("Unhandled error in user handler",{error:H}),M("INTERNAL_ERROR",H,500)}}}var l8=R("handlers:user:verify");async function u0($,J){if($.mode==="custom"){let H=await $.getEmail(J);if(!H)return;return{sub:H,email:H}}let Z=await $.getUser(J);if(!Z)return;return{sub:Z.id,email:Z.email}}function U8($){return async(J)=>{try{let Z=await u0($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let H=$.getClient();try{let K=await O2({env:$.env,apiCredentials:$.api,userInfo:Z,client:H});return P({verified:!0,timebackId:K.id})}catch(K){if(K instanceof d){if(l8.warn("Timeback user resolution failed",{code:K.code}),K.code==="timeback_user_ambiguous")return M("USER_AMBIGUOUS","Timeback user resolution ambiguous",409);if(K.code==="timeback_user_not_found")return P({verified:!1});return M("USER_RESOLUTION_FAILED",K.message,502)}let W=K instanceof Error?K.message:"Unknown error";return l8.error("Failed to verify user",{error:W}),M("INTERNAL_ERROR",W,502)}}catch(Z){let H=Z instanceof Error?Z.message:"Unknown error";return l8.error("Unhandled error in verify handler",{error:H}),M("INTERNAL_ERROR",H,500)}}}import*as T from"zod";var c0=T.object({email:T.email(),timebackId:T.string().optional()}),p0=T.object({id:E,name:E,course:M2}),a0=T.object({totalQuestions:T.number().int().nonnegative().optional(),correctQuestions:T.number().int().nonnegative().optional(),xpEarned:T.number(),masteredUnits:T.number().int().nonnegative().optional(),pctComplete:T.number().min(0).max(100).optional()}).superRefine(n2),i0=T.object({startedAt:T.union([T.iso.datetime(),T.date()]).optional(),endedAt:T.union([T.iso.datetime(),T.date()]).optional(),activeMs:T.number().int().nonnegative().optional(),inactiveMs:T.number().int().nonnegative().optional()}),s0=T.uuid().optional(),QJ=T.object({user:c0,activity:p0,metrics:a0,time:i0.optional(),runId:s0});var GJ=R("activity:record");class _J extends Error{code="VALIDATION_ERROR";details;constructor($,J){super($);this.name="ActivityRecordValidationError",this.details=J}}class u8 extends Error{code="COURSE_ERROR";constructor($){super($);this.name="ActivityRecordCourseError"}}function o0($){let J=$.time?.endedAt?typeof $.time.endedAt==="string"?$.time.endedAt:$.time.endedAt.toISOString():new Date().toISOString();return{id:$.activity.id,name:$.activity.name,course:$.activity.course,endedAt:J,metrics:{totalQuestions:$.metrics.totalQuestions,correctQuestions:$.metrics.correctQuestions,xpEarned:$.metrics.xpEarned,masteredUnits:$.metrics.masteredUnits},pctComplete:$.metrics.pctComplete}}function n0($,J){let Z=$.time,H=Z.activeMs??0,K=Z.inactiveMs??0,W;if(Z.startedAt)W=typeof Z.startedAt==="string"?Z.startedAt:Z.startedAt.toISOString();else if(Z.activeMs!==void 0||Z.inactiveMs!==void 0){let B=new Date(J),F=H+K;W=new Date(B.getTime()-F).toISOString()}else W=J;return{id:$.activity.id,name:$.activity.name,course:$.activity.course,startedAt:W,endedAt:J,elapsedMs:H,pausedMs:K}}function r0($,J,Z){let H;try{H=L2($.courses,J)}catch(W){if(W instanceof _2)throw new u8(W.code==="unknown_course"?`Unknown course: ${JSON.stringify(J)}`:"Ambiguous course selector in timeback.config.json");throw W}let K=S2($,H,Z);if(!K)throw new u8(`Course "${JSON.stringify(J)}" has no sensor configured.`);return{course:H,sensor:K}}var t0={recordCompletion:r2,sendTimeSpent:C$};function c8($,J=t0){return async(Z)=>{let H=QJ.safeParse(Z);if(!H.success)throw GJ.warn("Invalid record input",{error:H.error.flatten()}),new _J("Invalid activity record input",H.error.flatten());let K=H.data,{course:W,sensor:B}=r0($.appConfig,K.activity.course,$.env),F=o0(K),O={env:$.env,api:$.api,appConfig:$.appConfig,getClient:$.getClient,hooks:$.hooks};if(await J.recordCompletion(O,{userInfo:{email:K.user.email,timebackId:K.user.timebackId},payload:F,course:W,sensor:B,preview:!1,runId:K.runId}),K.time){let U=n0(K,F.endedAt);await J.sendTimeSpent(O,{userInfo:{email:K.user.email,timebackId:K.user.timebackId},payload:U,course:W,sensor:B,runId:K.runId})}GJ.debug("Recorded activity",{activityId:K.activity.id,hasTime:!!K.time})}}var x8=R("user:verify");function p8($){return async(J)=>{x8.debug("Verifying user by email");let Z=$.getClient();try{let H=await B2({email:J,client:Z});return x8.debug("User verified",{timebackId:H}),{verified:!0,timebackId:H}}catch(H){if(H instanceof d){if(H.code==="timeback_user_not_found")return x8.debug("User not found in Timeback"),{verified:!1};throw H}let K=H instanceof Error?H.message:"Unknown error";throw x8.error("Failed to verify user",{error:K}),H}}}var a8=R("user:getProfile");function i8($){return async(J)=>{a8.debug("Building user profile by email");let Z=$.getClient(),H=u($.env);try{let K=await O2({env:$.env,apiCredentials:$.api,userInfo:{sub:J,email:J},client:Z}),W=await D8(Z,K,$.appConfig,H);return a8.debug("User profile built",{userId:W.id}),W}catch(K){if(K instanceof d)throw K;let W=K instanceof Error?K.message:"Unknown error";throw a8.error("Failed to build user profile",{error:W}),K}}}async function $1($){let{loadConfig:J}=await import("./chunk-2kfgswby.js");return await J($)}function J1($){return $.map((J)=>{if(J.grade!==void 0)return J;if(J.courseCode)return J;throw Error(`Invalid course config: missing both grade and courseCode for subject "${J.subject}"`)})}async function DJ($){if(!globalThis.process?.versions?.node&&!globalThis.Bun)throw Error("createTimeback() requires Node.js or Bun — it reads timeback.config.json from the filesystem. "+"For edge runtimes (Cloudflare Workers, Vercel Edge, etc.), use createTimebackIdentity() from '@timeback/sdk' instead.");BJ({logger:$.logger,logLevel:$.logLevel});let J=FJ($.env),Z=await $1({configPath:$.configPath});if(!Z.success)throw Error(`Failed to load timeback config: ${Z.error}`);let H=Z.config,K,W=()=>{if(!K)K=new e0({env:u(J),auth:{clientId:$.api.clientId,clientSecret:$.api.clientSecret}});return K},B=$.hooks,F=(V,A)=>OJ(V,B,A),O={env:J,identity:$.identity,appConfig:{name:H.name,sensor:H.sensor,launchUrl:H.launchUrl,courses:J1(H.courses)},api:$.api,getClient:W,hooks:B},U={env:J,identity:$.identity,api:{credentials:$.api,getClient:W}};return{config:$,handle:{activity:{heartbeat:F("activity.heartbeat",$8(O)),submit:F("activity.submit",t2(O))},identity:{signIn:F("identity.signIn",Z8(U)),callback:F("identity.callback",H8(U)),signOut:K8()},lessons:{list:F("lessons.list",_8(O)),start:F("lessons.start",F8(O)),next:F("lessons.next",B8(O)),submit:F("lessons.submit",O8(O)),complete:F("lessons.complete",G8(O)),attempts:F("lessons.attempts",Q8(O)),attemptDetails:F("lessons.attemptDetails",Y8(O))},user:{me:F("user.me",w8(O)),verify:F("user.verify",U8(O))}},activity:{record:c8({...O})},lessons:m({...O}),user:{verify:p8({...O}),getProfile:i8({...O})},get api(){return W()},close(){if(K)K.close(),K=void 0}}}import{TimebackClient as IK}from"@timeback/core";import{ApiError as hK,ForbiddenError as SK,isApiError as EK,NotFoundError as vK,UnauthorizedError as yK,ValidationError as fK}from"@timeback/core";export{H1 as toNativeHandler,EK as isApiError,wJ as createTimebackIdentity,DJ as createTimeback,fK as ValidationError,yK as UnauthorizedError,IK as TimebackClient,Z1 as ROUTES,vK as NotFoundError,SK as ForbiddenError,hK as ApiError};
1
+ import"./chunk-ae6bkfs5.js";import{a as E,b as O$,c as D$}from"./chunk-nrn5gjey.js";import{e as N$,f as I$,g as b$,h as h$,i as S$,j as E$,k as xJ}from"./chunk-qpzke7wy.js";import{l as Y1}from"./server/adapters/native.js";import{m as d,n as F2,o as O2,p as _2,q as o2,r as h2,s as DJ}from"./chunk-k1awna2q.js";import{t as FJ,u as R,v as L2,w as P$,y as T$,z as R$}from"./chunk-b49f5qfw.js";import"./chunk-3pf3cpfk.js";import{G as f8,H as L$,I as S2,J as u,K as OJ,L as P,M,N as $8,Q as W1}from"./chunk-dbcsf5tk.js";import{R as M8,U as j8}from"./chunk-1s709w5e.js";import{TimebackClient as H1}from"@timeback/core";import{getServiceUrlsForEnv as T0}from"@timeback/core";class s extends Error{statusCode;response;name="ApiError";constructor($,J,Z){super($);this.statusCode=J,this.response=Z}get minorCodes(){let $=this.response;if(!$?.imsx_CodeMinor?.imsx_codeMinorField)return[];return $.imsx_CodeMinor.imsx_codeMinorField.map((J)=>({field:J.imsx_codeMinorFieldName,value:J.imsx_codeMinorFieldValue}))}get details(){return this.response?.imsx_error_details??[]}}class $9 extends s{name="UnauthorizedError";constructor($="Unauthorized",J){super($,401,J)}}class J9 extends s{name="ForbiddenError";constructor($="Forbidden",J){super($,403,J)}}class Z9 extends s{name="NotFoundError";constructor($="Not Found",J){super($,404,J)}}class H9 extends s{name="ValidationError";constructor($="Validation Error",J){super($,422,J)}}class K9 extends s{name="InputValidationError";issues;constructor($,J){let Z={imsx_codeMajor:"failure",imsx_severity:"error",imsx_description:$,imsx_error_details:J.map((H)=>({path:H.path,message:H.message}))};super($,400,Z);this.issues=J}}function N2($,J){return new K9($,J)}function MJ($){if(!($ instanceof Error))return!1;return"statusCode"in $&&"response"in $}var jJ="BEYOND_AI",X9="LEARNWITH_AI",W9=jJ,LJ={staging:"https://staging-beyond-timeback-api-2-idp.auth.us-east-1.amazoncognito.com/oauth2/token",production:"https://prod-beyond-timeback-api-2-idp.auth.us-east-1.amazoncognito.com/oauth2/token"},VJ={staging:"https://api.staging.alpha-1edtech.ai",production:"https://api.alpha-1edtech.ai"},kJ={staging:"https://caliper.staging.alpha-1edtech.ai",production:"https://caliper.alpha-1edtech.ai"},qJ={staging:"https://qti.alpha-1edtech.ai/api",production:"https://qti.alpha-1edtech.ai/api"},CJ={staging:"https://platform.dev.timeback.com/auth/1.0/token",production:"https://platform.timeback.com/auth/1.0/token"},AJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},zJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},PJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},TJ={BEYOND_AI:{token:LJ,tokenScope:void 0,api:VJ,caliper:kJ,qti:qJ},LEARNWITH_AI:{token:CJ,tokenScope:"https://purl.imsglobal.org/spec/caliper/v1p2/scope/events.write",api:AJ,caliper:zJ,qti:PJ}},RJ={tsc:"tsc",nativePreview:"@typescript/native-preview"},_1={package:RJ.nativePreview,bin:"tsgo"},e=null,L8=!1,V8=!1;function Y9(){try{let $=process.env.DEBUG?.trim();if(!$)return!1;return $.split(",").some((J)=>J.trim().length>0)}catch{return!1}}function s8($){let Z=$.replace(/[.+?^${}()|[\]\\]/g,"\\$&").replace(/\*/g,".*");return new RegExp(`^${Z}$`)}function NJ(){if(e!==null)return;if(e=[],!Y9()){V8=!1;return}V8=!0;let $=process.env.DEBUG.trim();if($==="1"||$==="true"||$==="*"){L8=!0;return}let J=$.split(",").map((H)=>H.trim()).filter(Boolean);for(let H of J)if(H.startsWith("-"))e.push({regex:s8(H.slice(1)),exclude:!0});else e.push({regex:s8(H),exclude:!1});if(!e.some((H)=>!H.exclude)&&e.length>0)L8=!0}function IJ($){if(NJ(),!V8)return!0;if(L8){if($){for(let J of e)if(J.exclude&&J.regex.test($))return!1}return!0}if(!$)return!1;for(let J of e)if(J.exclude&&J.regex.test($))return!1;for(let J of e)if(!J.exclude&&J.regex.test($))return!0;return!1}function Q9(){return typeof globalThis<"u"&&"window"in globalThis}function bJ(){if(Q9())return"browser";if(typeof process<"u"&&process.env){if(process.env.CI||process.env.GITHUB_ACTIONS||process.env.GITLAB_CI||process.env.CIRCLECI||process.env.JENKINS_URL||process.env.BUILDKITE)return"ci"}return"terminal"}var k8;if(!Q9())try{k8=(await import("./chunk-kkajswjx.js")).inspect}catch{}var $2={reset:"\x1B[0m",bold:"\x1B[1m",dim:"\x1B[2m",red:"\x1B[31m",yellow:"\x1B[33m",blue:"\x1B[34m",cyan:"\x1B[36m"};function hJ($){switch($){case"debug":return $2.blue;case"info":return $2.cyan;case"warn":return $2.yellow;case"error":return $2.red}}function SJ($){switch($){case"debug":return console.debug;case"info":return console.info;case"warn":return console.warn;case"error":return console.error}}function EJ($){if(k8)return k8($,{depth:null,colors:!0,breakLength:80,compact:!1});return JSON.stringify($,null,2)}var vJ=($)=>{let J=hJ($.level),Z=SJ($.level),H=$.level.toUpperCase().padEnd(5),K=$.timestamp.toISOString().replace(/\.\d{3}Z$/,""),W=`${$2.dim}[${K}]${$2.reset}`,_=`${J}${H}${$2.reset}`,F=$.scope?`${$2.bold}[${$.scope}]${$2.reset} `:"",O=`${W} ${_} ${F}${$.message}`;if($.context&&Object.keys($.context).length>0)Z(O,EJ($.context));else Z(O)},yJ={debug:"[DEBUG]",info:"[INFO]",warn:"[WARN]",error:"[ERROR]"};function fJ($){return Object.entries($).map(([J,Z])=>`${J}=${gJ(Z)}`).join(" ")}function gJ($){if(typeof $==="string")return $;if(typeof $==="number")return String($);if(typeof $==="boolean")return String($);if($===null)return"null";if($===void 0)return"undefined";return JSON.stringify($)}var dJ=($)=>{let J=[];if(J.push($.timestamp.toISOString()),J.push(yJ[$.level]),$.scope)J.push(`[${$.scope}]`);if(J.push($.message),$.context&&Object.keys($.context).length>0)J.push(fJ($.context));console.log(J.join(" "))},mJ=($)=>{let J={timestamp:$.timestamp.toISOString(),level:$.level,...$.scope&&{scope:$.scope},msg:$.message};if($.context&&Object.keys($.context).length>0)Object.assign(J,$.context);console.log(JSON.stringify(J))},lJ={debug:"color: gray",info:"color: #0ea5e9",warn:"color: #f59e0b",error:"color: #ef4444; font-weight: bold"},uJ={debug:"log",info:"info",warn:"warn",error:"error"},cJ=($)=>{let J=uJ[$.level],Z=lJ[$.level],K=`%c${$.scope?`[${$.scope}]`:""} ${$.message}`;if($.context&&Object.keys($.context).length>0)console[J](K,Z,$.context);else console[J](K,Z)},o8=["debug","info","warn","error"],n8=Symbol.for("@timeback/internal-logger/config");function pJ(){try{return process.env.TIMEBACK_INTERNAL_LOGGER==="silent"}catch{return!1}}function aJ(){let $=globalThis;return $[n8]??={},$[n8]}function iJ($){switch($){case"terminal":return vJ;case"ci":return dJ;case"production":return mJ;case"browser":return cJ;case"test":return()=>{}}}function sJ(){if(Y9())return"debug";return"info"}function oJ($,J){return J??aJ().formatter??(pJ()?()=>{}:iJ($))}function nJ($,J){return o8.indexOf($)>=o8.indexOf(J)}class g2{scope;minLevel;environment;explicitFormatter;defaultContext;constructor($={}){this.scope=$.scope,this.minLevel=$.minLevel??sJ(),this.defaultContext=$.defaultContext??{},this.environment=$.environment??bJ(),this.explicitFormatter=$.formatter}child($){let J=this.scope?`${this.scope}:${$}`:$;return new g2({scope:J,minLevel:this.minLevel,environment:this.environment,defaultContext:{...this.defaultContext},formatter:this.explicitFormatter})}withContext($){return new g2({scope:this.scope,minLevel:this.minLevel,environment:this.environment,defaultContext:{...this.defaultContext,...$},formatter:this.explicitFormatter})}debug($,J){this.log("debug",$,J)}info($,J){this.log("info",$,J)}warn($,J){this.log("warn",$,J)}error($,J){this.log("error",$,J)}log($,J,Z){if($==="debug"&&!IJ(this.scope))return;if(!nJ($,this.minLevel))return;let H={level:$,message:J,scope:this.scope,context:Z||Object.keys(this.defaultContext).length>0?{...this.defaultContext,...Z}:void 0,timestamp:new Date};oJ(this.environment,this.explicitFormatter)(H)}}function G9($={}){return new g2($)}function rJ(){try{let $=typeof process>"u"?void 0:process.env.DEBUG;return $==="1"||$==="true"}catch{return!1}}class d2{config;accessToken=null;tokenExpiry=0;pendingRequest=null;fetchFn;log;constructor($){this.config=$,this.fetchFn=$.fetch??globalThis.fetch.bind(globalThis),this.log=$.logger??G9({scope:"auth",minLevel:rJ()?"debug":"warn"})}async getToken(){if(this.accessToken&&Date.now()<this.tokenExpiry)return this.log.debug("Using cached token"),this.accessToken;if(this.pendingRequest)return this.log.debug("Waiting for in-flight token request"),this.pendingRequest;this.pendingRequest=this.fetchToken();try{return await this.pendingRequest}finally{this.pendingRequest=null}}async fetchToken(){this.log.debug("Fetching new access token...");let{clientId:$,clientSecret:J}=this.config.credentials,Z=btoa(`${$}:${J}`),H=performance.now(),K=await this.fetchFn(this.config.tokenUrl,{method:"POST",headers:{Authorization:`Basic ${Z}`,"Content-Type":"application/x-www-form-urlencoded"},body:this.config.scope?`grant_type=client_credentials&scope=${encodeURIComponent(this.config.scope)}`:"grant_type=client_credentials"}),W=Math.round(performance.now()-H);if(!K.ok)throw this.log.error(`Token request failed: ${K.status} ${K.statusText}`),Error(`Failed to obtain access token: ${K.status} ${K.statusText}`);let _=await K.json();return this.accessToken=_.access_token,this.tokenExpiry=Date.now()+(_.expires_in-60)*1000,this.log.debug(`Token acquired (${W}ms, expires in ${_.expires_in}s)`),this.accessToken}invalidate(){this.log.debug("Token invalidated"),this.accessToken=null,this.tokenExpiry=0}}var m2={caliper:{send:"/caliper/event",validate:"/caliper/event/validate",list:"/caliper/events",get:"/caliper/events/{id}",jobStatus:"/jobs/{id}/status"},oneroster:{rostering:"/ims/oneroster/rostering/v1p2",gradebook:"/ims/oneroster/gradebook/v1p2",resources:"/ims/oneroster/resources/v1p2"},edubridge:{base:"/edubridge"},powerpath:{base:"/powerpath"},clr:{credentials:"/ims/clr/v2p0/credentials",discovery:"/ims/clr/v2p0/discovery"},case:{base:"/ims/case/v1p1"},webhooks:{webhookList:"/webhooks/",webhookGet:"/webhooks/{id}",webhookCreate:"/webhooks/",webhookUpdate:"/webhooks/{id}",webhookDelete:"/webhooks/{id}",webhookActivate:"/webhooks/{id}/activate",webhookDeactivate:"/webhooks/{id}/deactivate",webhookFilterList:"/webhook-filters/",webhookFilterGet:"/webhook-filters/{id}",webhookFilterCreate:"/webhook-filters/",webhookFilterUpdate:"/webhook-filters/{id}",webhookFilterDelete:"/webhook-filters/{id}",webhookFiltersByWebhook:"/webhook-filters/webhook/{webhookId}"},reporting:{mcp:"/mcp/reporting",savedQueryExecute:"/reporting/saved-queries/{id}",adminGroupCheck:"/mcp/admin/users/{email}/group",adminGroupAdd:"/mcp/admin/users/{email}/group",adminGroupRemove:"/mcp/admin/users/{email}/group"}},tJ={caliper:{send:"/caliper/v1p2",validate:null,list:null,get:null,jobStatus:null},oneroster:{rostering:"/rostering/1.0",gradebook:"/gradebook/1.0",resources:"/resources/1.0"},webhooks:null,reporting:null,edubridge:null,powerpath:null,clr:null,case:{base:"/case/1.1"}},B9={BEYOND_AI:m2,LEARNWITH_AI:tJ};function eJ($){return"env"in $&&!("baseUrl"in $)&&!("services"in $)}function $5($){return"baseUrl"in $&&!("services"in $)}function J5($){return"services"in $}function r8($,J){let Z=$?B9[$]??m2:m2;return{caliper:J?.caliper??Z.caliper,oneroster:J?.oneroster??Z.oneroster,webhooks:J?.webhooks??Z.webhooks,reporting:J?.reporting??Z.reporting,edubridge:J?.edubridge??Z.edubridge,powerpath:J?.powerpath??Z.powerpath,clr:J?.clr??Z.clr,case:J?.case??Z.case}}class I2{platform;env;auth;timeout;_endpoints;_authUrl;_tokenScope;_pathProfiles;_tokenManagers=new Map;constructor($){if(this.timeout=$.timeout??30000,eJ($)){this.auth=$.auth;let J=$.platform??W9,Z=$.env;this.platform=J,this.env=Z;let H=TJ[J];if(!H)throw Error(`Unknown platform: ${J}`);this._authUrl=H.token[Z],this._tokenScope=H.tokenScope??void 0,this._pathProfiles=B9[J]??m2,this._endpoints={oneroster:{baseUrl:H.api[Z],authUrl:this._authUrl},edubridge:{baseUrl:H.api[Z],authUrl:this._authUrl},powerpath:{baseUrl:H.api[Z],authUrl:this._authUrl},clr:{baseUrl:H.api[Z],authUrl:this._authUrl},case:{baseUrl:H.api[Z],authUrl:this._authUrl},caliper:{baseUrl:H.caliper[Z],authUrl:this._authUrl},webhooks:{baseUrl:H.caliper[Z],authUrl:this._authUrl},reporting:{baseUrl:H.api[Z],authUrl:this._authUrl},qti:{baseUrl:H.qti[Z],authUrl:this._authUrl}}}else if($5($))this.auth=$.auth,this._authUrl=$.authUrl,this._pathProfiles=r8($.pathProfile,$.paths),this._endpoints={oneroster:{baseUrl:$.baseUrl,authUrl:this._authUrl},edubridge:{baseUrl:$.baseUrl,authUrl:this._authUrl},powerpath:{baseUrl:$.baseUrl,authUrl:this._authUrl},clr:{baseUrl:$.baseUrl,authUrl:this._authUrl},case:{baseUrl:$.baseUrl,authUrl:this._authUrl},caliper:{baseUrl:$.baseUrl,authUrl:this._authUrl},webhooks:{baseUrl:$.baseUrl,authUrl:this._authUrl},reporting:{baseUrl:$.baseUrl,authUrl:this._authUrl},qti:{baseUrl:$.baseUrl,authUrl:this._authUrl}};else if(J5($)){this.auth=$.auth,this._authUrl=$.authUrl,this._pathProfiles=r8($.pathProfile,$.paths),this._endpoints={};for(let[J,Z]of Object.entries($.services))if(Z)this._endpoints[J]={baseUrl:Z,authUrl:this._authUrl}}else throw Error("Invalid provider configuration");for(let J of Object.keys(this._pathProfiles))if(this._pathProfiles[J]===null)delete this._endpoints[J]}getEndpoint($){let J=this._endpoints[$];if(!J){let Z=$;if(Z in this._pathProfiles&&this._pathProfiles[Z]===null)throw Error(`Service "${$}" is not supported on ${this.platform??"this"} platform.`);throw Error(`Service "${$}" is not configured in this provider`)}return J}hasService($){return $ in this._endpoints}getAvailableServices(){return Object.keys(this._endpoints)}getTokenUrl(){return this._authUrl}getEndpointWithPaths($){let J=this.getEndpoint($),Z=this.getServicePaths($);return{...J,paths:Z}}getPaths(){return this._pathProfiles}getServicePaths($){let J=this._pathProfiles[$];if(!J)throw Error(`Service "${$}" is not supported on ${this.platform??"this"} platform.`);return J}hasServiceSupport($){return this._pathProfiles[$]!==null}getTokenProvider($){let J=this.getEndpoint($),{authUrl:Z}=J;if(!Z)return;if(!this.auth)throw Error(`Service "${$}" requires authentication but no credentials were provided`);let H=this._tokenManagers.get(Z);if(!H)H=new d2({tokenUrl:Z,credentials:{clientId:this.auth.clientId,clientSecret:this.auth.clientSecret},scope:this._tokenScope}),this._tokenManagers.set(Z,H);return H}async checkAuth(){if(!this._authUrl||!this.auth)throw Error("No auth configured on this provider");let $=Date.now(),J=this._tokenManagers.get(this._authUrl);if(!J)J=new d2({tokenUrl:this._authUrl,credentials:{clientId:this.auth.clientId,clientSecret:this.auth.clientSecret},scope:this._tokenScope}),this._tokenManagers.set(this._authUrl,J);try{return await J.getToken(),{ok:!0,latencyMs:Date.now()-$,checks:{tokenAcquisition:!0}}}catch(Z){return{ok:!1,latencyMs:Date.now()-$,error:Z instanceof Error?Z.message:String(Z),checks:{tokenAcquisition:!1}}}}invalidateTokens(){for(let $ of this._tokenManagers.values())$.invalidate?.();this._tokenManagers.clear()}}function b($){try{if(typeof process>"u")return;if(typeof $==="string")return process.env[$];for(let J of $){let Z=process.env[J];if(Z!==void 0)return Z}return}catch{return}}function D2(){let $=b("DEBUG");return $==="1"||$==="true"}function Z5(){return Math.random().toString(16).slice(2,10)}function q8($){return G9({scope:$,minLevel:D2()?"debug":"warn"})}var l2={defaultPlatform:W9,templates:{BEYOND_AI:{staging:{platform:"BEYOND_AI",env:"staging"},production:{platform:"BEYOND_AI",env:"production"}},LEARNWITH_AI:{staging:{platform:"LEARNWITH_AI",env:"staging"},production:{platform:"LEARNWITH_AI",env:"production"}}}};function H5($){if(typeof $==="string")return $;if($.length===0)throw Error(`Missing env var key: ${$}`);return $[0]}function w2($){return H5($)}function _9($){if($!=="staging"&&$!=="production")throw Error(`Invalid env "${$}": must be "staging" or "production"`);return $}function F9($,J){let Z=$?.clientId??b(J.clientId),H=$?.clientSecret??b(J.clientSecret);if(!Z)throw Error(`Missing clientId: provide in config or set ${w2(J.clientId)}`);if(!H)throw Error(`Missing clientSecret: provide in config or set ${w2(J.clientSecret)}`);return{clientId:Z,clientSecret:H}}function K5($,J,Z){if(!Z)return;return F9($,J)}function X5($){let J=b($.baseUrl),Z=b($.authUrl),H=b($.clientId),K=b($.clientSecret);if(J===void 0&&H===void 0)return`Missing env: provide in config or set ${w2($.env??$.baseUrl)}`;let W=[];if(J===void 0)W.push(w2($.env??$.baseUrl));if(J!==void 0&&Z===void 0)W.push(w2($.authUrl));if(H===void 0)W.push(w2($.clientId));if(K===void 0)W.push(w2($.clientSecret));return`Missing environment variables: ${W.join(", ")}`}var W5={name:"transport",matches($){return"transport"in $&&!!$.transport},resolve($){return{mode:"transport",transport:$.transport}}},Y5={name:"provider",matches($){return"provider"in $&&!!$.provider},resolve($){return{mode:"provider",provider:$.provider}}},Q5={name:"env-config",matches($){return"env"in $},resolve($,J,Z){let H=$,K=_9(H.env),W=F9(H.auth,J),_=H.platform??Z.defaultPlatform,F=Z.templates[_];if(!F){let x=Object.keys(Z.templates).join(", ");throw Error(`Unknown platform "${_}": available platforms are ${x}`)}let O=F[K];if(!O){let x=Object.keys(F).join(", ");throw Error(`Unknown env "${K}" for platform "${_}": available environments are ${x}`)}return{mode:"provider",provider:new I2({platform:O.platform,env:O.env,auth:W,timeout:H.timeout})}}},G5={name:"explicit-config",matches($){return"baseUrl"in $},resolve($,J){let Z=$,H=Z.authUrl??Z.auth?.authUrl,K=!!H,W=K5(Z.auth,J,K);return{mode:"provider",provider:new I2({baseUrl:Z.baseUrl,authUrl:H,auth:W,timeout:Z.timeout,pathProfile:Z.pathProfile,paths:Z.paths})}}},B5={name:"env-fallback-platform",matches($,J){if(Object.keys($).length>0)return!1;let Z=J.env?b(J.env):void 0,H=b(J.clientId),K=b(J.clientSecret);return Z!==void 0&&H!==void 0&&K!==void 0},resolve($,J,Z){let H=_9(b(J.env)),K=b(J.clientId),W=b(J.clientSecret),_=Z.defaultPlatform,F=Z.templates[_]?.[H];if(!F)throw Error(`Unknown env "${H}" for platform "${_}"`);return{mode:"provider",provider:new I2({platform:F.platform,env:F.env,auth:{clientId:K,clientSecret:W}})}}},_5={name:"env-fallback-explicit",matches($,J){if(Object.keys($).length>0)return!1;let Z=b(J.baseUrl),H=b(J.authUrl),K=b(J.clientId),W=b(J.clientSecret);return Z!==void 0&&H!==void 0&&K!==void 0&&W!==void 0},resolve($,J){let Z=b(J.baseUrl),H=b(J.authUrl),K=b(J.clientId),W=b(J.clientSecret);return{mode:"provider",provider:new I2({baseUrl:Z,authUrl:H,auth:{clientId:K,clientSecret:W}})}}},F5={name:"token-provider",matches($){return"tokenProvider"in $},resolve(){throw Error("TokenProvider mode is not supported with provider pattern. Use { provider: TimebackProvider } or { env, auth } instead.")}},O5=[W5,Y5,Q5,F5,G5,B5,_5];function O9($,J,Z=l2){for(let H of O5)if(H.matches($,J))return H.resolve($,J,Z);throw Error(X5(J))}var U8=3,D5=[429,503],t8=1000;class C8{config;log;constructor($){let{config:J,logger:Z}=$,H=J.fetch??globalThis.fetch.bind(globalThis),K;if("tokenProvider"in J&&J.tokenProvider)K=J.tokenProvider;else if("auth"in J&&J.auth)K=new d2({tokenUrl:J.auth.authUrl,credentials:{clientId:J.auth.clientId,clientSecret:J.auth.clientSecret},fetch:H});let W=J.baseUrl.endsWith("/")?J.baseUrl:`${J.baseUrl}/`;this.config={baseUrl:W,timeout:J.timeout??30000,fetch:H,tokenProvider:K},this.log=Z.child("http")}get baseUrl(){return this.config.baseUrl}async request($,J={}){let Z=await this.requestRaw($,J);return this.handleResponse(Z)}async exists($,J={}){try{return await this.requestRaw($,J),!0}catch(Z){if(MJ(Z)&&Z.statusCode===404)return!1;throw Z}}async requestRaw($,J={}){let{method:Z="GET",params:H,body:K,headers:W={}}=J,_=this.buildUrl($,H),F=J.requestId??Z5(),x=Date.now()+this.config.timeout;for(let V=0;V<U8;V++){let A=x-Date.now();if(A<=0)throw this.log.error("Request timeout before attempt",D2()?{requestId:F,path:$}:{path:$}),new s("Request timeout",408);let C=await this.getAccessToken(),q=V===U8-1,z=performance.now();this.log.debug(`→ ${Z} ${_}`,{requestId:F,attempt:V>0?V+1:void 0});let N={"Content-Type":"application/json",Accept:"application/json","X-Request-ID":F,...W};if(C)N.Authorization=`Bearer ${C}`;let I=await this.config.fetch(_,{method:Z,headers:N,body:K?JSON.stringify(K):void 0,signal:AbortSignal.timeout(Math.min(A,this.config.timeout))}),r=Math.round(performance.now()-z);if(this.log.debug(`← ${I.status} ${I.statusText} (${r}ms)`,{requestId:F}),D5.includes(I.status)&&!q){let v2=I.headers.get("Retry-After"),p=this.parseRetryAfter(v2,V),i8=x-Date.now();if(p>=i8)throw this.log.error("Request timeout during retry backoff",D2()?{requestId:F,path:$}:{path:$}),new s("Request timeout",408);this.log.warn(`Retrying in ${p}ms (attempt ${V+1}/${U8})`,{...D2()&&{requestId:F},status:I.status}),await this.sleep(p);continue}if(!I.ok)return this.handleErrorResponse(I,F);return I}throw this.log.error("Max retries exceeded",D2()?{requestId:F,path:$}:{path:$}),new s("Max retries exceeded")}getAccessToken(){if(!this.config.tokenProvider)return Promise.resolve(void 0);return this.config.tokenProvider.getToken()}buildUrl($,J){if(/^[a-z][a-z0-9+.-]*:/i.test($))throw Error(`Absolute URLs are not allowed in path: ${$}. Use relative paths only.`);let Z=$.startsWith("/")?$.slice(1):$,H=new URL(Z,this.config.baseUrl);if(J){for(let[K,W]of Object.entries(J))if(W!==void 0)H.searchParams.set(K,String(W))}return H.toString()}async handleResponse($){if(!$.ok)await this.handleErrorResponse($);if($.status===204)return;if($.headers.get("content-length")==="0")return;return this.parseJsonResponse($)}async parseJsonResponse($){let J=await $.text();if(!J||J.trim()==="")return;try{return JSON.parse(J)}catch(Z){let H=J.length>200?J.slice(0,200)+"...":J,K=Z instanceof Error?Z.message:String(Z),W=$.url||"unknown";throw this.log.error("Failed to parse JSON response",{url:W,status:$.status,contentType:$.headers.get("content-type"),bodyPreview:H,error:K}),new s(`Invalid JSON response from ${W}`,$.status,{parseError:K,body:H})}}async handleErrorResponse($,J){let Z,H=await $.text().catch(()=>"");if(H)try{Z=JSON.parse(H)}catch(W){let _=W instanceof Error?W.message:"Unknown parse error";Z={rawBody:H.slice(0,500),parseError:_},this.log.warn("Failed to parse error response as JSON",{...D2()&&{requestId:J},url:$.url,status:$.status,parseError:_,bodyPreview:H.slice(0,200)})}let K=this.extractErrorMessage(Z,$.statusText);if($.status!==404)this.log.error(`Request failed: ${$.status} ${K}`,D2()?{requestId:J}:void 0);switch($.status){case 401:throw this.config.tokenProvider?.invalidate?.(),new $9(K,Z);case 403:throw new J9(K,Z);case 404:throw new Z9(K,Z);case 422:throw new H9(K,Z);default:throw new s(K,$.status,Z)}}extractErrorMessage($,J){if(typeof $==="object"&&$!==null){let Z=$;if(typeof Z.message==="string")return Z.message;if(typeof Z.error==="string")return Z.error;if(typeof Z.imsx_description==="string")return Z.imsx_description}return J}sleep($){return new Promise((J)=>{setTimeout(J,$)})}parseRetryAfter($,J){if(!$)return t8*Math.pow(2,J);let Z=parseInt($,10);if(!isNaN(Z))return Z*1000;let H=Date.parse($);if(!isNaN(H)){let K=H-Date.now();return Math.max(0,K)}return t8*Math.pow(2,J)}}function w5($){if($ instanceof Date)return $.toISOString();if(typeof $==="boolean")return $?"true":"false";if(typeof $==="number")return String($);return $.replaceAll("'","''")}function t($){let J=w5($);return typeof $==="string"||$ instanceof Date?`'${J}'`:J}function x5($,J){if(typeof J==="string"||typeof J==="number"||typeof J==="boolean"||J instanceof Date)return[`${$}=${t(J)}`];if(typeof J==="object"&&J!==null){let Z=J,H=[];if(Z.ne!==void 0)H.push(`${$}!=${t(Z.ne)}`);if(Z.gt!==void 0)H.push(`${$}>${t(Z.gt)}`);if(Z.gte!==void 0)H.push(`${$}>=${t(Z.gte)}`);if(Z.lt!==void 0)H.push(`${$}<${t(Z.lt)}`);if(Z.lte!==void 0)H.push(`${$}<=${t(Z.lte)}`);if(Z.contains!==void 0)H.push(`${$}~${t(Z.contains)}`);if(Z.in!==void 0&&Z.in.length>0){let K=Z.in.map((_)=>`${$}=${t(_)}`),W=K.join(" OR ");H.push(K.length>1?`(${W})`:W)}if(Z.notIn!==void 0&&Z.notIn.length>0){let K=Z.notIn.map((W)=>`${$}!=${t(W)}`);H.push(K.join(" AND "))}return H}return[]}function U5($){return"OR"in $&&Array.isArray($.OR)}function D9($){if(U5($)){let Z=$.OR.map((H)=>D9(H)).filter((H)=>H!==void 0);return Z.length>0?Z.join(" OR "):void 0}let J=[];for(let[Z,H]of Object.entries($))if(H!==void 0)J.push(...x5(Z,H));return J.length>0?J.join(" AND "):void 0}var e8=100,M5=1e4;class A8{fetcher;path;params;max;unwrapKey;log;transform;paginationStyle;constructor($){this.fetcher=$.fetcher,this.path=$.path,this.params=$.params??{},this.max=$.max,this.unwrapKey=$.unwrapKey,this.log=$.logger?.child("pagination")??q8("pagination"),this.transform=$.transform,this.paginationStyle=$.paginationStyle??"offset"}buildRequestParams($,J){let{where:Z,fields:H,offset:K,limit:W,..._}=this.params,F={..._,filter:Z?D9(Z):void 0,fields:H?.join(","),limit:$};if(this.paginationStyle==="page"){let O=Math.floor(J/$)+1;return{...F,page:O}}return{...F,offset:J}}extractItems($,J){let Z;if(this.unwrapKey)Z=$[this.unwrapKey]??[];else Z=$;return this.validateItems(Z,J)}validateItems($,J){if($===null||$===void 0)return this.log.warn(`Page ${J}: response data is ${$}, treating as empty`),[];if(!Array.isArray($))throw Error(`Expected array for page ${J}, got ${typeof $}. `+(this.unwrapKey?`Check unwrapKey '${this.unwrapKey}' is correct.`:""));return $}hasMorePages($,J,Z,H){if(J===0)return!1;let K=$.hasMore,W=$.total!==void 0&&Z+J<$.total;return K||W||J===H&&$.total===void 0}async*[Symbol.asyncIterator](){let $=this.params.offset??0,J=this.params.limit??e8,Z=!0,H=1,K=0;while(Z){this.log.debug(`Fetching page ${H} (offset: ${$}, limit: ${J})`);let W=await this.fetcher(this.path,{params:this.buildRequestParams(J,$)}),_=this.extractItems(W.data,H);this.log.debug(`Page ${H}: ${_.length} items`);for(let F of _)if(yield this.transform?this.transform(F):F,K++,this.max!==void 0&&K>=this.max){this.log.debug(`Pagination complete: reached max of ${this.max} items`);return}Z=this.hasMorePages(W,_.length,$,J),$+=_.length,H++}this.log.debug(`Pagination complete: ${K} total items`)}async toArray($={}){let J=$.maxItems??M5,Z=[];for await(let H of this){if(Z.length>=J)throw Error(`toArray() exceeded maxItems limit of ${J}. Use 'for await...of' to stream large datasets, or increase maxItems.`);Z.push(H)}return Z}async firstPage(){let $=this.params.limit??e8,J=this.params.offset??0,Z=await this.fetcher(this.path,{params:this.buildRequestParams($,J)}),H=this.extractItems(Z.data,1),K=this.transform?H.map(this.transform):H,W=this.hasMorePages(Z,K.length,J,$);return this.log.debug(`First page: ${K.length} items, total: ${Z.total}, hasMore: ${W}`),{data:K,hasMore:W,total:Z.total,nextOffset:W?J+K.length:void 0}}}function f2($,J){return{path:$,message:J}}function o($,J,Z){let H=$.safeParse(J);if(H.success)return H.data;let K=H.error.issues.map((W)=>({path:W.path.join(".")||"(root)",message:W.message}));throw N2(`Invalid ${Z} data`,K)}function u2($,J){if(typeof $!=="string"||$.trim()==="")throw N2(`Invalid ${J}`,[f2(J,"Must be a non-empty string")])}function z8($){if(!$)return;let J=[];if($.limit!==void 0){if(!Number.isInteger($.limit)||$.limit<=0)J.push(f2("limit","Must be a positive integer"))}if($.offset!==void 0){if(!Number.isInteger($.offset)||$.offset<0)J.push(f2("offset","Must be a non-negative integer"))}if($.max!==void 0){if(!Number.isInteger($.max)||$.max<=0)J.push(f2("max","Must be a positive integer"))}if(J.length>0)throw N2("Invalid list parameters",J)}var w9={baseUrl:["TIMEBACK_API_BASE_URL","TIMEBACK_BASE_URL","CALIPER_BASE_URL"],authUrl:["TIMEBACK_API_AUTH_URL","TIMEBACK_AUTH_URL","CALIPER_TOKEN_URL"],clientId:["TIMEBACK_API_CLIENT_ID","TIMEBACK_CLIENT_ID","CALIPER_CLIENT_ID"],clientSecret:["TIMEBACK_API_CLIENT_SECRET","TIMEBACK_CLIENT_SECRET","CALIPER_CLIENT_SECRET"]},k2="http://purl.imsglobal.org/ctx/caliper/v1p2",x9="QUESTION_RESULT";import{z as j}from"zod/v4";import{z as Y}from"zod/v4";import{z as Z2}from"zod/v4";import{z as U}from"zod/v4";import{z as w}from"zod/v4";import{z as k}from"zod/v4";import{z as D}from"zod/v4";import{z as g}from"zod/v4";import{z as Q}from"zod/v4";import{z as B}from"zod/v4";import{z as G}from"zod/v4";import{z as A2}from"zod/v4";var S=q8("caliper");class q9 extends C8{paths;constructor($){super({config:$,logger:S});this.paths=$.paths}async requestPaginated($,J={}){let Z=await this.request($,J),{events:H,pagination:K}=Z,_=(J.params?.offset??0)+H.length<K.total;return{data:H,hasMore:_,total:K.total}}}class C9 extends A8{constructor($,J,Z={}){z8(Z);let{max:H,...K}=Z;super({fetcher:(W,_)=>$.requestPaginated(W,_),path:J,params:K,max:H,logger:S})}}function P8($){let J=$.id??`urn:uuid:${crypto.randomUUID()}`,Z=$.metricsId??`urn:uuid:${crypto.randomUUID()}`;return{"@context":k2,id:J,type:"ActivityEvent",action:"Completed",actor:$.actor,object:$.object,eventTime:$.eventTime??new Date().toISOString(),profile:"TimebackProfile",generated:{id:Z,type:"TimebackActivityMetricsCollection",items:$.metrics,...$.attempt===void 0?{}:{attempt:$.attempt},...$.generatedExtensions?{extensions:$.generatedExtensions}:{}},extensions:$.extensions,...$.edApp===void 0?{}:{edApp:$.edApp},...$.session===void 0?{}:{session:$.session}}}function T8($){let J=$.id??`urn:uuid:${crypto.randomUUID()}`,Z=$.metricsId??`urn:uuid:${crypto.randomUUID()}`;return{"@context":k2,id:J,type:"TimeSpentEvent",action:"SpentTime",actor:$.actor,object:$.object,eventTime:$.eventTime??new Date().toISOString(),profile:"TimebackProfile",generated:{id:Z,type:"TimebackTimeSpentMetricsCollection",items:$.metrics},extensions:$.extensions,...$.edApp===void 0?{}:{edApp:$.edApp},...$.session===void 0?{}:{session:$.session}}}function j5($){return{"@context":k2,id:$.id??`urn:uuid:${crypto.randomUUID()}`,type:"AssessmentItemEvent",action:"Started",profile:"AssessmentProfile",actor:$.actor,object:{...$.object,type:"AssessmentItem"},eventTime:$.eventTime??new Date().toISOString(),edApp:$.edApp,...$.session===void 0?{}:{session:$.session},...$.extensions===void 0?{}:{extensions:$.extensions}}}function L5($){return{"@context":k2,id:$.id??`urn:uuid:${crypto.randomUUID()}`,type:"AssessmentItemEvent",action:"Completed",profile:"AssessmentProfile",actor:$.actor,object:{...$.object,type:"AssessmentItem"},eventTime:$.eventTime??new Date().toISOString(),edApp:$.edApp,...$.generated===void 0?{}:{generated:{...$.generated,type:"Response"}},...$.session===void 0?{}:{session:$.session},...$.extensions===void 0?{}:{extensions:$.extensions}}}function V5($){let{id:J,...Z}=$.generated;return{"@context":k2,id:$.id??`urn:uuid:${crypto.randomUUID()}`,type:"GradeEvent",action:"Graded",profile:"GradingProfile",actor:$.actor,object:$.object,eventTime:$.eventTime??new Date().toISOString(),edApp:$.edApp,generated:{...Z,id:J??`urn:uuid:${crypto.randomUUID()}`,type:"Score",scoreType:x9},...$.session===void 0?{}:{session:$.session},...$.extensions===void 0?{}:{extensions:$.extensions}}}var k5="http://purl.imsglobal.org/ctx/caliper/v1p2",q5="urn:tag:auto-attach",Q2={PERSON:"Person",ASSIGNABLE_DIGITAL_RESOURCE:"AssignableDigitalResource",ACTIVITY_METRIC:"ActivityMetric",TIME_SPENT_METRIC:"TimeSpentMetric",ACTIVITY_METRICS_COLLECTION:"ActivityMetricsCollection",TIME_SPENT_METRICS_COLLECTION:"TimeSpentMetricsCollection",COURSE_OFFERING:"CourseOffering",SOFTWARE_APPLICATION:"SoftwareApplication"},C5=new Set(["ActivityEvent","TimeSpentEvent","AssignableEvent","ViewEvent"]);class A9{transformEnvelope($){let{data:J}=$;if(!Array.isArray(J))return $;let Z=J.map((H)=>x2(H)?A5(H):H);return{...$,data:Z}}}function A5($){let J={...$};if(!("@context"in J))J["@context"]=k5;if(J.profile==="TimebackProfile")J.profile="AggregationProfile";if(x2(J.actor)&&J.actor.type==="TimebackUser")J.actor=z5(J.actor);if(x2(J.object))J.object=P5(J.object,J.type);if(x2(J.generated)){let Z=J.type;if(Z==="ActivityEvent")J.generated=T5(J.generated);else if(Z==="TimeSpentEvent")J.generated=R5(J.generated)}if(!J.session)J.session=q5;return J}function z5($){return{type:Q2.PERSON,id:$.id}}function P5($,J){if(!($.type==="TimebackActivityContext"&&J!==void 0&&C5.has(J)))return{...$};let H={type:Q2.ASSIGNABLE_DIGITAL_RESOURCE,id:$.id};if($.name)H.name=$.name;if(x2($.course)){let K={type:Q2.COURSE_OFFERING,id:$.course.id};if($.course.name)K.name=$.course.name;H.isPartOf=K}if(x2($.app))H.extensions={app:{name:$.app.name,type:$.app.type??Q2.SOFTWARE_APPLICATION}};return H}function z9($,J,Z,H){let W={id:typeof $?.id==="string"?$.id:`urn:uuid:${crypto.randomUUID()}`,type:J};if(Array.isArray($.items))W.items=$.items.map((_)=>{if(!x2(_))return _;let F=typeof _.type==="string"?_.type:"",O=F?F[0].toUpperCase()+F.slice(1):H;return{id:`urn:uuid:${crypto.randomUUID()}`,type:Z,metricType:O,metricValue:_.value}});for(let[_,F]of Object.entries($))if(_!=="id"&&_!=="type"&&_!=="items")W[_]=F;return W}function T5($){return z9($,Q2.ACTIVITY_METRICS_COLLECTION,Q2.ACTIVITY_METRIC,"")}function R5($){return z9($,Q2.TIME_SPENT_METRICS_COLLECTION,Q2.TIME_SPENT_METRIC,"Active")}function x2($){return typeof $==="object"&&$!==null&&!Array.isArray($)}function N5($,J=l2){return O9($,w9,J)}var I5=/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(?:\.\d+)?(?:Z|[+-]\d{2}:\d{2})$/,y=j.string().min(1).regex(I5,"must be a valid ISO 8601 datetime"),C2=j.string().min(1).regex(/^\d{4}-\d{2}-\d{2}$/,"must be a valid ISO 8601 date (YYYY-MM-DD)"),X=j.string().trim().min(1),X2=j.enum(["Reading","Language","Vocabulary","Social Studies","Writing","Science","FastMath","Math","None","Other"]).meta({id:"TimebackSubject",description:"Subject area"}),b5=j.union([j.literal(-1),j.literal(0),j.literal(1),j.literal(2),j.literal(3),j.literal(4),j.literal(5),j.literal(6),j.literal(7),j.literal(8),j.literal(9),j.literal(10),j.literal(11),j.literal(12),j.literal(13)]),h5=j.string().transform(($,J)=>{let Z=$.toLowerCase().trim().replaceAll("_"," ");if(Z==="")return J.addIssue({code:"custom",message:"must be a valid Timeback grade"}),j.NEVER;let H=Z.replace(/\bgrade\b/g,"").replace(/(\d+)(st|nd|rd|th)\b/g,"$1").trim();if(H==="")return J.addIssue({code:"custom",message:"must be a valid Timeback grade"}),j.NEVER;if(H==="pre-k"||H==="pk")return-1;if(H==="k")return 0;if(H==="middle school")return 7;let W=H.replace(/\s+/g,"").replace(/^0+/,"")||"0",_=Number(W);if(!Number.isInteger(_))return J.addIssue({code:"custom",message:"must be a valid Timeback grade"}),j.NEVER;return _}),n=j.union([j.number(),h5]).pipe(b5).meta({id:"TimebackGrade",description:"Grade level (-1 = Pre-K, 0 = K, 1-12 = grades, 13 = AP)"}),S5=j.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),E5=j.enum(["department","school","district","local","state","national"]),P9=j.enum(["administrator","aide","guardian","parent","proctor","relative","student","teacher"]),v5=j.enum(["administrator","proctor","student","teacher"]),y5=j.enum(["qti","text","audio","video","interactive","visual","course-material","assessment-bank"]),f5=j.enum(["qti-test","qti-question","qti-stimulus","qti-test-bank"]),g5=j.enum(["unit","course","resource-collection"]),d5=j.enum(["choice","order","associate","match","hotspot","hottext","select-point","graphic-order","graphic-associate","graphic-gap-match","text-entry","extended-text","inline-choice","upload","slider","drawing","media","custom"]),m5=j.enum(["easy","medium","hard"]),T9=j.array(j.object({source:j.string(),learningObjectiveIds:j.array(j.string())})),R9=j.object({consecutive_failures:j.number().int().min(1).optional(),stagnation_limit:j.number().int().min(1).optional()}).optional(),l5=j.enum(["powerpath-100","quiz","test-out","placement","unit-test","alpha-read-article"]).nullable(),R1=j.object({imsx_codeMajor:j.enum(["failure","success"]),imsx_severity:j.enum(["error","warning","status"]),imsx_description:j.string(),imsx_CodeMinor:j.object({imsx_codeMinorField:j.array(j.object({imsx_codeMinorFieldName:j.string(),imsx_codeMinorFieldValue:j.string()}))}).optional()}),u5=Y.union([Y.url(),Y.string().regex(/^urn:/,"Must be a URL or URN")]),U2=Y.object({id:Y.url(),type:Y.literal("TimebackUser"),email:Y.string()}),R8=Y.object({id:Y.url(),type:Y.literal("TimebackActivityContext"),subject:X2,app:Y.object({name:Y.string()}),activity:Y.object({id:Y.url().optional(),name:Y.string()}).strict().optional(),course:Y.object({id:Y.url().optional(),name:Y.string().optional()}).strict(),process:Y.boolean().optional()}),N9=Y.object({"@context":Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),id:Y.string().regex(/^urn:uuid:/,"Event id must start with urn:uuid:"),type:Y.string(),eventTime:y,profile:Y.literal("TimebackProfile"),actor:U2,action:Y.string(),object:R8,edApp:Y.object({id:Y.string(),name:Y.string().optional()}).optional()}),I9=Y.object({type:Y.enum(["totalQuestions","correctQuestions","xpEarned","masteredUnits"]),value:Y.number()}),c5=Y.object({id:Y.url(),type:Y.literal("TimebackActivityMetricsCollection"),attempt:Y.number().optional(),items:Y.array(I9),extensions:Y.record(Y.string(),Y.unknown()).optional()}),p5=N9.extend({type:Y.literal("ActivityEvent"),action:Y.literal("Completed"),generated:c5}),b9=Y.object({type:Y.enum(["active","inactive","waste","unknown","anti-pattern"]),value:Y.number(),subType:Y.string().optional()}),a5=Y.object({id:Y.url(),type:Y.literal("TimebackTimeSpentMetricsCollection"),items:Y.array(b9)}),i5=N9.extend({type:Y.literal("TimeSpentEvent"),action:Y.literal("SpentTime"),generated:a5}),s5=Y.object({actor:U2,object:R8,metrics:Y.array(I9).min(1,"metrics must contain at least one metric"),eventTime:y.optional(),metricsId:Y.string().optional(),id:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional(),edApp:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),session:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),attempt:Y.number().int().min(1).optional(),generatedExtensions:Y.object({pctCompleteApp:Y.number().optional()}).loose().optional()}).strict(),o5=Y.object({actor:U2,object:R8,metrics:Y.array(b9).min(1,"metrics must contain at least one metric"),eventTime:y.optional(),metricsId:Y.string().optional(),id:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional(),edApp:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),session:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional()}).strict(),n5=Y.union([p5,i5]),N1=Y.object({sensor:Y.string(),sendTime:y,dataVersion:Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),data:Y.array(n5)}),r5=Y.enum(["AnnotationProfile","AssessmentProfile","ToolUseProfile","GeneralProfile","FeedbackProfile","MediaProfile","SurveyProfile","ResourceManagementProfile","ForumProfile","AssignableProfile","GradingProfile","ReadingProfile","SessionProfile","SearchProfile","ToolLaunchProfile","TimebackProfile"]),l=Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]),c2=Y.object({id:u5,type:Y.string(),name:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),h9=Y.object({"@context":Y.string().optional(),id:Y.string().regex(/^urn:uuid:/,"Event id must start with urn:uuid:"),type:Y.string(),actor:Y.union([Y.string(),c2,U2]),action:Y.string(),object:l,eventTime:y,profile:r5,edApp:l.optional(),generated:l.optional(),target:l.optional(),referrer:l.optional(),group:l.optional(),membership:l.optional(),session:l.optional(),federatedSession:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),U9=Y.object({sensor:X,sendTime:y,dataVersion:Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),data:Y.array(h9).min(1,"data must contain at least one event")}),t5=Y.object({sensor:X,events:Y.array(h9).min(1,"events must contain at least one event")}),M9=Y.object({limit:Y.number().int().positive().optional(),offset:Y.number().int().min(0).optional(),sensor:X.optional(),startDate:y.optional(),endDate:y.optional(),actorId:X.optional(),actorEmail:Y.email().optional()}).strict(),S9=Y.object({id:Y.string(),name:Y.string().optional(),isPartOf:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),e5=Y.object({id:Y.string(),attempt:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),startedAtTime:y.optional(),endedAtTime:y.optional(),duration:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),$Z=Y.object({id:Y.string().optional(),scoreGiven:Y.number(),maxScore:Y.number().optional(),attempt:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),comment:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),JZ=Y.object({actor:Y.union([Y.string(),c2,U2]),object:S9,edApp:l,id:Y.string().optional(),eventTime:y.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),ZZ=Y.object({actor:Y.union([Y.string(),c2,U2]),object:S9,edApp:l,generated:e5.optional(),id:Y.string().optional(),eventTime:y.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),HZ=Y.object({actor:Y.union([Y.string(),c2,U2]),object:Y.string(),generated:$Z,edApp:l,id:Y.string().optional(),eventTime:y.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),I1=Z2.object({name:X,targetUrl:Z2.url("targetUrl must be a valid URL"),secret:X,active:Z2.boolean(),sensor:Z2.string().nullable().optional(),description:Z2.string().nullable().optional()}).strict(),KZ=Z2.enum(["string","number","boolean"]),XZ=Z2.enum(["eq","neq","gt","gte","lt","lte","contains","notContains","in","notIn","startsWith","endsWith","regexp"]),b1=Z2.object({webhookId:X,filterKey:X,filterValue:X,filterType:KZ,filterOperator:XZ,active:Z2.boolean()}).strict(),p2=U.string().uuid(),j9=U.object({title:X,identifier:p2,uri:X}),WZ=U.object({identifier:p2,uri:X,lastChangeDateTime:X,title:X,creator:X,officialSourceURL:U.string().optional(),publisher:U.string().optional(),description:U.string().optional(),language:U.string().optional(),version:U.string().optional(),caseVersion:U.string().optional(),adoptionStatus:U.string().optional(),statusStartDate:U.string().optional(),statusEndDate:U.string().optional(),licenseUri:U.string().optional(),notes:U.string().optional(),subject:U.array(U.string()).optional(),extensions:U.unknown().optional()}),YZ=U.object({identifier:p2,uri:X,lastChangeDateTime:X,fullStatement:X,alternativeLabel:U.string().optional(),CFItemType:U.string().optional(),cfItemType:U.string().optional(),humanCodingScheme:U.string().optional(),listEnumeration:U.string().optional(),abbreviatedStatement:U.string().optional(),conceptKeywords:U.array(U.string()).optional(),notes:U.string().optional(),subject:U.array(U.string()).optional(),language:U.string().optional(),educationLevel:U.array(U.string()).optional(),CFItemTypeURI:U.unknown().optional(),licenseURI:U.unknown().optional(),statusStartDate:U.string().optional(),statusEndDate:U.string().optional(),extensions:U.unknown().optional()}),QZ=U.object({identifier:p2,uri:X,lastChangeDateTime:X,associationType:X,originNodeURI:j9,destinationNodeURI:j9,sequenceNumber:U.number().optional(),extensions:U.unknown().optional()}),GZ=U.object({CFItemTypes:U.array(U.unknown()).optional(),CFSubjects:U.array(U.unknown()).optional(),CFConcepts:U.array(U.unknown()).optional(),CFLicenses:U.array(U.unknown()).optional(),CFAssociationGroupings:U.array(U.unknown()).optional(),extensions:U.unknown().optional()}),h1=U.object({CFDocument:WZ,CFItems:U.array(YZ),CFAssociations:U.array(QZ),CFDefinitions:GZ.optional(),extensions:U.unknown().optional()}),L9="https://www.w3.org/ns/credentials/v2",BZ=w.array(w.url()).min(3,"@context must include W3C, CLR, and OB context URLs").refine(($)=>$[0]===L9,`First @context entry must be "${L9}"`),E9=w.object({id:w.url(),type:w.literal("Image"),caption:w.string().optional()}),N8=w.object({id:w.url(),type:w.array(w.string()).min(1).refine(($)=>$.includes("Profile"),'type must include "Profile"'),name:w.string().optional(),url:w.string().url().optional(),phone:w.string().optional(),description:w.string().optional(),image:E9.optional(),email:w.string().email().optional()}),v9=w.object({type:X,proofPurpose:X,verificationMethod:X,created:w.iso.datetime(),proofValue:X,cryptosuite:w.string().optional()}),_Z=w.object({id:X,type:w.literal("1EdTechJsonSchemaValidator2019")}),FZ=w.object({id:w.string().url().optional(),narrative:w.string().optional()}),OZ=w.object({id:w.url(),type:w.array(w.string()).min(1).refine(($)=>$.includes("Achievement"),'type must include "Achievement"'),name:X,description:X,criteria:FZ,image:E9.optional(),achievementType:w.string().optional(),creator:N8.optional()}),DZ=w.object({type:w.literal("IdentityObject"),identityHash:X,identityType:X,hashed:w.boolean(),salt:w.string().optional()}),wZ=w.enum(["exactMatchOf","isChildOf","isParentOf","isPartOf","isPeerOf","isRelatedTo","precedes","replacedBy"]),xZ=w.object({type:w.literal("Association"),associationType:wZ,sourceId:w.url(),targetId:w.url()}),UZ=w.object({"@context":w.array(w.string()).min(1),id:w.url(),type:w.array(w.string()).min(1).refine(($)=>$.includes("VerifiableCredential"),'type must include "VerifiableCredential"'),issuer:N8,validFrom:w.iso.datetime(),validUntil:w.string().datetime().optional(),credentialSubject:w.object({id:w.string().optional()}),proof:w.array(v9).optional()}),MZ=w.object({id:w.string().url().optional(),type:w.array(w.string()).min(1).refine(($)=>$.includes("ClrSubject"),'type must include "ClrSubject"'),identifier:w.array(DZ).optional(),achievement:w.array(OZ).optional(),verifiableCredential:w.array(UZ).min(1),association:w.array(xZ).optional()}),S1=w.object({"@context":BZ,id:w.url(),type:w.array(w.string()).min(1).refine(($)=>$.includes("VerifiableCredential")&&$.includes("ClrCredential"),'type must include "VerifiableCredential" and "ClrCredential"'),issuer:N8,name:X,description:w.string().optional(),validFrom:w.iso.datetime(),validUntil:w.iso.datetime().optional(),credentialSubject:MZ,proof:w.array(v9).optional(),credentialSchema:w.array(_Z).optional()}),jZ=k.object({staging:k.string().meta({description:"Course ID in staging environment"}).optional(),production:k.string().meta({description:"Course ID in production environment"}).optional()}).meta({id:"CourseIds",description:"Environment-specific course IDs (populated by sync)"}),LZ=k.enum(["base","hole-filling","optional"]).meta({id:"CourseType",description:"Course classification type"}),VZ=k.enum(["draft","testing","published","deactivated"]).meta({id:"PublishStatus",description:"Course publication status"}),kZ=k.object({dailyXp:k.number().int().positive().meta({description:"Target XP to earn per day"}).optional(),dailyLessons:k.number().int().positive().meta({description:"Target lessons to complete per day"}).optional(),dailyActiveMinutes:k.number().int().positive().meta({description:"Target active learning minutes per day"}).optional(),dailyAccuracy:k.number().int().min(0).max(100).meta({description:"Target accuracy percentage (0-100)"}).optional(),dailyMasteredUnits:k.number().int().positive().meta({description:"Target units to master per day"}).optional()}).meta({id:"CourseGoals",description:"Daily learning goals for a course"}),qZ=k.object({totalXp:k.number().int().positive().meta({description:"Total XP available in the course"}).optional(),totalLessons:k.number().int().positive().meta({description:"Total number of lessons/activities"}).optional(),totalGrades:k.number().int().positive().meta({description:"Total grade levels covered"}).optional()}).meta({id:"CourseMetrics",description:"Aggregate metrics for a course"}),y9=k.object({courseType:LZ.optional(),isSupplemental:k.boolean().meta({description:"Whether this is supplemental to a base course"}).optional(),isCustom:k.boolean().meta({description:"Whether this is a custom course for an individual student"}).optional(),publishStatus:VZ.optional(),contactEmail:k.email().meta({description:"Contact email for course issues"}).optional(),primaryApp:k.string().meta({description:"Primary application identifier"}).optional(),goals:kZ.optional(),metrics:qZ.optional()}).meta({id:"CourseMetadata",description:"Course metadata (matches API metadata object)"}),f9=k.object({courseCode:k.string().meta({description:"Course code (e.g., 'MATH101')"}).optional(),level:k.string().meta({description:"Course level (e.g., 'AP', 'Honors')"}).optional(),metadata:y9.optional()}).meta({id:"CourseDefaults",description:"Default properties that apply to all courses unless overridden"}),V9=k.object({level:k.string().meta({description:"Course level for this environment"}).optional(),sensor:k.url().meta({description:"Caliper sensor endpoint URL for this environment"}).optional(),launchUrl:k.url().meta({description:"LTI launch URL for this environment"}).optional(),metadata:y9.optional()}).meta({id:"CourseEnvOverrides",description:"Environment-specific course overrides (non-identity fields)"}),CZ=k.object({staging:V9.meta({description:"Overrides for staging environment"}).optional(),production:V9.meta({description:"Overrides for production environment"}).optional()}).meta({id:"CourseOverrides",description:"Per-environment course overrides"}),AZ=f9.extend({subject:X2.meta({description:"Subject area for this course"}),grade:n.meta({description:"Grade level (-1 = Pre-K, 0 = K, 1-12 = grades, 13 = AP)"}).optional(),ids:jZ.nullable().optional(),sensor:k.url().meta({description:"Caliper sensor endpoint URL for this course"}).optional(),launchUrl:k.url().meta({description:"LTI launch URL for this course"}).optional(),overrides:CZ.optional()}).meta({id:"CourseConfig",description:"Configuration for a single course. Must have either grade or courseCode (or both)."}),E1=k.object({$schema:k.string().meta({description:"JSON Schema reference for editor support"}).optional(),name:k.string().min(1,"App name is required").meta({description:"Display name for your app"}),defaults:f9.meta({description:"Default properties applied to all courses"}).optional(),courses:k.array(AZ).min(1,"At least one course is required").meta({description:"Courses available in this app"}),sensor:k.url().meta({description:"Default Caliper sensor endpoint URL for all courses"}).optional(),launchUrl:k.url().meta({description:"Default LTI launch URL for all courses"}).optional(),studio:k.object({telemetry:k.boolean().meta({description:"Enable anonymous usage telemetry for Studio (default: true)"}).optional().default(!0)}).meta({description:"Studio-specific configuration"}).optional()}).meta({id:"TimebackConfig",title:"Timeback Config",description:"Configuration schema for timeback.config.json files"}).refine(($)=>{return $.courses.every((J)=>J.grade!==void 0||J.courseCode!==void 0)},{message:"Each course must have either a grade or a courseCode",path:["courses"]}).refine(($)=>{let Z=$.courses.filter((H)=>H.grade!==void 0).map((H)=>`${H.subject}:${H.grade}`);return new Set(Z).size===Z.length},{message:"Duplicate (subject, grade) pair found; each must be unique",path:["courses"]}).refine(($)=>{let Z=$.courses.filter((H)=>H.courseCode!==void 0).map((H)=>H.courseCode);return new Set(Z).size===Z.length},{message:"Duplicate courseCode found; each must be unique",path:["courses"]}).refine(($)=>{return $.courses.every((J)=>{if(J.sensor!==void 0||$.sensor!==void 0)return!0;return[J.launchUrl,$.launchUrl,J.overrides?.staging?.launchUrl,J.overrides?.production?.launchUrl].filter(Boolean).length>0})},{message:"Each course must have an effective sensor. Either set `sensor` explicitly (top-level or per-course), or provide a `launchUrl` so sensor can be derived from its origin.",path:["courses"]}),H2=D.union([y,C2]),v1=D.object({id:D.string(),role:D.string(),beginDate:H2.nullable(),endDate:H2.nullable(),metadata:D.object({goals:D.object({dailyXp:D.number().optional()}).optional(),metrics:D.object({totalXp:D.number().optional(),totalLessons:D.number().optional()}).optional()}).optional(),course:D.object({id:D.string(),title:D.string(),subjects:D.array(D.string()).nullable(),grades:D.array(D.string()).nullable()}),school:D.object({id:D.string(),name:D.string()}),testOutSupported:D.boolean(),testOutEligible:D.boolean()}),zZ=D.object({activityMetrics:D.object({xpEarned:D.number(),totalQuestions:D.number(),correctQuestions:D.number(),masteredUnits:D.number()}),timeSpentMetrics:D.object({activeSeconds:D.number(),inactiveSeconds:D.number(),wasteSeconds:D.number()}),apps:D.array(D.string())}),PZ=D.record(D.string(),D.record(D.string(),zZ)),y1=D.object({message:D.string(),enrollmentId:D.string(),startDate:H2,endDate:H2,facts:PZ,factsByApp:D.unknown()}),g9=D.object({email:D.email().optional(),studentId:X.optional()}).superRefine(($,J)=>{if(!$.email&&!$.studentId)J.addIssue({code:D.ZodIssueCode.custom,message:"must provide either email or studentId",path:["email"]}),J.addIssue({code:D.ZodIssueCode.custom,message:"must provide either email or studentId",path:["studentId"]})}),f1=D.object({subject:X,grade:X,courseId:X,orgSourcedId:X.optional()}),g1=D.object({userId:X}),d1=D.object({sourcedId:X.optional(),role:v5.optional(),beginDate:y.optional(),metadata:D.record(D.string(),D.unknown()).optional()}),m1=D.object({fields:D.string().optional(),limit:D.number().int().positive().optional(),offset:D.number().int().nonnegative().optional(),sort:D.string().optional(),orderBy:D.enum(["asc","desc"]).optional(),filter:D.string().optional(),search:D.string().optional(),roles:D.array(P9).min(1),orgSourcedIds:D.array(X).optional()}),l1=g9.extend({startDate:H2,endDate:H2,timezone:D.string().optional()}),u1=g9.extend({weekDate:H2,timezone:D.string().optional()}),c1=D.object({enrollmentId:X,startDate:H2.optional(),endDate:H2.optional(),timezone:D.string().optional()}),p1=g.object({email:g.email()}),a1=g.object({name:X.optional(),timeback_id:X.optional(),grade:X.optional(),subject:X.optional(),all:g.boolean().optional()}),i1=g.object({student_email:g.email(),timeback_id:X.optional(),subject:X.optional(),grade_rank:g.number().int().min(0).max(12).optional(),assessment_line_item_sourced_id:X.optional(),assessment_result_sourced_id:X.optional()}).superRefine(($,J)=>{let Z=!!$.timeback_id,H=!!$.subject,K=$.grade_rank!==void 0;if(!Z&&!H)J.addIssue({code:g.ZodIssueCode.custom,message:"must provide either timeback_id or subject",path:["timeback_id"]}),J.addIssue({code:g.ZodIssueCode.custom,message:"must provide either timeback_id or subject",path:["subject"]});if(K&&!H)J.addIssue({code:g.ZodIssueCode.custom,message:"grade_rank requires subject",path:["grade_rank"]});let W=!!$.assessment_line_item_sourced_id,_=!!$.assessment_result_sourced_id;if(W!==_)J.addIssue({code:g.ZodIssueCode.custom,message:"assessment_line_item_sourced_id and assessment_result_sourced_id must be provided together",path:W?["assessment_result_sourced_id"]:["assessment_line_item_sourced_id"]})}),s1=g.object({student_email:g.email(),assignment_id:g.number().int().positive().optional(),timeback_id:X.optional(),subject:X.optional(),grade_rank:g.number().int().min(0).max(12).optional()}).superRefine(($,J)=>{let Z=$.assignment_id!==void 0,H=!!$.timeback_id,K=!!$.subject&&$.grade_rank!==void 0;if(!Z&&!H&&!K)J.addIssue({code:g.ZodIssueCode.custom,message:"Either assignment_id, timeback_id, or (subject and grade_rank) is required",path:["assignment_id"]});if($.grade_rank!==void 0&&!$.subject)J.addIssue({code:g.ZodIssueCode.custom,message:"grade_rank requires subject",path:["grade_rank"]})}),f=Q.enum(["active","tobedeleted"]),v=Q.record(Q.string(),Q.unknown()).nullable().optional(),L=Q.object({sourcedId:X,type:Q.string().optional(),href:Q.string().optional()}).strict(),TZ=Q.enum(["academicSession","assessmentLineItem","category","class","course","demographics","enrollment","gradingPeriod","lineItem","org","resource","result","scoreScale","student","teacher","term","user","componentResource","courseComponent"]),RZ=Q.object({sourcedId:X,type:TZ,href:Q.url()}).strict(),K2=Q.union([C2,y]).transform(($)=>$.includes("T")?$:`${$}T00:00:00Z`),NZ=Q.object({learningObjectiveId:Q.string(),score:Q.number().optional(),textScore:Q.string().optional()}),d9=Q.array(Q.object({source:Q.string(),learningObjectiveResults:Q.array(NZ)})),IZ=Q.object({roleType:Q.enum(["primary","secondary"]),role:P9,org:L,userProfile:Q.string().optional(),metadata:v,beginDate:K2.optional(),endDate:K2.optional()}).strict(),o1=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:f.optional(),enabledUser:Q.union([Q.boolean(),Q.enum(["true","false"]).transform(($)=>$==="true")]),givenName:X.describe("givenName must be a non-empty string"),familyName:X.describe("familyName must be a non-empty string"),middleName:X.nullable().optional(),preferredFirstName:X.nullable().optional(),preferredMiddleName:X.nullable().optional(),preferredLastName:X.nullable().optional(),username:X.nullable().optional(),email:Q.email(),userMasterIdentifier:Q.string().nullable().optional(),roles:Q.array(IZ).min(1,"roles must include at least one role"),userIds:Q.array(Q.object({type:X,identifier:X}).strict()).optional(),agents:Q.array(L).optional(),primaryOrg:L.optional(),grades:Q.array(n).optional(),sms:X.nullable().optional(),phone:X.nullable().optional(),pronouns:X.nullable().optional(),password:X.nullable().optional(),metadata:v}).strict(),n1=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:f,title:X.describe("title must be a non-empty string"),org:L,courseCode:X.nullable().optional(),subjects:Q.array(X2).nullable().optional(),subjectCodes:Q.array(Q.string()).nullable().optional(),grades:Q.array(n).nullable().optional(),level:X.nullable().optional(),academicSession:L.nullable().optional(),schoolYear:RZ.nullable().optional(),gradingScheme:X.nullable().optional(),metadata:v}).strict(),r1=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:f.optional(),title:X.describe("title must be a non-empty string"),terms:Q.array(L).min(1,"terms must have at least one item"),course:L,org:L,classCode:X.nullable().optional(),classType:Q.enum(["homeroom","scheduled"]).optional(),location:X.nullable().optional(),grades:Q.array(n).optional(),subjects:Q.array(X2).optional(),subjectCodes:Q.array(X).optional(),periods:Q.array(X).optional(),metadata:v}).strict(),J2=Q.enum(["true","false"]),t1=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:f.optional(),user:L,class:L,school:L.optional(),role:Q.enum(["administrator","proctor","student","teacher"]),primary:J2.optional(),beginDate:C2.optional(),endDate:C2.optional(),metadata:v}).strict(),e1=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),status:f,weight:Q.number().nullable().optional(),metadata:v}).strict(),$6=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),class:L,school:L,category:L,assignDate:K2,dueDate:K2,status:f.optional(),description:Q.string().nullable().optional(),resultValueMin:Q.number().nullable().optional(),resultValueMax:Q.number().nullable().optional(),gradingPeriod:L.nullable().optional(),academicSession:L.nullable().optional(),scoreScale:L.nullable().optional(),learningObjectiveSet:T9.nullable().optional(),metadata:v}).strict(),J6=Q.object({sourcedId:X.optional(),lineItem:L,student:L,class:L.nullable().optional(),scoreDate:K2,scoreStatus:Q.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),score:Q.number().nullable().optional(),textScore:Q.string().nullable().optional(),status:f,scoreScale:L.nullable().optional(),comment:Q.string().nullable().optional(),learningObjectiveSet:d9.nullable().optional(),inProgress:Q.string().optional(),incomplete:Q.string().optional(),late:Q.string().optional(),missing:Q.string().optional(),metadata:v}).strict(),Z6=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),status:f,type:X,class:L,course:L.nullable().optional(),scoreScaleValue:Q.array(Q.object({itemValueLHS:X,itemValueRHS:X,value:Q.string().optional(),description:Q.string().optional()}).strict()),metadata:v}).strict(),H6=Q.object({sourcedId:X.optional(),status:f,dateLastModified:y.optional(),title:X.describe("title must be a non-empty string"),description:Q.string().nullable().optional(),class:L.nullable().optional(),parentAssessmentLineItem:L.nullable().optional(),scoreScale:L.nullable().optional(),resultValueMin:Q.number().nullable().optional(),resultValueMax:Q.number().nullable().optional(),component:L.nullable().optional(),componentResource:L.nullable().optional(),learningObjectiveSet:Q.array(Q.object({source:Q.string(),learningObjectiveIds:Q.array(Q.string())})).optional().nullable(),course:L.nullable().optional(),metadata:v}).strict(),K6=Q.object({sourcedId:X.optional(),status:f,dateLastModified:y.optional(),metadata:v,assessmentLineItem:L,student:L,score:Q.number().nullable().optional(),textScore:Q.string().nullable().optional(),scoreDate:K2,scoreScale:L.nullable().optional(),scorePercentile:Q.number().nullable().optional(),scoreStatus:Q.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),comment:Q.string().nullable().optional(),learningObjectiveSet:d9.nullable().optional(),inProgress:Q.string().nullable().optional(),incomplete:Q.string().nullable().optional(),late:Q.string().nullable().optional(),missing:Q.string().nullable().optional()}).strict(),X6=Q.object({sourcedId:X.optional(),status:f.optional(),name:X.describe("name must be a non-empty string"),type:E5,identifier:Q.string().nullish(),parent:L.optional(),metadata:v}).strict(),W6=Q.object({sourcedId:X.optional(),status:f.optional(),name:X.describe("name must be a non-empty string"),type:Q.literal("school").default("school"),identifier:Q.string().nullish(),parent:L.optional(),metadata:v}).strict(),Y6=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:f,title:X.describe("title must be a non-empty string"),startDate:K2,endDate:K2,type:Q.enum(["gradingPeriod","semester","schoolYear","term"]),schoolYear:X.describe("schoolYear must be a non-empty string"),org:L,parent:L.optional(),children:Q.array(L).optional(),metadata:v}).strict(),Q6=Q.object({sourcedId:X,title:X.describe("title must be a non-empty string"),courseComponent:L,resource:L,status:f,sortOrder:Q.number().optional(),lessonType:l5.optional(),metadata:v}).strict(),G6=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),course:L,status:f,sortOrder:Q.number().optional(),parent:L.nullable().optional(),courseComponent:L.nullable().optional(),prerequisites:Q.array(Q.string()).nullable().optional(),prerequisiteCriteria:Q.string().nullable().optional(),unlockDate:K2.nullable().optional(),metadata:v}).strict().transform(({courseComponent:$,parent:J,...Z})=>({...Z,parent:J===void 0?$??void 0:J})),B6=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string"),role:Q.enum(["student","teacher"]),primary:J2.optional(),beginDate:C2.optional(),endDate:C2.optional(),metadata:v}).strict(),_6=Q.object({agentSourcedId:X.describe("agentSourcedId must be a non-empty string")}).strict(),F6=Q.object({applicationName:X.describe("applicationName must be a non-empty string"),credentials:Q.object({username:X.describe("username must be a non-empty string"),password:X.describe("password must be a non-empty string")})}).strict(),O6=Q.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:f.optional(),metadata:v,birthDate:Q.string().regex(/^\d{4}-\d{2}-\d{2}$/).nullable().optional(),sex:Q.enum(["male","female"]).nullable().optional(),americanIndianOrAlaskaNative:J2.nullable().optional(),asian:J2.nullable().optional(),blackOrAfricanAmerican:J2.nullable().optional(),nativeHawaiianOrOtherPacificIslander:J2.nullable().optional(),white:J2.nullable().optional(),demographicRaceTwoOrMoreRaces:J2.nullable().optional(),hispanicOrLatinoEthnicity:J2.nullable().optional(),countryOfBirthCode:Q.string().nullable().optional(),stateOfBirthAbbreviation:Q.string().nullable().optional(),cityOfBirth:Q.string().nullable().optional(),publicSchoolResidenceStatus:Q.string().nullable().optional()}).strict(),G2=Q.object({type:y5,subject:X2.nullish(),grades:Q.array(n).nullish(),language:Q.string().nullish(),xp:Q.number().nullish(),url:Q.url().nullish(),keywords:Q.array(Q.string()).nullish(),learningObjectiveSet:T9.nullish(),lessonType:Q.string().nullish()}).passthrough(),bZ=G2.extend({type:Q.literal("qti"),subType:f5,questionType:d5.optional(),difficulty:m5.optional()}),hZ=G2.extend({type:Q.literal("text"),format:Q.string(),author:Q.string().optional(),pageCount:Q.number().optional()}),SZ=G2.extend({type:Q.literal("audio"),duration:Q.string().regex(/^\d{2}:\d{2}:\d{2}(\.\d{2})?$/).optional(),format:Q.string(),speaker:Q.string().optional()}),EZ=G2.extend({type:Q.literal("video"),duration:Q.string().regex(/^\d{2}:\d{2}:\d{2}(\.\d{2})?$/).optional(),captionsAvailable:Q.boolean().optional(),format:Q.string()}),vZ=G2.extend({type:Q.literal("interactive"),launchUrl:Q.url().optional(),toolProvider:Q.string().optional(),instructionalMethod:Q.string().optional(),courseIdOnFail:Q.string().nullable().optional(),fail_fast:R9}),yZ=G2.extend({type:Q.literal("visual"),format:Q.string(),resolution:Q.string().optional()}),fZ=G2.extend({type:Q.literal("course-material"),subType:g5,author:Q.string().optional(),format:Q.string(),instructionalMethod:Q.string().optional()}),gZ=G2.extend({type:Q.literal("assessment-bank"),resources:Q.array(Q.string())}),dZ=Q.discriminatedUnion("type",[bZ,hZ,SZ,EZ,vZ,yZ,fZ,gZ]),mZ=new Set(["type","subject","grades","language","xp","url","keywords","learningObjectiveSet","lessonType","subType","questionType","difficulty","format","author","pageCount","duration","speaker","captionsAvailable","launchUrl","toolProvider","instructionalMethod","courseIdOnFail","resolution","resources"]),lZ=Q.record(Q.string(),Q.unknown()).superRefine(($,J)=>{if("fail_fast"in $){let H=R9.safeParse($.fail_fast);if(!H.success)for(let K of H.error.issues)J.addIssue({...K,path:["fail_fast",...K.path]})}let Z=Object.keys($).filter((H)=>mZ.has(H));if(Z.length===0)return;J.addIssue({code:"custom",message:`metadata keys ${Z.map((H)=>`\`${H}\``).join(", ")} require a valid \`type\` discriminator`})}),D6=Q.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),vendorResourceId:X.describe("vendorResourceId must be a non-empty string"),roles:Q.array(Q.enum(["primary","secondary"])).optional(),importance:Q.enum(["primary","secondary"]).optional(),vendorId:X.nullable().optional(),applicationId:X.nullable().optional(),status:f.optional(),metadata:Q.union([dZ,lZ]).nullable().optional()}).strict(),uZ=Q.object({url:X.describe("courseStructure.url must be a non-empty string"),skillCode:X.describe("courseStructure.skillCode must be a non-empty string"),lessonCode:X.describe("courseStructure.lessonCode must be a non-empty string"),title:X.describe("courseStructure.title must be a non-empty string"),"unit-title":X.describe("courseStructure.unit-title must be a non-empty string"),status:X.describe("courseStructure.status must be a non-empty string"),xp:Q.number()}).loose(),w6=Q.object({course:Q.object({sourcedId:X.describe("course.sourcedId must be a non-empty string").optional(),title:X.describe("course.title must be a non-empty string"),org:L,status:f,metadata:v}).strict(),courseStructure:Q.record(Q.string(),uZ)}).strict(),cZ=Q.object({student:L}).loose(),x6=Q.array(cZ).min(1,"results must have at least one item"),pZ=B.enum(["edulastic","mastery-track"]),aZ=B.enum(["powerpath-100","quiz","test-out","placement","unit-test","alpha-read-article"]),m9=B.array(n),l9=B.record(B.string(),B.unknown()).optional(),u9=B.object({courseId:X,lessonTitle:X.optional(),launchUrl:X.optional(),toolProvider:pZ,unitTitle:X.optional(),courseComponentSourcedId:X.optional(),vendorId:X.optional(),description:X.optional(),resourceMetadata:l9.nullable().optional(),grades:m9}),U6=u9.extend({lessonType:B.literal("test-out"),xp:B.number()}),M6=u9.extend({lessonType:B.literal("placement"),courseIdOnFail:X.nullable().optional(),xp:B.number().optional()}),k9=B.object({courseId:X,lessonType:aZ,lessonTitle:X.optional(),unitTitle:X.optional(),courseComponentSourcedId:X.optional(),resourceMetadata:l9.nullable().optional(),xp:B.number().optional(),grades:m9.optional(),courseIdOnFail:X.nullable().optional()}),j6=B.discriminatedUnion("testType",[k9.extend({testType:B.literal("qti"),qti:B.object({url:B.url(),title:X.optional(),metadata:B.record(B.string(),B.unknown()).optional()})}),k9.extend({testType:B.literal("assessment-bank"),assessmentBank:B.object({resources:B.array(B.object({url:B.url(),title:X.optional(),metadata:B.record(B.string(),B.unknown()).optional()})).min(1)})})]),L6=B.object({student:X,lesson:X}),V6=B.object({student:X,lesson:X}),k6=B.object({courseId:X,userId:X,classId:X.optional()}),q2=B.object({type:B.enum(["component","resource"]),id:X}),iZ=B.union([B.object({type:B.literal("set-skipped"),payload:B.object({target:q2,value:B.boolean()})}),B.object({type:B.literal("add-custom-resource"),payload:B.object({resource_id:X,parent_component_id:X,skipped:B.boolean().optional()})}),B.object({type:B.literal("move-item-before"),payload:B.object({target:q2,reference_id:X})}),B.object({type:B.literal("move-item-after"),payload:B.object({target:q2,reference_id:X})}),B.object({type:B.literal("move-item-to-start"),payload:B.object({target:q2})}),B.object({type:B.literal("move-item-to-end"),payload:B.object({target:q2})}),B.object({type:B.literal("change-item-parent"),payload:B.object({target:q2,new_parent_id:X,position:B.enum(["start","end"]).optional()})})]),q6=B.object({operation:iZ,reason:X.optional()}),C6=B.object({studentId:X,componentResourceId:X,result:B.object({status:B.enum(["active","tobedeleted"]),metadata:B.record(B.string(),B.unknown()).optional(),score:B.number().optional(),textScore:X.optional(),scoreDate:B.string().datetime(),scorePercentile:B.number().optional(),scoreStatus:S5,comment:X.optional(),learningObjectiveSet:B.array(B.object({source:X,learningObjectiveResults:B.array(B.object({learningObjectiveId:X,score:B.number().optional(),textScore:X.optional()}))})).optional(),inProgress:X.optional(),incomplete:X.optional(),late:X.optional(),missing:X.optional()})}),A6=B.object({student:X,lesson:X,applicationName:X.optional(),testId:X.optional(),skipCourseEnrollment:B.boolean().optional()}),z6=B.object({student:X,subject:X2}),P6=B.object({student:X,lesson:X}),T6=B.object({userId:X}),R6=B.object({userId:X,subject:B.enum(["Math","Reading","Language","Science"])}),N6=B.object({student:X,subject:X2,grade:n,testName:X.optional()}),I6=B.object({testName:X}),sZ=B.object({student:X,subject:X2,grade:n,testName:X.optional()}),b6=B.object({items:B.array(sZ)}),h6=B.object({spreadsheetUrl:B.url(),sheet:X}),S6=B.object({student:X,status:B.enum(["assigned","in_progress","completed","failed","expired","cancelled"]).optional(),subject:X.optional(),grade:n.optional(),limit:B.number().int().positive().max(3000).optional(),offset:B.number().int().nonnegative().optional()}),E6=B.object({student:X.optional(),status:B.enum(["assigned","in_progress","completed","failed","expired","cancelled"]).optional(),subject:X.optional(),grade:n.optional(),limit:B.number().int().positive().max(3000).optional(),offset:B.number().int().nonnegative().optional()}),v6=B.object({student:X,question:X,response:B.union([X,B.array(X)]).optional(),responses:B.record(B.string(),B.union([X,B.array(X)])).optional(),lesson:X,rendererOutcomes:B.object({score:B.number(),maxScore:B.number().min(0),isCorrect:B.boolean()}).optional(),playerState:B.string().optional()}).refine(($)=>$.response!==void 0||$.responses!==void 0,{message:"Either 'response' or 'responses' must be provided",path:["response","responses"]}).transform(($)=>{if($.response!==void 0&&$.responses===void 0)return{...$,responses:{RESPONSE:$.response}};return $}),y6=B.object({student:X,lesson:X,attempt:B.number().int().positive().optional()}),f6=B.object({student:X,lesson:X}),g6=B.object({student:X,lesson:X}),d6=B.object({student:X,course:X}),m6=B.object({student:X,lesson:X,applicationName:X.optional()}),l6=B.object({student:X,subject:X2}),u6=B.object({oneRosterSourcedId:X,subject:X}),c6=B.object({courseIds:B.array(X).min(1),rendererId:X,rendererUrl:B.url(),rendererVersion:X.optional(),suppressFeedback:B.boolean().optional(),suppressCorrectResponse:B.boolean().optional()}),p6=B.object({status:B.enum(["active","tobedeleted"]).optional()}),c9=G.enum(["choice","text-entry","extended-text","inline-choice","match","order","associate","select-point","graphic-order","graphic-associate","graphic-gap-match","hotspot","hottext","slider","drawing","media","upload"]),I8=G.enum(["single","multiple","ordered","record"]),b8=G.enum(["identifier","boolean","integer","float","string","point","pair","directedPair","duration","file","uri"]),oZ=G.enum(["easy","medium","hard"]),nZ=G.enum(["linear","nonlinear"]),rZ=G.enum(["individual","simultaneous"]),h8=G.enum(["show","hide"]),p9=G.enum(["test","item","stimulus"]),a6=G.enum(["QUESTION","LESSON"]),tZ=G.object({value:G.array(G.string())}).strict(),a9=G.object({identifier:X,cardinality:I8,baseType:b8.optional(),correctResponse:tZ}).strict(),i9=G.object({identifier:X,cardinality:I8,baseType:b8.optional()}).strict(),s9=G.object({identifier:X,cardinality:I8.optional(),baseType:b8,normalMaximum:G.number().optional(),normalMinimum:G.number().optional(),defaultValue:G.object({value:G.unknown().optional()}).strict().optional()}).strict(),eZ=G.object({outcomeIdentifier:X,variableIdentifier:X}).strict(),o9=G.object({templateType:G.enum(["match_correct","map_response"]),responseDeclarationIdentifier:X,outcomeIdentifier:X,correctResponseIdentifier:X,incorrectResponseIdentifier:X,inlineFeedback:eZ.optional()}).strict(),$0=G.object({source:X,learningObjectiveIds:G.array(G.string())}).strict(),S8=G.object({subject:G.string().optional(),grade:n.optional(),difficulty:oZ.optional(),learningObjectiveSet:G.array($0).optional()}).loose(),n9=G.object({outcomeIdentifier:X,identifier:X,showHide:h8,content:G.string(),title:G.string()}).strict(),r9=G.object({outcomeIdentifier:X,identifier:X,showHide:h8,content:G.string(),class:G.array(G.string())}).strict(),t9=G.object({outcomeIdentifier:X,identifier:X,showHide:h8,content:G.string(),class:G.array(G.string())}).strict(),e9=G.object({href:X,type:X}).strict(),$$=G.object({id:X,support:G.string(),content:G.string()}).strict(),i6=G.object({page:G.number().int().positive().optional(),limit:G.number().int().positive().optional(),sort:G.string().optional(),order:G.enum(["asc","desc"]).optional()}).strict(),J0=G.object({format:G.string().pipe(G.literal("xml")),xml:X,metadata:S8.optional()}).strict(),Z0=G.object({identifier:X,title:X,type:c9,qtiVersion:G.string().optional(),timeDependent:G.boolean().optional(),adaptive:G.boolean().optional(),responseDeclarations:G.array(a9).optional(),outcomeDeclarations:G.array(i9).optional(),responseProcessing:o9.optional(),metadata:S8.optional(),modalFeedback:G.array(n9).optional(),feedbackInline:G.array(r9).optional(),feedbackBlock:G.array(t9).optional()}).strict(),s6=G.union([J0,Z0]),o6=G.object({identifier:X.optional(),title:X,type:c9,qtiVersion:G.string().optional(),timeDependent:G.boolean().optional(),adaptive:G.boolean().optional(),responseDeclarations:G.array(a9).optional(),outcomeDeclarations:G.array(i9).optional(),responseProcessing:o9.optional(),metadata:S8.optional(),modalFeedback:G.array(n9).optional(),feedbackInline:G.array(r9).optional(),feedbackBlock:G.array(t9).optional(),rawXml:G.string(),content:G.record(G.string(),G.unknown())}).strict(),n6=G.object({identifier:X,response:G.union([G.string(),G.array(G.string())])}).strict(),J$=G.object({identifier:X,href:X,sequence:G.number().int().positive().optional()}).strict(),H0=G.object({identifier:X,title:X,visible:G.boolean(),required:G.boolean().optional(),fixed:G.boolean().optional(),sequence:G.number().int().positive(),"qti-assessment-item-ref":G.array(J$).optional()}).strict(),Z$=G.object({identifier:X,navigationMode:G.string().pipe(nZ),submissionMode:G.string().pipe(rZ),"qti-assessment-section":G.array(H0)}).strict(),r6=G.object({items:G.array(J$).min(1)}).strict(),t6=G.object({metadata:G.record(G.string(),G.unknown()).optional()}).strict(),e6=G.object({identifier:X,title:X,qtiVersion:G.string().optional(),toolName:G.string().optional(),toolVersion:G.string().optional(),timeLimit:G.number().optional(),maxAttempts:G.number().optional(),toolsEnabled:G.record(G.string(),G.boolean()).optional(),metadata:G.record(G.string(),G.unknown()).optional(),"qti-test-part":G.array(Z$),"qti-outcome-declaration":G.array(s9).optional()}).strict(),$7=G.object({identifier:X.optional(),title:X,qtiVersion:G.string().optional(),toolName:G.string().optional(),toolVersion:G.string().optional(),timeLimit:G.number().optional(),maxAttempts:G.number().optional(),toolsEnabled:G.record(G.string(),G.boolean()).optional(),metadata:G.record(G.string(),G.unknown()).optional(),"qti-test-part":G.array(Z$),"qti-outcome-declaration":G.array(s9).optional()}).strict(),J7=G.object({identifier:X,title:X,label:G.string().optional(),language:G.string().optional(),stylesheet:e9.optional(),content:G.string(),catalogInfo:G.array($$).optional(),toolName:G.string().optional(),toolVersion:G.string().optional(),metadata:G.record(G.string(),G.unknown()).optional()}).strict(),Z7=G.object({identifier:X.optional(),title:X,label:G.string().optional(),language:G.string().optional(),stylesheet:e9.optional(),content:G.string(),catalogInfo:G.array($$).optional(),toolName:G.string().optional(),toolVersion:G.string().optional(),metadata:G.record(G.string(),G.unknown()).optional()}).strict(),H7=G.object({xml:G.string().optional(),schema:p9,entityId:G.string().optional()}).strict(),K7=G.object({xml:G.array(G.string()),schema:p9,entityIds:G.array(G.string())}).strict(),X7=G.object({questionId:G.string().optional(),userId:X,feedback:X,lessonId:X,humanApproved:G.boolean().optional()}).strict(),K0=A2.enum(["creator_only","same_org","all_orgs"]),X0=A2.string().uuid(),W7=A2.email(),Y7=A2.object({sql:X}),Q7=A2.object({endpointName:X,sql:X,authMode:K0,description:X.optional()}),G7=A2.object({endpointId:X0});class H${transport;eventTransformer;constructor($,J){this.transport=$,this.eventTransformer=J?.eventTransformer}send($,J){o(t5,{sensor:$,events:J},"send events");let Z={sensor:$,sendTime:new Date().toISOString(),dataVersion:"http://purl.imsglobal.org/ctx/caliper/v1p2",data:J};return this.sendEnvelope(Z)}async sendEnvelope($){o(U9,$,"caliper envelope"),S.debug("Sending Caliper envelope",{sensor:$.sensor,eventCount:$.data.length});let J=this.eventTransformer?this.eventTransformer.transformEnvelope($):$,H=(await this.transport.request(this.transport.paths.send,{method:"POST",body:J}))?.jobId;return S.debug("Events sent",{jobId:H??"(none)"}),{jobId:H}}async validate($){let J=this.transport.paths.validate;if(!J)throw Error("validate() is not supported on this platform");o(U9,$,"caliper envelope"),S.debug("Validating Caliper envelope",{sensor:$.sensor,eventCount:$.data.length});let Z=await this.transport.request(J,{method:"POST",body:$});return S.debug("Validation complete",{status:Z.status}),Z}async list($={}){let J=this.transport.paths.list;if(!J)throw Error("list() is not supported on this platform");o(M9,$,"list events params"),S.debug("Listing events",{params:$});let Z={limit:$.limit??100,offset:$.offset??0};if($.sensor)Z.sensor=$.sensor;if($.startDate)Z.startDate=$.startDate;if($.endDate)Z.endDate=$.endDate;if($.actorId)Z.actorId=$.actorId;if($.actorEmail)Z.actorEmail=$.actorEmail;let H=await this.transport.request(J,{params:Z});return{events:H.events??[],pagination:H.pagination}}stream($={}){let J=this.transport.paths.list;if(!J)throw Error("stream() is not supported on this platform");let{max:Z,...H}=$;o(M9,H,"list events params"),z8($),S.debug("Streaming events",{params:$});let K={};if(H.limit!==void 0)K.limit=H.limit;if(H.offset!==void 0)K.offset=H.offset;if(H.sensor)K.sensor=H.sensor;if(H.startDate)K.startDate=H.startDate;if(H.endDate)K.endDate=H.endDate;if(H.actorId)K.actorId=H.actorId;if(H.actorEmail)K.actorEmail=H.actorEmail;return new C9(this.transport,J,{...K,max:Z})}async get($){let J=this.transport.paths.get;if(!J)throw Error("get() is not supported on this platform");u2($,"externalId"),S.debug("Getting event",{externalId:$});let Z=J.replace("{id}",encodeURIComponent($)),H=await this.transport.request(Z);if(!H.event)throw Error(`Event not found: ${$}`);return H.event}sendActivity($,J){o(s5,J,"activity event");let Z=P8(J);return S.debug("Sending ActivityCompletedEvent",{eventId:Z.id,actor:J.actor.email,subject:J.object.subject,metricsCount:J.metrics.length}),this.send($,[Z])}sendTimeSpent($,J){o(o5,J,"time spent event");let Z=T8(J);return S.debug("Sending TimeSpentEvent",{eventId:Z.id,actor:J.actor.email,subject:J.object.subject,metricsCount:J.metrics.length}),this.send($,[Z])}sendQuestionSeen($,J){o(JZ,J,"question seen event");let Z=j5(J);return S.debug("Sending AssessmentItemEvent.Started",{eventId:Z.id,actor:J.actor,object:J.object.id}),this.send($,[Z])}sendQuestionAnswered($,J){o(ZZ,J,"question answered event");let Z=L5(J);return S.debug("Sending AssessmentItemEvent.Completed",{eventId:Z.id,actor:J.actor,object:J.object.id}),this.send($,[Z])}sendQuestionGraded($,J){o(HZ,J,"question graded event");let Z=V5(J);return S.debug("Sending GradeEvent.Graded",{eventId:Z.id,actor:J.actor,object:J.object,scoreGiven:J.generated.scoreGiven}),this.send($,[Z])}}class K${transport;constructor($){this.transport=$}async getStatus($){let J=this.transport.paths.jobStatus;if(!J)throw Error("getStatus() is not supported on this platform");u2($,"jobId"),S.debug("Getting job status",{jobId:$});let Z=J.replace("{id}",encodeURIComponent($));return(await this.transport.request(Z)).job}async waitForCompletion($,J={}){u2($,"jobId"),W0(J);let{timeoutMs:Z=30000,pollIntervalMs:H=1000}=J,K=Date.now();S.debug("Waiting for job completion",{jobId:$,timeoutMs:Z,pollIntervalMs:H});while(Date.now()-K<Z){let W=await this.getStatus($);if(W.state==="completed")return S.info("Job completed",{jobId:$}),W;if(W.state==="failed")throw S.error("Job failed",{jobId:$}),Error(`Job ${$} failed`);await new Promise((_)=>{setTimeout(_,H)})}throw Error(`Job ${$} timed out after ${Z}ms`)}}function W0($){if(!$)return;let J=[];if($.timeoutMs!==void 0){if(!Number.isInteger($.timeoutMs)||$.timeoutMs<=0)J.push({path:"timeoutMs",message:"Must be a positive integer"})}if($.pollIntervalMs!==void 0){if(!Number.isInteger($.pollIntervalMs)||$.pollIntervalMs<=0)J.push({path:"pollIntervalMs",message:"Must be a positive integer"})}if(J.length>0)throw N2("Invalid polling options",J)}function Y0($=l2){return class{transport;_provider;events;jobs;constructor(Z={}){let H=N5(Z,$),K;if(H.mode==="transport")this.transport=H.transport,S.info("Client initialized with custom transport");else{let{provider:W}=H,{baseUrl:_,paths:F}=W.getEndpointWithPaths("caliper"),O=W.getTokenProvider("caliper");if(!O)throw Error("Caliper API requires authentication");if(this._provider=W,this.transport=new q9({baseUrl:_,tokenProvider:O,timeout:W.timeout,paths:F}),W.platform===X9)K=new A9;S.info("Client initialized",{platform:W.platform,env:W.env,baseUrl:_})}this.events=new H$(this.transport,{eventTransformer:K}),this.jobs=new K$(this.transport)}getTransport(){return this.transport}checkAuth(){if(!this._provider)throw Error("Cannot check auth: client initialized with custom transport");return this._provider.checkAuth()}}}var B7=Y0();class W2 extends Error{course;env;constructor($,J){let Z=$.grade===void 0?$.courseCode??$.subject:`${$.subject} grade ${$.grade}`;super(`Course "${Z}" is missing a synced ID for ${J}. Run \`timeback resources push\` first.`);this.name="MissingSyncedCourseIdError",this.course=$,this.env=J}}class a2 extends Error{sensor;constructor($){super(`Invalid sensor URL "${$}". Sensor must be a valid absolute URL (e.g., "https://sensor.example.com") to support slug-based activity IDs.`);this.name="InvalidSensorUrlError",this.sensor=$}}var X$="/ims/oneroster/rostering/v1p2";function W$($,J){return`${$.replace(/\/+$/,"")}/${J.replace(/^\/+/,"")}`}function Q0($,J){return W$($,`${X$}/courses/${J}`)}function G0($,J){return W$($,`${X$}/users/${J}`)}function Y$($,J){let Z=$.ids?.[J];if(!Z)throw new W2($,J);return Z}function B0($,J,Z){let H=Y$(J,Z);return Q0($,H)}function _0($){if($.courseCode)return $.courseCode;if($.grade!==void 0)return`${$.subject} G${String($.grade)}`;return $.subject}function E8($,J,Z){let H;try{H=new URL($)}catch{throw new a2($)}let K="grade"in J?`${J.subject}/g${String(J.grade)}`:J.code,W=H.pathname.replace(/\/+$/,"");return H.pathname=`${W}/activities/${K}/${encodeURIComponent(Z)}`,H.toString()}function Q$($,J,Z,H,K,W,_){return{id:E8(K,$.course,$.id),type:"TimebackActivityContext",subject:J.subject,app:{name:Z},activity:{name:$.name},course:{id:B0(W,J,H),name:_0(J)},process:_??!0}}function F0($){let J=[];if($.totalQuestions!==void 0)J.push({type:"totalQuestions",value:$.totalQuestions});if($.correctQuestions!==void 0)J.push({type:"correctQuestions",value:$.correctQuestions});if($.xpEarned!==void 0)J.push({type:"xpEarned",value:$.xpEarned});if($.masteredUnits!==void 0)J.push({type:"masteredUnits",value:$.masteredUnits});return J}function O0($,J){let Z=[{type:"active",value:Math.max(0,$)/1000}];if(J>0)Z.push({type:"inactive",value:Math.max(0,J)/1000});return Z}function D0($){return $===void 0?void 0:{pctCompleteApp:$}}function G$($,J){return{courseId:Y$($,J)}}function B$($){if(!$)return;return`urn:uuid:${$}`}function _$($,J,Z){return{id:G0($,J),type:"TimebackUser",email:Z}}function F$($){let{sensor:J,timebackId:Z,email:H,payload:K,process:W,course:_,appName:F,apiEnv:O,onerosterBaseUrl:x,attempt:V,runId:A}=$,C=_$(x,Z,H),q=Q$(K,_,F,O,J,x,W??K.process),z=F0(K.metrics),N=D0(K.pctComplete),I=B$(A),r=G$(_,O),Y2=P8({actor:C,object:q,metrics:z,eventTime:K.endedAt,generatedExtensions:N,extensions:r,attempt:V,session:I});return{sensor:J,actor:C,object:q,event:Y2,payload:K,course:_,appName:F,apiEnv:O,email:H,timebackId:Z}}function i2($){let{sensor:J,timebackId:Z,email:H,payload:K,course:W,appName:_,apiEnv:F,onerosterBaseUrl:O,runId:x}=$,V=_$(O,Z,H),A=Q$(K,W,_,F,J,O),C=O0(K.elapsedMs,K.pausedMs),q=G$(W,F),z=B$(x),N=T8({actor:V,object:A,metrics:C,eventTime:K.endedAt,extensions:q,session:z});return{sensor:J,actor:V,object:A,event:N,payload:{id:K.id,name:K.name,course:K.course,startedAt:K.startedAt,endedAt:K.endedAt,elapsedMs:K.elapsedMs,pausedMs:K.pausedMs},course:W,appName:_,apiEnv:F,email:H,timebackId:Z,runId:x}}import*as h from"zod";import*as b2 from"zod";var w0=b2.object({subject:O$,grade:D$}),x0=b2.object({code:E}),M2=b2.union([w0,x0]);class B2 extends Error{code;selector;count;constructor($,J,Z){super($);this.code=$,this.selector=J,this.count=Z}get selectorDescription(){if("grade"in this.selector)return`${this.selector.subject} grade ${this.selector.grade}`;return`code "${this.selector.code}"`}}function j2($,J){let Z;if("grade"in J)Z=$.filter((H)=>H.subject===J.subject&&H.grade===J.grade);else Z=$.filter((H)=>H.courseCode===J.code);if(Z.length===0)throw new B2("unknown_course",J);if(Z.length>1)throw new B2("ambiguous_course",J,Z.length);return Z[0]}var z2=R("handlers:activity:schema"),U0=h.object({totalQuestions:h.number().int().nonnegative().optional(),correctQuestions:h.number().int().nonnegative().optional(),xpEarned:h.number(),masteredUnits:h.number().int().nonnegative().optional()}).superRefine(o2);function M0($){return Math.min(100,Math.max(0,$))}function s2($){if("grade"in $)return`${$.subject} grade ${$.grade}`;return`code "${$.code}"`}var j0=h.object({id:E,name:E,course:M2,runId:h.string().uuid(),startedAt:h.iso.datetime(),endedAt:h.iso.datetime(),elapsedMs:h.number().int().nonnegative(),pausedMs:h.number().int().nonnegative()});function w$($,J,Z){let H=j0.safeParse($);if(!H.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:H.error.flatten()})};let K=H.data,W;try{W=j2(J.courses,K.course)}catch(F){if(F instanceof B2){let O=s2(K.course);if(F.code==="unknown_course")return z2.warn("Unknown course selector",{selector:K.course}),{ok:!1,response:M("UNKNOWN_COURSE",`Unknown course: ${O}`,400)};return z2.error("Ambiguous course selector",{selector:K.course}),{ok:!1,response:M("AMBIGUOUS_COURSE_SELECTOR","Ambiguous course selector in timeback.config.json",500)}}throw F}let _=h2(J,W,Z);if(!_){let F=s2(K.course);return z2.error("Missing sensor for course",{selector:K.course}),{ok:!1,response:M("MISSING_SENSOR",`Course "${F}" has no sensor configured.`,500)}}return{ok:!0,payload:K,course:W,sensor:_}}var L0=h.object({id:E,name:E,course:M2,process:h.boolean().optional(),runId:h.string().uuid(),endedAt:h.iso.datetime(),metrics:U0,pctComplete:h.number().optional().transform(($)=>$===void 0?void 0:M0($))});function x$($,J,Z){let H=L0.safeParse($);if(!H.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:H.error.flatten()})};let K=H.data,W;try{W=j2(J.courses,K.course)}catch(F){if(F instanceof B2){let O=s2(K.course);if(F.code==="unknown_course")return z2.warn("Unknown course selector",{selector:K.course}),{ok:!1,response:M("UNKNOWN_COURSE",`Unknown course: ${O}`,400)};return z2.error("Ambiguous course selector",{selector:K.course}),{ok:!1,response:M("AMBIGUOUS_COURSE_SELECTOR","Ambiguous course selector in timeback.config.json",500)}}throw F}let _=h2(J,W,Z);if(!_){let F=s2(K.course);return z2.error("Missing sensor for course",{selector:K.course}),{ok:!1,response:M("MISSING_SENSOR",`Course "${F}" has no sensor configured.`,500)}}return{ok:!0,payload:K,course:W,sensor:_}}import{getServiceUrlsForEnv as q$}from"@timeback/core";var v8=R("handlers:activity:attempts");async function U$($,J){let Z=[$,J].join("_");return`caliper_${await S2(Z)}`}function V0($,J){return $.find((Z)=>Z.scoreDate===J)}function k0($){let J=0;for(let Z of $){let K=Z.metadata?.attempt;if(typeof K==="number"&&K>J)J=K}return J}function q0($){let J=$.metadata;return typeof J?.attempt==="number"&&J.attempt>=1?J.attempt:1}async function M$($,J,Z,H){let K;try{K=await $.oneroster.assessmentResults.listAll({where:{status:"active","assessmentLineItem.sourcedId":J,"student.sourcedId":Z}})}catch{return v8.debug("No existing results found (line item may not exist yet)",{lineItemId:J}),1}if(K.length===0)return 1;let W=V0(K,H);if(W){let O=q0(W);return v8.debug("Retry detected, reusing attempt number",{lineItemId:J,timebackId:Z,endedAt:H,attempt:O}),O}let _=k0(K),F=_+1;return v8.debug("New attempt computed",{lineItemId:J,timebackId:Z,endedAt:H,maxAttempt:_,nextAttempt:F}),F}var j$=R("handlers:activity:completion");async function V$($){let{client:J,courseId:Z,timebackId:H,pctComplete:K,appName:W}=$;if(K!==100)return;let _=L$(Z),F=`timeback_sdk_${await S2(`mastery-completion_${Z}`)}`,O=`timeback_sdk_${await S2(`mastery-completion_${Z}_${H}`)}`;try{let x=!1;try{await J.oneroster.assessmentLineItems(F).get(),x=!0}catch{}if(!x)await J.oneroster.assessmentLineItems.create({sourcedId:F,title:`${W}: Complete`,status:"active",course:{sourcedId:_.course},componentResource:{sourcedId:_.componentResource},resultValueMin:0,resultValueMax:100,metadata:{appName:W}});let V=new Date().toISOString();await J.oneroster.assessmentResults.upsert(O,{status:"active",assessmentLineItem:{sourcedId:F},student:{sourcedId:H},score:100,scoreDate:V,scoreStatus:"fully graded",metadata:{isMasteryCompletion:!0,completedAt:V,appName:W}}),j$.debug("Created mastery completion entry",{courseId:Z,timebackId:H,lineItemId:F,resultId:O})}catch(x){let V=x instanceof Error?x.message:"Unknown error";j$.error("Failed to create mastery completion entry",{courseId:Z,timebackId:H,lineItemId:F,error:V})}}import{aggregateActivityMetrics as C0}from"@timeback/core/utils";var P2=R("handlers:activity:progress");function A0($,J){let Z=u(J),H=$.overrides?.[Z]?.metadata?.metrics,K=$.metadata?.metrics;return H?.totalLessons??K?.totalLessons}function z0($,J){let Z=$/J*100;return Math.min(100,Math.max(0,Math.round(Z)))}async function k$($){let{client:J,courseId:Z,timebackId:H,payload:K,course:W,env:_}=$;if(K.pctComplete!==void 0)return;let F=K.metrics.masteredUnits;if(typeof F!=="number"||F<=0)return;let O=A0(W,_);if(!O||O<=0){P2.debug("Skipping progress computation: totalLessons not configured",{courseId:Z});return}let x;try{x=(await J.edubridge.enrollments.list({userId:H})).find((N)=>N.course.id===Z)?.id}catch(z){let N=z instanceof Error?z.message:"Unknown error";P2.warn("Failed to fetch enrollments for progress computation",{courseId:Z,timebackId:H,error:N});return}if(!x){P2.warn("Skipping progress computation: enrollment not found for student/course",{courseId:Z,timebackId:H});return}let V=0;try{let z=await J.edubridge.analytics.getEnrollmentFacts({enrollmentId:x});V=C0(z).masteredUnits}catch(z){let N=z instanceof Error?z.message:"Unknown error";P2.warn("Failed to fetch enrollment facts for progress computation",{courseId:Z,timebackId:H,enrollmentId:x,error:N});return}let A=!0;try{if((await J.edubridge.analytics.getWeeklyFacts({studentId:H,weekDate:K.endedAt})).some((I)=>{if(I.activityId!==K.id)return!1;if(I.enrollmentId&&I.enrollmentId===x)return!0;if(I.courseId&&I.courseId===Z)return!0;return!1}))A=!1}catch(z){let N=z instanceof Error?z.message:"Unknown error";P2.warn("Failed to fetch weekly facts for retry-safe progress computation",{courseId:Z,timebackId:H,enrollmentId:x,activityId:K.id,error:N}),A=!1}let C=V+(A?F:0),q=z0(C,O);return P2.debug("Computed pctComplete",{courseId:Z,timebackId:H,enrollmentId:x,historicalMasteredUnits:V,currentMasteredUnits:F,shouldIncludeCurrentMasteredUnits:A,totalMastered:C,totalLessons:O,pctComplete:q}),{pctComplete:q}}var y8=R("handlers:activity:submit"),P0={computeProgress:k$,maybeWriteCompletionEntry:V$};function C$($,J){return $.timebackId??_2({email:$.email,client:J})}async function n2($,J,Z=P0){let H=u($.env),{userInfo:K,payload:W,course:_,sensor:F,preview:O=!1,runId:x}=J,V=_.ids?.[H];if(!V)throw new W2(_,H);let A=$.getClient(),C=await C$(K,A),q=W;if(W.pctComplete===void 0){let y2=await Z.computeProgress({client:A,courseId:V,timebackId:C,payload:W,course:_,env:$.env});if(y2)q={...W,pctComplete:y2.pctComplete}}let z;if(!O){let y2=E8(F,q.course,q.id),UJ=await U$(y2,V);z=await M$(A,UJ,C,q.endedAt)}let I=q$(H).oneroster,r=F$({sensor:F,timebackId:C,email:K.email,payload:q,process:q.process,course:_,appName:$.appConfig.name,apiEnv:H,onerosterBaseUrl:I,attempt:z,runId:x}),Y2=$.hooks?.beforeActivitySend,v2=Y2?await Y2(r):r,p=v2??r;if(O||v2===null)return{success:!0,preview:!0,sensor:p.sensor,actor:p.actor,object:p.object,event:p.event};return await A.caliper.events.send(p.sensor,[p.event]),await Z.maybeWriteCompletionEntry({client:A,courseId:V,timebackId:p.timebackId,pctComplete:p.payload.pctComplete,appName:p.appName}),y8.debug("Recorded completion",{courseSelector:W.course,activityId:W.id}),{success:!0}}async function A$($,J){let Z=u($.env),{userInfo:H,payload:K,course:W,sensor:_,runId:F}=J;if(!W.ids?.[Z])throw new W2(W,Z);let x=$.getClient(),V=await C$(H,x),C=q$(Z).oneroster,q=i2({sensor:_,timebackId:V,email:H.email,payload:K,course:W,appName:$.appConfig.name,apiEnv:Z,onerosterBaseUrl:C,runId:F}),z=$.hooks?.beforeTimeSpentSend,N=z?await z(q):q,I=N??q;if(N===null){y8.debug("Hook skipped time-spent send",{activityId:K.id});return}await x.caliper.events.send(I.sensor,[I.event]),y8.debug("Sent time-spent event",{activityId:K.id,elapsedMs:K.elapsedMs})}var E2=R("handlers:activity:submit");async function T2($,J){if($.mode==="custom"){let H=await $.getEmail(J);return H?{email:H}:void 0}let Z=await $.getUser(J);return Z?{email:Z.email,timebackId:Z.id}:void 0}function r2($){return async(J)=>{let Z=await T2($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let H=x$(await J.json(),$.appConfig,$.env);if(!H.ok)return H.response;let{payload:K,course:W,sensor:_}=H,F={id:K.id,name:K.name,course:K.course,process:K.process,endedAt:K.endedAt,metrics:K.metrics,pctComplete:K.pctComplete};try{if((await n2({env:$.env,api:$.api,appConfig:$.appConfig,getClient:$.getClient,hooks:$.hooks},{userInfo:Z,payload:F,course:W,sensor:_,preview:!1,runId:K.runId})).preview)return P({preview:!0});return E2.debug("Submitted activity",{runId:K.runId,activityId:K.id}),new Response(null,{status:204})}catch(O){if(O instanceof d)return E2.warn("Failed to resolve Timeback user",{code:O.code}),M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",F2(O));if(O instanceof W2)return E2.warn("Course not synced",{course:O.course,env:O.env}),M("MISSING_SYNCED_COURSE_ID",O.message,503);if(O instanceof a2)return E2.error("Invalid sensor URL",{sensor:O.sensor}),M("INVALID_SENSOR_URL",O.message,500);let x=O instanceof Error?O.message:"Unknown error";return E2.error("Failed to submit activity",{error:x}),M("INTERNAL_ERROR",x,502)}}}var R2=R("handlers:activity:heartbeat"),t2=new Map,R0=300000;function N0(){let $=Date.now();for(let[J,Z]of t2)if($-Z>R0)t2.delete(J)}function I0($){return N0(),t2.has($)}function z$($){t2.set($,Date.now())}function e2($){return async(J)=>{let Z=await T2($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let H=w$(await J.json(),$.appConfig,$.env);if(!H.ok)return H.response;let{payload:K,course:W,sensor:_}=H,F=new Date(K.startedAt).getTime(),O=`${K.runId}:${F}`;if(I0(O))return R2.debug("Duplicate heartbeat, skipping",{runId:K.runId,windowKey:O}),P({duplicate:!0});let x=u($.env);if(!W.ids?.[x]){let C=new W2(W,x);return R2.warn("Course not synced",{course:K.course,env:x}),M("MISSING_SYNCED_COURSE_ID",C.message,503)}let A=$.getClient();try{let C=Z.timebackId??await _2({email:Z.email,client:A}),z=T0(x).oneroster,N=i2({sensor:_,timebackId:C,email:Z.email,payload:K,course:W,appName:$.appConfig.name,apiEnv:x,onerosterBaseUrl:z,runId:K.runId}),I=$.hooks?.beforeTimeSpentSend,r=I?await I(N):N,Y2=r??N;if(r===null)return z$(O),R2.debug("Hook skipped heartbeat send",{runId:K.runId}),new Response(null,{status:204});return await A.caliper.events.send(Y2.sensor,[Y2.event]),z$(O),R2.debug("Sent heartbeat",{runId:K.runId,activityId:K.id,elapsedMs:K.elapsedMs}),new Response(null,{status:204})}catch(C){if(C instanceof d)return R2.warn("Failed to resolve Timeback user",{code:C.code}),M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",F2(C));let q=C instanceof Error?C.message:"Unknown error";return R2.error("Failed to send heartbeat",{error:q}),M("INTERNAL_ERROR",q,502)}}}async function v$($,J,Z){if(Z.mode!=="sso")return L2.warn("SSO not configured"),P({error:"SSO not configured"},400);return await h$({req:$,env:J,clientId:Z.clientId,issuer:Z.issuer,redirectUri:Z.redirectUri,buildState:Z.buildState})}async function b0($,J,Z,H,K,W,_){try{let F=W.issuer??P$(K),O=E$(J,W.redirectUri);L2.debug("Exchanging auth code for tokens",{issuer:F,clientId:W.clientId});let x=await T$({issuer:F,clientId:W.clientId,clientSecret:W.clientSecret,code:$,redirectUri:O}),V=await R$({issuer:F,accessToken:x.access_token}),A=typeof V.identities==="string"?JSON.parse(V.identities):V.identities;L2.debug("SSO completed, resolving Timeback user",{user:{...V,identities:A}});let C=await O2({env:K,apiCredentials:_.credentials,userInfo:V,client:_.getClient()});L2.debug("Timeback user resolved",{timebackId:C.id});let q={user:C,idp:{tokens:x,userInfo:V},state:Z,req:H,redirect:$8,json:P};return W.onCallbackSuccess(q)}catch(F){let O=F instanceof Error?F:Error("Unknown error"),x=F instanceof d?F.code:"token_exchange_failed";if(L2.error("SSO callback failed",{error:O.message,errorCode:x}),W.onCallbackError)return W.onCallbackError(N$(O,x,Z,H));if(F instanceof d)return P({error:O.message},F2(F));return P({error:O.message},500)}}async function y$($,J,Z,H){if(Z.mode!=="sso")return L2.warn("SSO not configured"),P({error:"SSO not configured"},400);let{url:K,code:W,errorParam:_,state:F}=S$($);if(_)return I$(_,K,F,$,Z.onCallbackError);if(!W)return b$(F,$,Z.onCallbackError);return await b0(W,K,F,$,J,Z,H)}function J8($){let{env:J,identity:Z}=$;return(H)=>v$(H,J,Z)}function Z8($){let{env:J,identity:Z,api:H}=$;return(K)=>y$(K,J,Z,H)}function H8(){return()=>$8("/")}var f$=($)=>{return async(J)=>{return await $.getClient().powerpath.assessments.getAssessmentProgress({student:J.student,lesson:J.lesson,attempt:J.attempt})}};var g$=($)=>{return async(J)=>{return(await $.getClient().powerpath.assessments.getAttempts({student:J.student,lesson:J.lesson})).attempts.map((H)=>({attempt:H.attempt??void 0,score:H.score,startedAt:H.startedAt??void 0,completedAt:H.completedAt??void 0,scoreStatus:H.scoreStatus}))}};var d$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments,H=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson}),K;if(H.lessonType!=="powerpath-100"&&H.finalized!==!0){try{K=await Z.finalStudentAssessmentResponse({student:J.student,lesson:J.lesson})}catch{}H=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson})}let W=H.lessonType==="powerpath-100"?!0:H.finalized===!0;return{...K??{},score:H.score??0,totalQuestions:H.totalQuestions,correctQuestions:H.correctQuestions,accuracy:H.accuracy,finalized:W}}};function m$($){let J=$.metadata?.questionCount;return typeof J==="number"?J:void 0}function K8($){return{id:$.id,title:$.title,difficulty:$.difficulty,content:$.content?{rawXml:$.content.rawXml}:void 0,index:$.index,url:$.url,humanApproved:$.humanApproved,response:$.response,responses:$.responses,correct:$.correct,result:$.result,resultId:$.resultId,learningObjectives:$.learningObjectives}}function l$($,J){if(!J)return{lessonId:$,lessonType:"quiz",score:0,seenQuestions:0,finalized:!1};return{lessonId:$,lessonType:J.lessonType,attempt:J.attempt,score:J.score??0,questionCount:J.totalQuestions,seenQuestions:J.lessonType==="powerpath-100"?J.seenQuestions.length:0,finalized:J.lessonType==="powerpath-100"?!1:J.finalized}}function u$($){let J=$.questions,Z=J.filter((H)=>H.response!==void 0).map((H)=>H.id);return{questions:J.map(K8),answeredIds:Z,score:$.score??0,finalized:$.finalized,complete:$.finalized}}function h0($){if($.lessonType==="powerpath-100")return $.responseResult.isCorrect;return!1}function S0($){if($.lessonType==="powerpath-100")return $.powerpathScore;return 0}function c$($){let J=S0($);return{correct:h0($),score:J,complete:$.lessonType==="powerpath-100"&&J>=100,questionResult:$.questionResult}}function p$($,J){return Array.from(new Set($.map((Z)=>Z.ids?.[J]).filter((Z)=>Boolean(Z))))}function a$($){return String($.sourcedId??"")}var i$=($)=>{return async(J={})=>{let Z=$.getClient(),H=u($.env),K;if(J.course){let F=j2($.appConfig.courses,J.course).ids?.[H];K=F?[F]:[]}else K=p$($.appConfig.courses,H);let W=[];for(let _ of K){let F=await Z.oneroster.courses(_).components({where:{status:"active"}}),O=f8(F);for(let x of O){let V=a$(x);if(!V)continue;let A=await Z.oneroster.courses.componentResources({where:{"courseComponent.sourcedId":V,status:"active"}}),C=f8(A);for(let q of C){if(!q.sourcedId)continue;let z=q.metadata??{},N=m$(q);W.push({id:q.sourcedId,name:q.title??q.sourcedId,type:z.lessonType??"quiz",courseId:_,questionCount:N,metadata:z})}}}return W}};class X8 extends Error{name="LessonTypeMismatchError"}var s$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments;if(J.lessonType==="powerpath-100"){let K=await Z.getNextQuestion({student:J.student,lesson:J.lesson});return K8(K.question)}let H=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson});if(H.lessonType==="powerpath-100"){if(J.lessonType!==void 0)throw new X8(`Expected quiz progress but got powerpath-100 for lesson ${J.lesson}`);let K=await Z.getNextQuestion({student:J.student,lesson:J.lesson});return K8(K.question)}return u$(H)}};var E0=R("lessons:start"),o$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments,H=J.lesson,K=J.student,W=await Z.getAssessmentProgress({student:K,lesson:H});if(J.forceNew){if(W.lessonType!=="powerpath-100"&&W.finalized!==!0)try{await Z.finalStudentAssessmentResponse({student:K,lesson:H})}catch{}try{await Z.createNewAttempt({student:K,lesson:H}),W=await Z.getAssessmentProgress({student:K,lesson:H})}catch(_){E0.warn("Failed to create new lesson attempt",{lessonId:H,error:_})}}else if(W.lessonType!=="powerpath-100"&&W.finalized!==!0&&(W.questions?.length??0)===0)try{await Z.createNewAttempt({student:K,lesson:H}),W=await Z.getAssessmentProgress({student:K,lesson:H})}catch{}return l$(H,W)}};var n$=($)=>{return async(J)=>{let H=await $.getClient().powerpath.assessments.updateStudentQuestionResponse({student:J.student,lesson:J.lesson,question:J.question,responses:{RESPONSE:J.response}});return c$(H)}};function m($){return{list:i$($),start:o$($),next:s$($),submit:n$($),complete:d$($),attempts:g$($),attemptDetails:f$($)}}async function a($,J){let Z=await T2($.identity,J);if(!Z)return null;if(Z.timebackId)return Z.timebackId;return await _2({email:Z.email,client:$.getClient()})}async function r$($,J){let Z=await T2($.identity,J);return Boolean(Z)}var v0=R("handlers:lessons");function c($,J){if($ instanceof d)return M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",F2($));if($ instanceof X8)return M("LESSON_TYPE_MISMATCH",$.message,400);let Z=$ instanceof Error?$.message:"Unknown error";return v0.error(J,{error:Z}),M("INTERNAL_ERROR",Z,500)}import*as i from"zod";var y0=i.object({course:M2.optional()}).partial(),f0=i.object({lessonId:E,forceNew:i.boolean().optional()}),g0=i.object({lessonId:E,lessonType:E.optional()}),d0=i.object({lessonId:E,questionId:E,response:E}),m0=i.object({lessonId:E}),l0=i.object({lessonId:E}),u0=i.object({lessonId:E,attempt:i.number().int().positive()});async function c0($){try{let J=await $.json();if(!J||typeof J!=="object"||Array.isArray(J))return null;return J}catch{return null}}async function V2($,J){let Z=await c0($);if(!Z)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid JSON body",400)};let H=J.safeParse(Z);if(!H.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:H.error.flatten()})};return{ok:!0,payload:H.data}}async function t$($){return await V2($,y0)}async function e$($){return await V2($,f0)}async function $J($){return await V2($,g0)}async function JJ($){return await V2($,d0)}async function ZJ($){return await V2($,m0)}async function HJ($){return await V2($,l0)}async function KJ($){return await V2($,u0)}function W8($){let J=m($);return async(Z)=>{try{let H=await KJ(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.attemptDetails({lesson:H.payload.lessonId,attempt:H.payload.attempt,student:K});return P(W)}catch(H){return c(H,"Failed to fetch lesson attempt details")}}}function Y8($){let J=m($);return async(Z)=>{try{let H=await HJ(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.attempts({lesson:H.payload.lessonId,student:K});return P({attempts:W})}catch(H){return c(H,"Failed to list lesson attempts")}}}function Q8($){let J=m($);return async(Z)=>{try{let H=await ZJ(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.complete({lesson:H.payload.lessonId,student:K});return P(W)}catch(H){return c(H,"Failed to complete lesson")}}}function G8($){let J=m($);return async(Z)=>{try{if(!await r$($,Z))return M("UNAUTHORIZED","Unauthorized",401);let H=await t$(Z);if(!H.ok)return H.response;let K=await J.list({course:H.payload.course});return P({lessons:K})}catch(H){return c(H,"Failed to list lessons")}}}function B8($){let J=m($);return async(Z)=>{try{let H=await $J(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.next({lesson:H.payload.lessonId,lessonType:H.payload.lessonType,student:K});return P(W)}catch(H){return c(H,"Failed to fetch next lesson question")}}}function _8($){let J=m($);return async(Z)=>{try{let H=await e$(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.start({lesson:H.payload.lessonId,student:K,forceNew:H.payload.forceNew});return P(W)}catch(H){return c(H,"Failed to start lesson")}}}function F8($){let J=m($);return async(Z)=>{try{let H=await JJ(Z);if(!H.ok)return H.response;let K=await a($,Z);if(!K)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.submit({lesson:H.payload.lessonId,question:H.payload.questionId,response:H.payload.response,student:K});return P(W)}catch(H){return c(H,"Failed to submit lesson answer")}}}function XJ($,J){let Z=new Map;for(let H of $){let K=H.ids?.[J];if(K)Z.set(K,H)}return Z}function WJ($,J){return $.map((Z)=>{let H=J.get(Z.course.id);return{id:Z.course.id,code:H?.courseCode??Z.course.id,name:Z.course.title}})}function YJ($){return $.map((J)=>J.metadata?.goals).find(Boolean)}function QJ($){let J=new Date(Date.UTC($.getUTCFullYear(),$.getUTCMonth(),$.getUTCDate())),Z=new Date(Date.UTC($.getUTCFullYear(),$.getUTCMonth(),$.getUTCDate(),23,59,59,999));return{start:J,end:Z}}function g8($){return Object.values($).reduce((J,Z)=>{return J+Object.values(Z).reduce((H,K)=>{return H+(K.activityMetrics?.xpEarned??0)},0)},0)}async function O8($,J,Z,H){let K=await $.edubridge.enrollments.list({userId:J.id}),W=XJ(Z.courses,H),_=WJ(K,W),F=YJ(K),{start:O,end:x}=QJ(new Date),[V,A]=await Promise.all([$.edubridge.analytics.getActivity({studentId:J.id,startDate:O.toISOString(),endDate:x.toISOString()}),$.edubridge.analytics.getActivity({studentId:J.id,startDate:"2000-01-01",endDate:x.toISOString()})]);return{id:J.id,email:J.email,name:J.name,school:J.school,grade:J.grade,courses:_.length?_:void 0,goals:F,xp:{today:g8(V),all:g8(A)}}}var d8=R("handlers:user");async function p0($,J){if($.mode==="custom"){let H=await $.getEmail(J);if(!H)return;return{sub:H,email:H}}let Z=await $.getUser(J);if(!Z)return;return{sub:Z.id,email:Z.email}}function a0($){if(d8.warn("Timeback user resolution failed",{code:$.code}),$.code==="timeback_user_ambiguous")return M("USER_AMBIGUOUS","Timeback user resolution ambiguous",409);if($.code==="timeback_user_not_found")return M("USER_NOT_FOUND","Timeback user not found",404);return M("USER_RESOLUTION_FAILED",$.message,502)}function D8($){return async(J)=>{try{let Z=await p0($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let H=u($.env),K=$.getClient();try{let W=await O2({env:$.env,apiCredentials:$.api,userInfo:Z,client:K}),_=await O8(K,W,$.appConfig,H);return P(_)}catch(W){if(W instanceof d)return a0(W);let _=W instanceof Error?W.message:"Unknown error";return d8.error("Failed to build user profile",{error:_}),M("INTERNAL_ERROR",_,502)}}catch(Z){let H=Z instanceof Error?Z.message:"Unknown error";return d8.error("Unhandled error in user handler",{error:H}),M("INTERNAL_ERROR",H,500)}}}var m8=R("handlers:user:verify");async function i0($,J){if($.mode==="custom"){let H=await $.getEmail(J);if(!H)return;return{sub:H,email:H}}let Z=await $.getUser(J);if(!Z)return;return{sub:Z.id,email:Z.email}}function w8($){return async(J)=>{try{let Z=await i0($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let H=$.getClient();try{let K=await O2({env:$.env,apiCredentials:$.api,userInfo:Z,client:H});return P({verified:!0,timebackId:K.id})}catch(K){if(K instanceof d){if(m8.warn("Timeback user resolution failed",{code:K.code}),K.code==="timeback_user_ambiguous")return M("USER_AMBIGUOUS","Timeback user resolution ambiguous",409);if(K.code==="timeback_user_not_found")return P({verified:!1});return M("USER_RESOLUTION_FAILED",K.message,502)}let W=K instanceof Error?K.message:"Unknown error";return m8.error("Failed to verify user",{error:W}),M("INTERNAL_ERROR",W,502)}}catch(Z){let H=Z instanceof Error?Z.message:"Unknown error";return m8.error("Unhandled error in verify handler",{error:H}),M("INTERNAL_ERROR",H,500)}}}import*as T from"zod";var s0=T.object({email:T.email(),timebackId:T.string().optional()}),o0=T.object({id:E,name:E,course:M2}),n0=T.object({totalQuestions:T.number().int().nonnegative().optional(),correctQuestions:T.number().int().nonnegative().optional(),xpEarned:T.number(),masteredUnits:T.number().int().nonnegative().optional(),pctComplete:T.number().min(0).max(100).optional()}).superRefine(o2),r0=T.object({startedAt:T.union([T.iso.datetime(),T.date()]).optional(),endedAt:T.union([T.iso.datetime(),T.date()]).optional(),activeMs:T.number().int().nonnegative().optional(),inactiveMs:T.number().int().nonnegative().optional()}),t0=T.uuid().optional(),GJ=T.object({user:s0,activity:o0,metrics:n0,time:r0.optional(),runId:t0});var BJ=R("activity:record");class _J extends Error{code="VALIDATION_ERROR";details;constructor($,J){super($);this.name="ActivityRecordValidationError",this.details=J}}class l8 extends Error{code="COURSE_ERROR";constructor($){super($);this.name="ActivityRecordCourseError"}}function e0($){let J=$.time?.endedAt?typeof $.time.endedAt==="string"?$.time.endedAt:$.time.endedAt.toISOString():new Date().toISOString();return{id:$.activity.id,name:$.activity.name,course:$.activity.course,endedAt:J,metrics:{totalQuestions:$.metrics.totalQuestions,correctQuestions:$.metrics.correctQuestions,xpEarned:$.metrics.xpEarned,masteredUnits:$.metrics.masteredUnits},pctComplete:$.metrics.pctComplete}}function $1($,J){let Z=$.time,H=Z.activeMs??0,K=Z.inactiveMs??0,W;if(Z.startedAt)W=typeof Z.startedAt==="string"?Z.startedAt:Z.startedAt.toISOString();else if(Z.activeMs!==void 0||Z.inactiveMs!==void 0){let _=new Date(J),F=H+K;W=new Date(_.getTime()-F).toISOString()}else W=J;return{id:$.activity.id,name:$.activity.name,course:$.activity.course,startedAt:W,endedAt:J,elapsedMs:H,pausedMs:K}}function J1($,J,Z){let H;try{H=j2($.courses,J)}catch(W){if(W instanceof B2)throw new l8(W.code==="unknown_course"?`Unknown course: ${JSON.stringify(J)}`:"Ambiguous course selector in timeback.config.json");throw W}let K=h2($,H,Z);if(!K)throw new l8(`Course "${JSON.stringify(J)}" has no sensor configured.`);return{course:H,sensor:K}}var Z1={recordCompletion:n2,sendTimeSpent:A$};function u8($,J=Z1){return async(Z)=>{let H=GJ.safeParse(Z);if(!H.success)throw BJ.warn("Invalid record input",{error:H.error.flatten()}),new _J("Invalid activity record input",H.error.flatten());let K=H.data,{course:W,sensor:_}=J1($.appConfig,K.activity.course,$.env),F=e0(K),O={env:$.env,api:$.api,appConfig:$.appConfig,getClient:$.getClient,hooks:$.hooks};if(await J.recordCompletion(O,{userInfo:{email:K.user.email,timebackId:K.user.timebackId},payload:F,course:W,sensor:_,preview:!1,runId:K.runId}),K.time){let x=$1(K,F.endedAt);await J.sendTimeSpent(O,{userInfo:{email:K.user.email,timebackId:K.user.timebackId},payload:x,course:W,sensor:_,runId:K.runId})}BJ.debug("Recorded activity",{activityId:K.activity.id,hasTime:!!K.time})}}var x8=R("user:verify");function c8($){return async(J)=>{x8.debug("Verifying user by email");let Z=$.getClient();try{let H=await _2({email:J,client:Z});return x8.debug("User verified",{timebackId:H}),{verified:!0,timebackId:H}}catch(H){if(H instanceof d){if(H.code==="timeback_user_not_found")return x8.debug("User not found in Timeback"),{verified:!1};throw H}let K=H instanceof Error?H.message:"Unknown error";throw x8.error("Failed to verify user",{error:K}),H}}}var p8=R("user:getProfile");function a8($){return async(J)=>{p8.debug("Building user profile by email");let Z=$.getClient(),H=u($.env);try{let K=await O2({env:$.env,apiCredentials:$.api,userInfo:{sub:J,email:J},client:Z}),W=await O8(Z,K,$.appConfig,H);return p8.debug("User profile built",{userId:W.id}),W}catch(K){if(K instanceof d)throw K;let W=K instanceof Error?K.message:"Unknown error";throw p8.error("Failed to build user profile",{error:W}),K}}}async function K1($){let{loadConfig:J}=await import("./chunk-2kfgswby.js");return await J($)}function X1($){return $.map((J)=>{if(J.grade!==void 0)return J;if(J.courseCode)return J;throw Error(`Invalid course config: missing both grade and courseCode for subject "${J.subject}"`)})}async function wJ($){if(!globalThis.process?.versions?.node&&!globalThis.Bun)throw Error("createTimeback() requires Node.js or Bun — it reads timeback.config.json from the filesystem. "+"For edge runtimes (Cloudflare Workers, Vercel Edge, etc.), use createTimebackIdentity() from '@timeback/sdk' instead.");FJ({logger:$.logger,logLevel:$.logLevel});let J=OJ($.env),Z=await K1({configPath:$.configPath});if(!Z.success)throw Error(`Failed to load timeback config: ${Z.error}`);let H=Z.config,K,W=()=>{if(!K)K=new H1({env:u(J),auth:{clientId:$.api.clientId,clientSecret:$.api.clientSecret}});return K},_=$.hooks,F=(V,A)=>DJ(V,_,A),O={env:J,identity:$.identity,appConfig:{name:H.name,sensor:H.sensor,launchUrl:H.launchUrl,courses:X1(H.courses)},api:$.api,getClient:W,hooks:_},x={env:J,identity:$.identity,api:{credentials:$.api,getClient:W}};return{config:$,handle:{activity:{heartbeat:F("activity.heartbeat",e2(O)),submit:F("activity.submit",r2(O))},identity:{signIn:F("identity.signIn",J8(x)),callback:F("identity.callback",Z8(x)),signOut:H8()},lessons:{list:F("lessons.list",G8(O)),start:F("lessons.start",_8(O)),next:F("lessons.next",B8(O)),submit:F("lessons.submit",F8(O)),complete:F("lessons.complete",Q8(O)),attempts:F("lessons.attempts",Y8(O)),attemptDetails:F("lessons.attemptDetails",W8(O))},user:{me:F("user.me",D8(O)),verify:F("user.verify",w8(O))}},activity:{record:u8({...O})},lessons:m({...O}),user:{verify:c8({...O}),getProfile:a8({...O})},get api(){return W()},close(){if(K)K.close(),K=void 0}}}import{TimebackClient as EK}from"@timeback/core";import{ApiError as yK,ForbiddenError as fK,isApiError as gK,NotFoundError as dK,UnauthorizedError as mK,ValidationError as lK}from"@timeback/core";export{Y1 as toNativeHandler,gK as isApiError,xJ as createTimebackIdentity,wJ as createTimeback,lK as ValidationError,mK as UnauthorizedError,EK as TimebackClient,W1 as ROUTES,dK as NotFoundError,fK as ForbiddenError,yK as ApiError};
@@ -1 +1 @@
1
- import"../../chunk-bh0ewh45.js";import"../../chunk-nw6brg34.js";import{B as f,C as $,D as V,E as X,F as M}from"../../chunk-3pf3cpfk.js";import{L as _}from"../../chunk-dbcsf5tk.js";import"../../chunk-1s709w5e.js";function D(K){let z=K.protocol||"http",C=K.get("host")||"localhost",G=`${z}://${C}${K.url}`,B=new Headers;for(let[Q,P]of Object.entries(K.headers))if(P)B.set(Q,Array.isArray(P)?P.join(", "):P);let J={method:K.method,headers:B};if(["POST","PUT","PATCH"].includes(K.method)&&K.body)J.body=JSON.stringify(K.body);return new Request(G,J)}async function Z(K,z){z.status(K.status);let C={};if(K.headers.forEach((B,J)=>{C[J]=B}),z.set(C),K.status>=300&&K.status<400){let B=K.headers.get("Location");if(B){z.redirect(B);return}}let G=await K.text();z.send(G)}function N(K){let z=$(K);return(C,G,B)=>{let J=D(C),Q=new URL(J.url).pathname;(()=>{let F=f({pathname:Q,method:C.method});if(!F)return Promise.resolve(void 0);if(F==="identity.signOut")return Promise.resolve(z.identity.signOut());if(F==="identity.signIn")return z.identity.signIn(J);if(F==="identity.callback")return z.identity.callback(J);if(F==="user.me"){if(!X(z))return Promise.resolve(void 0);return z.user.me(J)}if(F==="user.verify"){if(!X(z))return Promise.resolve(void 0);return z.user.verify(J)}if(F==="activity.heartbeat"){if(!V(z))return Promise.resolve(void 0);return z.activity.heartbeat(J)}if(F==="activity.submit"){if(!V(z))return Promise.resolve(void 0);return z.activity.submit(J)}if(F==="lessons.list"){if(!M(z))return Promise.resolve(void 0);return z.lessons.list(J)}if(F==="lessons.start"){if(!M(z))return Promise.resolve(void 0);return z.lessons.start(J)}if(F==="lessons.next"){if(!M(z))return Promise.resolve(void 0);return z.lessons.next(J)}if(F==="lessons.submit"){if(!M(z))return Promise.resolve(void 0);return z.lessons.submit(J)}if(F==="lessons.complete"){if(!M(z))return Promise.resolve(void 0);return z.lessons.complete(J)}if(F==="lessons.attempts"){if(!M(z))return Promise.resolve(void 0);return z.lessons.attempts(J)}if(F==="lessons.attemptDetails"){if(!M(z))return Promise.resolve(void 0);return z.lessons.attemptDetails(J)}return Promise.resolve(void 0)})().then((F)=>{if(F)return Z(F,G);B()}).catch(()=>{let F=_({error:"Internal server error"},500);return Z(F,G)})}}function O(K,z){let C=$(K);function G(B){return(J,Q)=>{let P=D(J);Promise.resolve(B(P)).then((F)=>Z(F,Q)).catch(()=>{let F=_({error:"Internal server error"},500);return Z(F,Q)})}}if(z.get("/identity/signin",G((B)=>C.identity.signIn(B))),z.get("/identity/callback",G((B)=>C.identity.callback(B))),z.get("/identity/signout",G(()=>C.identity.signOut())),X(C))z.get("/user/me",G((B)=>C.user.me(B))),z.get("/user/verify",G((B)=>C.user.verify(B)));if(V(C))z.post("/activity/heartbeat",G((B)=>C.activity.heartbeat(B))),z.post("/activity/submit",G((B)=>C.activity.submit(B)));if(M(C))z.post("/lessons/list",G((B)=>C.lessons.list(B))),z.post("/lessons/start",G((B)=>C.lessons.start(B))),z.post("/lessons/next",G((B)=>C.lessons.next(B))),z.post("/lessons/submit",G((B)=>C.lessons.submit(B))),z.post("/lessons/complete",G((B)=>C.lessons.complete(B))),z.post("/lessons/attempts",G((B)=>C.lessons.attempts(B))),z.post("/lessons/attempt-details",G((B)=>C.lessons.attemptDetails(B)))}export{N as toExpressMiddleware,O as mountExpressRoutes};
1
+ import"../../chunk-k1awna2q.js";import"../../chunk-b49f5qfw.js";import{B as f,C as $,D as V,E as X,F as M}from"../../chunk-3pf3cpfk.js";import{L as _}from"../../chunk-dbcsf5tk.js";import"../../chunk-1s709w5e.js";function D(K){let z=K.protocol||"http",C=K.get("host")||"localhost",G=`${z}://${C}${K.url}`,B=new Headers;for(let[Q,P]of Object.entries(K.headers))if(P)B.set(Q,Array.isArray(P)?P.join(", "):P);let J={method:K.method,headers:B};if(["POST","PUT","PATCH"].includes(K.method)&&K.body)J.body=JSON.stringify(K.body);return new Request(G,J)}async function Z(K,z){z.status(K.status);let C={};if(K.headers.forEach((B,J)=>{C[J]=B}),z.set(C),K.status>=300&&K.status<400){let B=K.headers.get("Location");if(B){z.redirect(B);return}}let G=await K.text();z.send(G)}function N(K){let z=$(K);return(C,G,B)=>{let J=D(C),Q=new URL(J.url).pathname;(()=>{let F=f({pathname:Q,method:C.method});if(!F)return Promise.resolve(void 0);if(F==="identity.signOut")return Promise.resolve(z.identity.signOut());if(F==="identity.signIn")return z.identity.signIn(J);if(F==="identity.callback")return z.identity.callback(J);if(F==="user.me"){if(!X(z))return Promise.resolve(void 0);return z.user.me(J)}if(F==="user.verify"){if(!X(z))return Promise.resolve(void 0);return z.user.verify(J)}if(F==="activity.heartbeat"){if(!V(z))return Promise.resolve(void 0);return z.activity.heartbeat(J)}if(F==="activity.submit"){if(!V(z))return Promise.resolve(void 0);return z.activity.submit(J)}if(F==="lessons.list"){if(!M(z))return Promise.resolve(void 0);return z.lessons.list(J)}if(F==="lessons.start"){if(!M(z))return Promise.resolve(void 0);return z.lessons.start(J)}if(F==="lessons.next"){if(!M(z))return Promise.resolve(void 0);return z.lessons.next(J)}if(F==="lessons.submit"){if(!M(z))return Promise.resolve(void 0);return z.lessons.submit(J)}if(F==="lessons.complete"){if(!M(z))return Promise.resolve(void 0);return z.lessons.complete(J)}if(F==="lessons.attempts"){if(!M(z))return Promise.resolve(void 0);return z.lessons.attempts(J)}if(F==="lessons.attemptDetails"){if(!M(z))return Promise.resolve(void 0);return z.lessons.attemptDetails(J)}return Promise.resolve(void 0)})().then((F)=>{if(F)return Z(F,G);B()}).catch(()=>{let F=_({error:"Internal server error"},500);return Z(F,G)})}}function O(K,z){let C=$(K);function G(B){return(J,Q)=>{let P=D(J);Promise.resolve(B(P)).then((F)=>Z(F,Q)).catch(()=>{let F=_({error:"Internal server error"},500);return Z(F,Q)})}}if(z.get("/identity/signin",G((B)=>C.identity.signIn(B))),z.get("/identity/callback",G((B)=>C.identity.callback(B))),z.get("/identity/signout",G(()=>C.identity.signOut())),X(C))z.get("/user/me",G((B)=>C.user.me(B))),z.get("/user/verify",G((B)=>C.user.verify(B)));if(V(C))z.post("/activity/heartbeat",G((B)=>C.activity.heartbeat(B))),z.post("/activity/submit",G((B)=>C.activity.submit(B)));if(M(C))z.post("/lessons/list",G((B)=>C.lessons.list(B))),z.post("/lessons/start",G((B)=>C.lessons.start(B))),z.post("/lessons/next",G((B)=>C.lessons.next(B))),z.post("/lessons/submit",G((B)=>C.lessons.submit(B))),z.post("/lessons/complete",G((B)=>C.lessons.complete(B))),z.post("/lessons/attempts",G((B)=>C.lessons.attempts(B))),z.post("/lessons/attempt-details",G((B)=>C.lessons.attemptDetails(B)))}export{N as toExpressMiddleware,O as mountExpressRoutes};
@@ -1 +1 @@
1
- import"../../chunk-bh0ewh45.js";import"../../chunk-nw6brg34.js";import{A as N,B as I,C as O,D as z,E as B,F as S}from"../../chunk-3pf3cpfk.js";import{L as X}from"../../chunk-dbcsf5tk.js";import"../../chunk-1s709w5e.js";function A(x){return new Promise((J,K)=>{let Y=[];x.on("data",(C)=>{if(!C)return;if(C instanceof Buffer)return void Y.push(C);if(typeof C==="string")return void Y.push(Buffer.from(C,"utf-8"));if(C instanceof Uint8Array)return void Y.push(Buffer.from(C))}),x.on("end",()=>J(Buffer.concat(Y).toString("utf-8"))),x.on("error",(C)=>{if(C instanceof Error)K(C)})})}function V(x){if(x.path)return N(x.path);if(x.request){let K=x.request.url;if(K.startsWith("http://")||K.startsWith("https://"))return new URL(K).pathname;return K.split("?")[0]??"/"}return(x.node?.req?.url?.split("?")[0]??"/")||"/"}function M(x){if(!x)return;if(Array.isArray(x))return x[0]?.trim();return x.split(",")[0]?.trim()}function E(x){let J=x.url??"/";if(J.startsWith("http://")||J.startsWith("https://"))return new URL(J).href;let K=M(x.headers["x-forwarded-proto"]),Y=K==="https"?"https":K==="http"?"http":"http",_=M(x.headers["x-forwarded-host"])||x.headers.host||"localhost";return new URL(J,`${Y}://${_}`).href}function w(x){if(x.request)return x.request.url;let J=x.node?.req;if(!J)throw Error("Cannot determine URL from Nuxt event");return E(J)}function W(x){if(x.request)return x.request.method;return x.node?.req?.method??"GET"}async function T(x){if(x.request)return x.request;let J=x.node?.req;if(!J)throw Error("Cannot convert Nuxt event to Request: no request object found");let K=w(x),Y=J.method??"GET",C=new Headers;for(let[$,D]of Object.entries(J.headers))if(D)C.set($,Array.isArray(D)?D.join(", "):D);let _={method:Y,headers:C};if(["POST","PUT","PATCH","DELETE"].includes(Y)){let $=await A(J);if($)_.body=$}return new Request(K,_)}async function P(x){let{timeback:J,event:K,basePath:Y="/api/timeback",callbackPath:C}=x,_=V(K),$=W(K),D=O(J),Q=I({pathname:_,method:$,basePath:Y,callbackPath:C});if(!Q)return;if(Q==="identity.signOut")return D.identity.signOut();let G=await T(K);if(Q==="identity.signIn")return D.identity.signIn(G);if(Q==="identity.callback")return D.identity.callback(G);if(Q==="user.me"){if(!B(D))return;return D.user.me(G)}if(Q==="user.verify"){if(!B(D))return;return D.user.verify(G)}if(Q==="activity.heartbeat"){if(!z(D))return;return D.activity.heartbeat(G)}if(Q==="activity.submit"){if(!z(D))return;return D.activity.submit(G)}if(Q==="lessons.list"){if(!S(D))return;return D.lessons.list(G)}if(Q==="lessons.start"){if(!S(D))return;return D.lessons.start(G)}if(Q==="lessons.next"){if(!S(D))return;return D.lessons.next(G)}if(Q==="lessons.submit"){if(!S(D))return;return D.lessons.submit(G)}if(Q==="lessons.complete"){if(!S(D))return;return D.lessons.complete(G)}if(Q==="lessons.attempts"){if(!S(D))return;return D.lessons.attempts(G)}if(Q==="lessons.attemptDetails"){if(!S(D))return;return D.lessons.attemptDetails(G)}return}function y(x){let J=F(x)?x:{timeback:x},{timeback:K,callbackPath:Y}=J,C=O(K),_=async($)=>{let D=V($),Q=W($),G=I({pathname:D,method:Q,callbackPath:Y});if(!G)return X({error:"Not found"},404);if(G==="identity.signOut")return C.identity.signOut();let Z=await T($);if(G==="identity.signIn")return C.identity.signIn(Z);if(G==="identity.callback")return C.identity.callback(Z);if(G==="user.me"){if(!B(C))return X({error:"Not found"},404);return C.user.me(Z)}if(G==="user.verify"){if(!B(C))return X({error:"Not found"},404);return C.user.verify(Z)}if(G==="activity.heartbeat"){if(!z(C))return X({error:"Not found"},404);return C.activity.heartbeat(Z)}if(G==="activity.submit"){if(!z(C))return X({error:"Not found"},404);return C.activity.submit(Z)}if(G==="lessons.list"){if(!S(C))return X({error:"Not found"},404);return C.lessons.list(Z)}if(G==="lessons.start"){if(!S(C))return X({error:"Not found"},404);return C.lessons.start(Z)}if(G==="lessons.next"){if(!S(C))return X({error:"Not found"},404);return C.lessons.next(Z)}if(G==="lessons.submit"){if(!S(C))return X({error:"Not found"},404);return C.lessons.submit(Z)}if(G==="lessons.complete"){if(!S(C))return X({error:"Not found"},404);return C.lessons.complete(Z)}if(G==="lessons.attempts"){if(!S(C))return X({error:"Not found"},404);return C.lessons.attempts(Z)}if(G==="lessons.attemptDetails"){if(!S(C))return X({error:"Not found"},404);return C.lessons.attemptDetails(Z)}return X({error:"Not found"},404)};return{GET:_,POST:_,PUT:_,DELETE:_,PATCH:_}}function F(x){return typeof x==="object"&&x!==null&&"timeback"in x}export{y as toNuxtHandler,P as nuxtHandler};
1
+ import"../../chunk-k1awna2q.js";import"../../chunk-b49f5qfw.js";import{A as N,B as I,C as O,D as z,E as B,F as S}from"../../chunk-3pf3cpfk.js";import{L as X}from"../../chunk-dbcsf5tk.js";import"../../chunk-1s709w5e.js";function A(x){return new Promise((J,K)=>{let Y=[];x.on("data",(C)=>{if(!C)return;if(C instanceof Buffer)return void Y.push(C);if(typeof C==="string")return void Y.push(Buffer.from(C,"utf-8"));if(C instanceof Uint8Array)return void Y.push(Buffer.from(C))}),x.on("end",()=>J(Buffer.concat(Y).toString("utf-8"))),x.on("error",(C)=>{if(C instanceof Error)K(C)})})}function V(x){if(x.path)return N(x.path);if(x.request){let K=x.request.url;if(K.startsWith("http://")||K.startsWith("https://"))return new URL(K).pathname;return K.split("?")[0]??"/"}return(x.node?.req?.url?.split("?")[0]??"/")||"/"}function M(x){if(!x)return;if(Array.isArray(x))return x[0]?.trim();return x.split(",")[0]?.trim()}function E(x){let J=x.url??"/";if(J.startsWith("http://")||J.startsWith("https://"))return new URL(J).href;let K=M(x.headers["x-forwarded-proto"]),Y=K==="https"?"https":K==="http"?"http":"http",_=M(x.headers["x-forwarded-host"])||x.headers.host||"localhost";return new URL(J,`${Y}://${_}`).href}function w(x){if(x.request)return x.request.url;let J=x.node?.req;if(!J)throw Error("Cannot determine URL from Nuxt event");return E(J)}function W(x){if(x.request)return x.request.method;return x.node?.req?.method??"GET"}async function T(x){if(x.request)return x.request;let J=x.node?.req;if(!J)throw Error("Cannot convert Nuxt event to Request: no request object found");let K=w(x),Y=J.method??"GET",C=new Headers;for(let[$,D]of Object.entries(J.headers))if(D)C.set($,Array.isArray(D)?D.join(", "):D);let _={method:Y,headers:C};if(["POST","PUT","PATCH","DELETE"].includes(Y)){let $=await A(J);if($)_.body=$}return new Request(K,_)}async function P(x){let{timeback:J,event:K,basePath:Y="/api/timeback",callbackPath:C}=x,_=V(K),$=W(K),D=O(J),Q=I({pathname:_,method:$,basePath:Y,callbackPath:C});if(!Q)return;if(Q==="identity.signOut")return D.identity.signOut();let G=await T(K);if(Q==="identity.signIn")return D.identity.signIn(G);if(Q==="identity.callback")return D.identity.callback(G);if(Q==="user.me"){if(!B(D))return;return D.user.me(G)}if(Q==="user.verify"){if(!B(D))return;return D.user.verify(G)}if(Q==="activity.heartbeat"){if(!z(D))return;return D.activity.heartbeat(G)}if(Q==="activity.submit"){if(!z(D))return;return D.activity.submit(G)}if(Q==="lessons.list"){if(!S(D))return;return D.lessons.list(G)}if(Q==="lessons.start"){if(!S(D))return;return D.lessons.start(G)}if(Q==="lessons.next"){if(!S(D))return;return D.lessons.next(G)}if(Q==="lessons.submit"){if(!S(D))return;return D.lessons.submit(G)}if(Q==="lessons.complete"){if(!S(D))return;return D.lessons.complete(G)}if(Q==="lessons.attempts"){if(!S(D))return;return D.lessons.attempts(G)}if(Q==="lessons.attemptDetails"){if(!S(D))return;return D.lessons.attemptDetails(G)}return}function y(x){let J=F(x)?x:{timeback:x},{timeback:K,callbackPath:Y}=J,C=O(K),_=async($)=>{let D=V($),Q=W($),G=I({pathname:D,method:Q,callbackPath:Y});if(!G)return X({error:"Not found"},404);if(G==="identity.signOut")return C.identity.signOut();let Z=await T($);if(G==="identity.signIn")return C.identity.signIn(Z);if(G==="identity.callback")return C.identity.callback(Z);if(G==="user.me"){if(!B(C))return X({error:"Not found"},404);return C.user.me(Z)}if(G==="user.verify"){if(!B(C))return X({error:"Not found"},404);return C.user.verify(Z)}if(G==="activity.heartbeat"){if(!z(C))return X({error:"Not found"},404);return C.activity.heartbeat(Z)}if(G==="activity.submit"){if(!z(C))return X({error:"Not found"},404);return C.activity.submit(Z)}if(G==="lessons.list"){if(!S(C))return X({error:"Not found"},404);return C.lessons.list(Z)}if(G==="lessons.start"){if(!S(C))return X({error:"Not found"},404);return C.lessons.start(Z)}if(G==="lessons.next"){if(!S(C))return X({error:"Not found"},404);return C.lessons.next(Z)}if(G==="lessons.submit"){if(!S(C))return X({error:"Not found"},404);return C.lessons.submit(Z)}if(G==="lessons.complete"){if(!S(C))return X({error:"Not found"},404);return C.lessons.complete(Z)}if(G==="lessons.attempts"){if(!S(C))return X({error:"Not found"},404);return C.lessons.attempts(Z)}if(G==="lessons.attemptDetails"){if(!S(C))return X({error:"Not found"},404);return C.lessons.attemptDetails(Z)}return X({error:"Not found"},404)};return{GET:_,POST:_,PUT:_,DELETE:_,PATCH:_}}function F(x){return typeof x==="object"&&x!==null&&"timeback"in x}export{y as toNuxtHandler,P as nuxtHandler};
@@ -1 +1 @@
1
- import"../../chunk-bh0ewh45.js";import"../../chunk-nw6brg34.js";import{B as Q,C as V,D as J,E as K,F}from"../../chunk-3pf3cpfk.js";import{L as G}from"../../chunk-dbcsf5tk.js";import"../../chunk-1s709w5e.js";function $(N){let B=V(N),I=(C)=>{let D=new URL(C.request.url).pathname,O=C.request.method,z=Q({pathname:D,method:O});if(!z)return G({error:"Not found"},404);if(z==="identity.signOut")return B.identity.signOut();if(z==="identity.signIn")return B.identity.signIn(C.request);if(z==="identity.callback")return B.identity.callback(C.request);if(z==="user.me"){if(!K(B))return G({error:"Not found"},404);return B.user.me(C.request)}if(z==="user.verify"){if(!K(B))return G({error:"Not found"},404);return B.user.verify(C.request)}if(z==="activity.heartbeat"){if(!J(B))return G({error:"Not found"},404);return B.activity.heartbeat(C.request)}if(z==="activity.submit"){if(!J(B))return G({error:"Not found"},404);return B.activity.submit(C.request)}if(z==="lessons.list"){if(!F(B))return G({error:"Not found"},404);return B.lessons.list(C.request)}if(z==="lessons.start"){if(!F(B))return G({error:"Not found"},404);return B.lessons.start(C.request)}if(z==="lessons.next"){if(!F(B))return G({error:"Not found"},404);return B.lessons.next(C.request)}if(z==="lessons.submit"){if(!F(B))return G({error:"Not found"},404);return B.lessons.submit(C.request)}if(z==="lessons.complete"){if(!F(B))return G({error:"Not found"},404);return B.lessons.complete(C.request)}if(z==="lessons.attempts"){if(!F(B))return G({error:"Not found"},404);return B.lessons.attempts(C.request)}if(z==="lessons.attemptDetails"){if(!F(B))return G({error:"Not found"},404);return B.lessons.attemptDetails(C.request)}return G({error:"Not found"},404)};return{GET:I,POST:I,PUT:I,DELETE:I,PATCH:I}}function g(N){let{timeback:B,event:I,basePath:C="/api/timeback",callbackPath:W}=N,D=I.request,O=new URL(D.url),z=V(B),X=O.pathname,Y=D.method,E=Q({pathname:X,method:Y,basePath:C,callbackPath:W});if(!E)return;if(E==="identity.signOut")return z.identity.signOut();if(E==="identity.signIn")return z.identity.signIn(D);if(E==="identity.callback")return z.identity.callback(D);if(E==="user.me"){if(!K(z))return;return z.user.me(D)}if(E==="user.verify"){if(!K(z))return;return z.user.verify(D)}if(E==="activity.heartbeat"){if(!J(z))return;return z.activity.heartbeat(D)}if(E==="activity.submit"){if(!J(z))return;return z.activity.submit(D)}if(E==="lessons.list"){if(!F(z))return;return z.lessons.list(D)}if(E==="lessons.start"){if(!F(z))return;return z.lessons.start(D)}if(E==="lessons.next"){if(!F(z))return;return z.lessons.next(D)}if(E==="lessons.submit"){if(!F(z))return;return z.lessons.submit(D)}if(E==="lessons.complete"){if(!F(z))return;return z.lessons.complete(D)}if(E==="lessons.attempts"){if(!F(z))return;return z.lessons.attempts(D)}if(E==="lessons.attemptDetails"){if(!F(z))return;return z.lessons.attemptDetails(D)}return}export{$ as toSolidStartHandler,g as solidStartHandler};
1
+ import"../../chunk-k1awna2q.js";import"../../chunk-b49f5qfw.js";import{B as Q,C as V,D as J,E as K,F}from"../../chunk-3pf3cpfk.js";import{L as G}from"../../chunk-dbcsf5tk.js";import"../../chunk-1s709w5e.js";function $(N){let B=V(N),I=(C)=>{let D=new URL(C.request.url).pathname,O=C.request.method,z=Q({pathname:D,method:O});if(!z)return G({error:"Not found"},404);if(z==="identity.signOut")return B.identity.signOut();if(z==="identity.signIn")return B.identity.signIn(C.request);if(z==="identity.callback")return B.identity.callback(C.request);if(z==="user.me"){if(!K(B))return G({error:"Not found"},404);return B.user.me(C.request)}if(z==="user.verify"){if(!K(B))return G({error:"Not found"},404);return B.user.verify(C.request)}if(z==="activity.heartbeat"){if(!J(B))return G({error:"Not found"},404);return B.activity.heartbeat(C.request)}if(z==="activity.submit"){if(!J(B))return G({error:"Not found"},404);return B.activity.submit(C.request)}if(z==="lessons.list"){if(!F(B))return G({error:"Not found"},404);return B.lessons.list(C.request)}if(z==="lessons.start"){if(!F(B))return G({error:"Not found"},404);return B.lessons.start(C.request)}if(z==="lessons.next"){if(!F(B))return G({error:"Not found"},404);return B.lessons.next(C.request)}if(z==="lessons.submit"){if(!F(B))return G({error:"Not found"},404);return B.lessons.submit(C.request)}if(z==="lessons.complete"){if(!F(B))return G({error:"Not found"},404);return B.lessons.complete(C.request)}if(z==="lessons.attempts"){if(!F(B))return G({error:"Not found"},404);return B.lessons.attempts(C.request)}if(z==="lessons.attemptDetails"){if(!F(B))return G({error:"Not found"},404);return B.lessons.attemptDetails(C.request)}return G({error:"Not found"},404)};return{GET:I,POST:I,PUT:I,DELETE:I,PATCH:I}}function g(N){let{timeback:B,event:I,basePath:C="/api/timeback",callbackPath:W}=N,D=I.request,O=new URL(D.url),z=V(B),X=O.pathname,Y=D.method,E=Q({pathname:X,method:Y,basePath:C,callbackPath:W});if(!E)return;if(E==="identity.signOut")return z.identity.signOut();if(E==="identity.signIn")return z.identity.signIn(D);if(E==="identity.callback")return z.identity.callback(D);if(E==="user.me"){if(!K(z))return;return z.user.me(D)}if(E==="user.verify"){if(!K(z))return;return z.user.verify(D)}if(E==="activity.heartbeat"){if(!J(z))return;return z.activity.heartbeat(D)}if(E==="activity.submit"){if(!J(z))return;return z.activity.submit(D)}if(E==="lessons.list"){if(!F(z))return;return z.lessons.list(D)}if(E==="lessons.start"){if(!F(z))return;return z.lessons.start(D)}if(E==="lessons.next"){if(!F(z))return;return z.lessons.next(D)}if(E==="lessons.submit"){if(!F(z))return;return z.lessons.submit(D)}if(E==="lessons.complete"){if(!F(z))return;return z.lessons.complete(D)}if(E==="lessons.attempts"){if(!F(z))return;return z.lessons.attempts(D)}if(E==="lessons.attemptDetails"){if(!F(z))return;return z.lessons.attemptDetails(D)}return}export{$ as toSolidStartHandler,g as solidStartHandler};
@@ -1 +1 @@
1
- import"../../chunk-bh0ewh45.js";import"../../chunk-nw6brg34.js";import{B as I,C as J,D as F,E as G,F as C}from"../../chunk-3pf3cpfk.js";import{L as B}from"../../chunk-dbcsf5tk.js";import"../../chunk-1s709w5e.js";function W(D){let K=X(D)?D:{timeback:D},{timeback:M,callbackPath:N}=K,q=J(M),E=(x)=>{let Q=new URL(x.request.url).pathname,V=x.request.method,z=I({pathname:Q,method:V,callbackPath:N});if(!z)return B({error:"Not found"},404);if(z==="identity.signOut")return q.identity.signOut();if(z==="identity.signIn")return q.identity.signIn(x.request);if(z==="identity.callback")return q.identity.callback(x.request);if(z==="user.me"){if(!G(q))return B({error:"Not found"},404);return q.user.me(x.request)}if(z==="user.verify"){if(!G(q))return B({error:"Not found"},404);return q.user.verify(x.request)}if(z==="activity.heartbeat"){if(!F(q))return B({error:"Not found"},404);return q.activity.heartbeat(x.request)}if(z==="activity.submit"){if(!F(q))return B({error:"Not found"},404);return q.activity.submit(x.request)}if(z==="lessons.list"){if(!C(q))return B({error:"Not found"},404);return q.lessons.list(x.request)}if(z==="lessons.start"){if(!C(q))return B({error:"Not found"},404);return q.lessons.start(x.request)}if(z==="lessons.next"){if(!C(q))return B({error:"Not found"},404);return q.lessons.next(x.request)}if(z==="lessons.submit"){if(!C(q))return B({error:"Not found"},404);return q.lessons.submit(x.request)}if(z==="lessons.complete"){if(!C(q))return B({error:"Not found"},404);return q.lessons.complete(x.request)}if(z==="lessons.attempts"){if(!C(q))return B({error:"Not found"},404);return q.lessons.attempts(x.request)}if(z==="lessons.attemptDetails"){if(!C(q))return B({error:"Not found"},404);return q.lessons.attemptDetails(x.request)}return B({error:"Not found"},404)};return{GET:E,POST:E,PUT:E,DELETE:E,PATCH:E}}function X(D){return"timeback"in D}var $=W;export{W as toTanStackStartHandler,$ as tanstackStartHandler};
1
+ import"../../chunk-k1awna2q.js";import"../../chunk-b49f5qfw.js";import{B as I,C as J,D as F,E as G,F as C}from"../../chunk-3pf3cpfk.js";import{L as B}from"../../chunk-dbcsf5tk.js";import"../../chunk-1s709w5e.js";function W(D){let K=X(D)?D:{timeback:D},{timeback:M,callbackPath:N}=K,q=J(M),E=(x)=>{let Q=new URL(x.request.url).pathname,V=x.request.method,z=I({pathname:Q,method:V,callbackPath:N});if(!z)return B({error:"Not found"},404);if(z==="identity.signOut")return q.identity.signOut();if(z==="identity.signIn")return q.identity.signIn(x.request);if(z==="identity.callback")return q.identity.callback(x.request);if(z==="user.me"){if(!G(q))return B({error:"Not found"},404);return q.user.me(x.request)}if(z==="user.verify"){if(!G(q))return B({error:"Not found"},404);return q.user.verify(x.request)}if(z==="activity.heartbeat"){if(!F(q))return B({error:"Not found"},404);return q.activity.heartbeat(x.request)}if(z==="activity.submit"){if(!F(q))return B({error:"Not found"},404);return q.activity.submit(x.request)}if(z==="lessons.list"){if(!C(q))return B({error:"Not found"},404);return q.lessons.list(x.request)}if(z==="lessons.start"){if(!C(q))return B({error:"Not found"},404);return q.lessons.start(x.request)}if(z==="lessons.next"){if(!C(q))return B({error:"Not found"},404);return q.lessons.next(x.request)}if(z==="lessons.submit"){if(!C(q))return B({error:"Not found"},404);return q.lessons.submit(x.request)}if(z==="lessons.complete"){if(!C(q))return B({error:"Not found"},404);return q.lessons.complete(x.request)}if(z==="lessons.attempts"){if(!C(q))return B({error:"Not found"},404);return q.lessons.attempts(x.request)}if(z==="lessons.attemptDetails"){if(!C(q))return B({error:"Not found"},404);return q.lessons.attemptDetails(x.request)}return B({error:"Not found"},404)};return{GET:E,POST:E,PUT:E,DELETE:E,PATCH:E}}function X(D){return"timeback"in D}var $=W;export{W as toTanStackStartHandler,$ as tanstackStartHandler};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@timeback/sdk",
3
- "version": "0.2.4-beta.20260401223329",
3
+ "version": "0.2.4-beta.20260403024308",
4
4
  "description": "Timeback SDK for frontend and backend integration",
5
5
  "type": "module",
6
6
  "exports": {
@@ -1,2 +0,0 @@
1
- import{R as C,U as w}from"./chunk-1s709w5e.js";var q=null,G=!1,N=!1;function T(z){let Q=z.replace(/[.+?^${}()|[\]\\]/g,"\\$&").replace(/\*/g,".*");return new RegExp(`^${Q}$`)}function b(){if(q!==null)return;if(q=[],typeof process>"u"||!process.env?.DEBUG){N=!1;return}N=!0;let z=process.env.DEBUG.trim();if(z==="1"||z==="true"||z==="*"){G=!0;return}let K=z.split(",").map((Z)=>Z.trim()).filter(Boolean);for(let Z of K)if(Z.startsWith("-"))q.push({regex:T(Z.slice(1)),exclude:!0});else q.push({regex:T(Z),exclude:!1});if(!q.some((Z)=>!Z.exclude)&&q.length>0)G=!0}function V(z){if(b(),!N)return!0;if(G){if(z){for(let K of q)if(K.exclude&&K.regex.test(z))return!1}return!0}if(!z)return!1;for(let K of q)if(K.exclude&&K.regex.test(z))return!1;for(let K of q)if(!K.exclude&&K.regex.test(z))return!0;return!1}function P(){return typeof globalThis<"u"&&"window"in globalThis}function U(){if(P())return"browser";if(typeof process<"u"&&process.env){if(process.env.CI||process.env.GITHUB_ACTIONS||process.env.GITLAB_CI||process.env.CIRCLECI||process.env.JENKINS_URL||process.env.BUILDKITE)return"ci"}return"terminal"}var A;if(!P())try{A=(await import("./chunk-zpbvsyr1.js")).inspect}catch{}var W={reset:"\x1B[0m",bold:"\x1B[1m",dim:"\x1B[2m",red:"\x1B[31m",yellow:"\x1B[33m",blue:"\x1B[34m",cyan:"\x1B[36m"};function L(z){switch(z){case"debug":return W.blue;case"info":return W.cyan;case"warn":return W.yellow;case"error":return W.red}}function d(z){switch(z){case"debug":return console.debug;case"info":return console.info;case"warn":return console.warn;case"error":return console.error}}function m(z){if(A)return A(z,{depth:null,colors:!0,breakLength:80,compact:!1});return JSON.stringify(z,null,2)}var j=(z)=>{let K=L(z.level),Q=d(z.level),Z=z.level.toUpperCase().padEnd(5),$=z.timestamp.toISOString().replace(/\.\d{3}Z$/,""),h=`${W.dim}[${$}]${W.reset}`,f=`${K}${Z}${W.reset}`,x=z.scope?`${W.bold}[${z.scope}]${W.reset} `:"",O=`${h} ${f} ${x}${z.message}`;if(z.context&&Object.keys(z.context).length>0)Q(O,m(z.context));else Q(O)};var v={debug:"[DEBUG]",info:"[INFO]",warn:"[WARN]",error:"[ERROR]"};function y(z){return Object.entries(z).map(([K,Q])=>`${K}=${u(Q)}`).join(" ")}function u(z){if(typeof z==="string")return z;if(typeof z==="number")return String(z);if(typeof z==="boolean")return String(z);if(z===null)return"null";if(z===void 0)return"undefined";return JSON.stringify(z)}var B=(z)=>{let K=[];if(K.push(z.timestamp.toISOString()),K.push(v[z.level]),z.scope)K.push(`[${z.scope}]`);if(K.push(z.message),z.context&&Object.keys(z.context).length>0)K.push(y(z.context));console.log(K.join(" "))};var F=(z)=>{let K={timestamp:z.timestamp.toISOString(),level:z.level,...z.scope&&{scope:z.scope},msg:z.message};if(z.context&&Object.keys(z.context).length>0)Object.assign(K,z.context);console.log(JSON.stringify(K))};var i={debug:"color: gray",info:"color: #0ea5e9",warn:"color: #f59e0b",error:"color: #ef4444; font-weight: bold"},c={debug:"log",info:"info",warn:"warn",error:"error"},H=(z)=>{let K=c[z.level],Q=i[z.level],$=`%c${z.scope?`[${z.scope}]`:""} ${z.message}`;if(z.context&&Object.keys(z.context).length>0)console[K]($,Q,z.context);else console[K]($,Q)};var D=["debug","info","warn","error"];function p(z){switch(z){case"terminal":return j;case"ci":return B;case"production":return F;case"browser":return H;case"test":return()=>{}}}function g(){if(typeof process<"u"&&process.env?.DEBUG)return"debug";return"info"}function l(z,K){return D.indexOf(z)>=D.indexOf(K)}class X{scope;minLevel;environment;formatter;defaultContext;constructor(z={}){this.scope=z.scope,this.minLevel=z.minLevel??g(),this.defaultContext=z.defaultContext??{},this.environment=z.environment??U(),this.formatter=p(this.environment)}child(z){let K=this.scope?`${this.scope}:${z}`:z;return new X({scope:K,minLevel:this.minLevel,environment:this.environment,defaultContext:{...this.defaultContext}})}withContext(z){return new X({scope:this.scope,minLevel:this.minLevel,environment:this.environment,defaultContext:{...this.defaultContext,...z}})}debug(z,K){this.log("debug",z,K)}info(z,K){this.log("info",z,K)}warn(z,K){this.log("warn",z,K)}error(z,K){this.log("error",z,K)}log(z,K,Q){if(z==="debug"&&!V(this.scope))return;if(!l(z,this.minLevel))return;let Z={level:z,message:K,scope:this.scope,context:Q||Object.keys(this.defaultContext).length>0?{...this.defaultContext,...Q}:void 0,timestamp:new Date};this.formatter(Z)}}function M(z={}){return new X(z)}var R,_,a={debug(){},info(){},warn(){},error(){}};function n(){try{let z=typeof process>"u"?void 0:process.env.DEBUG;return z==="1"||z==="true"}catch{return!1}}var k=new Map;function o(z,K){let Q=`[${K}]`;return{debug:(...Z)=>z.debug(`${Q} ${Z[0]}`,...Z.slice(1)),info:(...Z)=>z.info(`${Q} ${Z[0]}`,...Z.slice(1)),warn:(...Z)=>z.warn(`${Q} ${Z[0]}`,...Z.slice(1)),error:(...Z)=>z.error(`${Q} ${Z[0]}`,...Z.slice(1))}}function J(z){let K=k.get(z);if(K)return K;if(R)K=o(R,z);else{let Q=_??(n()?"debug":"warn");if(Q==="silent")K=a;else K=M({scope:z,minLevel:Q})}return k.set(z,K),K}function Mz(z){R=z.logger,_=z.logLevel,k.clear()}function E(z){let K=`timeback:${z}`;return{debug(Q,...Z){J(K).debug(Q,...Z)},info(Q,...Z){J(K).info(Q,...Z)},warn(Q,...Z){J(K).warn(Q,...Z)},error(Q,...Z){J(K).error(Q,...Z)}}}var Rz=E("sso"),Y=E("oidc");var S=new Map;async function I(z){let K=S.get(z);if(K)return K;let Q=`${z}/.well-known/openid-configuration`,Z=await fetch(Q);if(!Z.ok)throw Y.error("Discovery fetch failed",{status:Z.status}),Error(`Failed to fetch OIDC discovery: ${Z.statusText}`);let $=await Z.json();return Y.debug("Fetched OIDC discovery document",{authEndpoint:$.authorization_endpoint,tokenEndpoint:$.token_endpoint}),S.set(z,$),$}function Oz(z){switch(z){case"production":return"https://cognito-idp.us-east-1.amazonaws.com/us-east-1_3uhuoRM3R";case"staging":return"https://cognito-idp.us-east-1.amazonaws.com/us-east-1_5EUwTP9XD";case"local":throw Error("Local environment is not yet supported for OIDC")}}async function Tz(z){let K=await I(z.issuer),Q=new URL(K.authorization_endpoint);return Q.searchParams.set("response_type","code"),Q.searchParams.set("client_id",z.clientId),Q.searchParams.set("redirect_uri",z.redirectUri),Q.searchParams.set("scope","openid profile email"),Q.searchParams.set("state",z.state),Q.toString()}async function Vz(z){let K=await I(z.issuer),Q=await fetch(K.token_endpoint,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:new URLSearchParams({grant_type:"authorization_code",client_id:z.clientId,client_secret:z.clientSecret,code:z.code,redirect_uri:z.redirectUri})});if(!Q.ok){let $=await Q.text();throw Y.error("Token exchange failed",{status:Q.status,body:$}),Error(`Token exchange failed: ${Q.status} ${$}`)}let Z=await Q.json();return Y.debug("Received tokens from IdP",{expiresIn:Z.expires_in}),Z}async function Cz(z){let K=await I(z.issuer),Q=await fetch(K.userinfo_endpoint,{headers:{Authorization:`Bearer ${z.accessToken}`}});if(!Q.ok)throw Error(`UserInfo request failed: ${Q.statusText}`);return Q.json()}
2
- export{Mz as t,E as u,Rz as v,Oz as w,Tz as x,Vz as y,Cz as z};