@timeback/sdk 0.2.1 → 0.2.2-beta.20260320221119

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.
Files changed (2) hide show
  1. package/dist/index.js +1 -1
  2. package/package.json +1 -1
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- import"./chunk-ae6bkfs5.js";import{a as E,b as B$,c as _$}from"./chunk-5qkmnsbm.js";import{e as P$,f as T$,g as R$,h as N$,i as I$,j as b$,k as OJ}from"./chunk-6nybr8hs.js";import{l as r0}from"./server/adapters/native.js";import{m as d,n as B2,o as _2,p as G2,q as n2,r as S2,s as _J}from"./chunk-bh0ewh45.js";import{t as GJ,u as R,v as M2,w as C$,y as A$,z as z$}from"./chunk-nw6brg34.js";import"./chunk-3pf3cpfk.js";import{G as g8,H as U$,I as E2,J as u,K as BJ,L as P,M,N as J8,Q as n0}from"./chunk-dbcsf5tk.js";import{R as L8,U as j8}from"./chunk-1s709w5e.js";import{TimebackClient as i0}from"@timeback/core";import{getServiceUrlsForEnv as M0}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 K9 extends s{name="ValidationError";constructor($="Validation Error",J){super($,422,J)}}class H9 extends s{name="InputValidationError";issues;constructor($,J){let Z={imsx_codeMajor:"failure",imsx_severity:"error",imsx_description:$,imsx_error_details:J.map((K)=>({path:K.path,message:K.message}))};super($,400,Z);this.issues=J}}function R2($,J){return new H9($,J)}function DJ($){if(!($ instanceof Error))return!1;return"statusCode"in $&&"response"in $}var wJ="BEYOND_AI",X9="LEARNWITH_AI",W9=wJ,UJ={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"},MJ={staging:"https://api.staging.alpha-1edtech.ai",production:"https://api.alpha-1edtech.ai"},LJ={staging:"https://caliper.staging.alpha-1edtech.ai",production:"https://caliper.alpha-1edtech.ai"},jJ={staging:"https://qti.alpha-1edtech.ai/api",production:"https://qti.alpha-1edtech.ai/api"},VJ={staging:"https://platform.dev.timeback.com/auth/1.0/token",production:"https://platform.timeback.com/auth/1.0/token"},kJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},qJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},CJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},AJ={BEYOND_AI:{token:UJ,tokenScope:void 0,api:MJ,caliper:LJ,qti:jJ},LEARNWITH_AI:{token:VJ,tokenScope:"https://purl.imsglobal.org/spec/caliper/v1p2/scope/events.write",api:kJ,caliper:qJ,qti:CJ}},zJ={tsc:"tsc",nativePreview:"@typescript/native-preview"},J1={package:zJ.nativePreview,bin:"tsgo"},e=null,V8=!1,k8=!1;function o8($){let Z=$.replace(/[.+?^${}()|[\]\\]/g,"\\$&").replace(/\*/g,".*");return new RegExp(`^${Z}$`)}function PJ(){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((K)=>K.trim()).filter(Boolean);for(let K of J)if(K.startsWith("-"))e.push({regex:o8(K.slice(1)),exclude:!0});else e.push({regex:o8(K),exclude:!1});if(!e.some((K)=>!K.exclude)&&e.length>0)V8=!0}function TJ($){if(PJ(),!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 RJ(){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 NJ($){switch($){case"debug":return $2.blue;case"info":return $2.cyan;case"warn":return $2.yellow;case"error":return $2.red}}function IJ($){switch($){case"debug":return console.debug;case"info":return console.info;case"warn":return console.warn;case"error":return console.error}}function bJ($){if(q8)return q8($,{depth:null,colors:!0,breakLength:80,compact:!1});return JSON.stringify($,null,2)}var hJ=($)=>{let J=NJ($.level),Z=IJ($.level),K=$.level.toUpperCase().padEnd(5),H=$.timestamp.toISOString().replace(/\.\d{3}Z$/,""),W=`${$2.dim}[${H}]${$2.reset}`,F=`${J}${K}${$2.reset}`,_=$.scope?`${$2.bold}[${$.scope}]${$2.reset} `:"",O=`${W} ${F} ${_}${$.message}`;if($.context&&Object.keys($.context).length>0)Z(O,bJ($.context));else Z(O)},SJ={debug:"[DEBUG]",info:"[INFO]",warn:"[WARN]",error:"[ERROR]"};function EJ($){return Object.entries($).map(([J,Z])=>`${J}=${vJ(Z)}`).join(" ")}function vJ($){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 yJ=($)=>{let J=[];if(J.push($.timestamp.toISOString()),J.push(SJ[$.level]),$.scope)J.push(`[${$.scope}]`);if(J.push($.message),$.context&&Object.keys($.context).length>0)J.push(EJ($.context));console.log(J.join(" "))},fJ=($)=>{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))},gJ={debug:"color: gray",info:"color: #0ea5e9",warn:"color: #f59e0b",error:"color: #ef4444; font-weight: bold"},dJ={debug:"log",info:"info",warn:"warn",error:"error"},mJ=($)=>{let J=dJ[$.level],Z=gJ[$.level],H=`%c${$.scope?`[${$.scope}]`:""} ${$.message}`;if($.context&&Object.keys($.context).length>0)console[J](H,Z,$.context);else console[J](H,Z)},n8=["debug","info","warn","error"];function lJ($){switch($){case"terminal":return hJ;case"ci":return yJ;case"production":return fJ;case"browser":return mJ;case"test":return()=>{}}}function uJ(){if(typeof process<"u"&&process.env?.DEBUG)return"debug";return"info"}function cJ($,J){return n8.indexOf($)>=n8.indexOf(J)}class d2{scope;minLevel;environment;formatter;defaultContext;constructor($={}){this.scope=$.scope,this.minLevel=$.minLevel??uJ(),this.defaultContext=$.defaultContext??{},this.environment=$.environment??RJ(),this.formatter=lJ(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"&&!TJ(this.scope))return;if(!cJ($,this.minLevel))return;let K={level:$,message:J,scope:this.scope,context:Z||Object.keys(this.defaultContext).length>0?{...this.defaultContext,...Z}:void 0,timestamp:new Date};this.formatter(K)}}function Q9($={}){return new d2($)}function pJ(){try{let $=typeof process>"u"?void 0:process.env.DEBUG;return $==="1"||$==="true"}catch{return!1}}var j2=Q9({scope:"auth",minLevel:pJ()?"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 j2.debug("Using cached token"),this.accessToken;if(this.pendingRequest)return j2.debug("Waiting for in-flight token request"),this.pendingRequest;this.pendingRequest=this.fetchToken();try{return await this.pendingRequest}finally{this.pendingRequest=null}}async fetchToken(){j2.debug("Fetching new access token...");let{clientId:$,clientSecret:J}=this.config.credentials,Z=btoa(`${$}:${J}`),K=performance.now(),H=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()-K);if(!H.ok)throw j2.error(`Token request failed: ${H.status} ${H.statusText}`),Error(`Failed to obtain access token: ${H.status} ${H.statusText}`);let F=await H.json();return this.accessToken=F.access_token,this.tokenExpiry=Date.now()+(F.expires_in-60)*1000,j2.debug(`Token acquired (${W}ms, expires in ${F.expires_in}s)`),this.accessToken}invalidate(){j2.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"}},aJ={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:aJ};function iJ($){return"env"in $&&!("baseUrl"in $)&&!("services"in $)}function sJ($){return"baseUrl"in $&&!("services"in $)}function oJ($){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 N2{platform;env;auth;timeout;endpoints;authUrl;tokenScope;pathProfiles;tokenManagers=new Map;constructor($){if(this.timeout=$.timeout??30000,iJ($)){this.auth=$.auth;let J=$.platform??W9,Z=$.env;this.platform=J,this.env=Z;let K=AJ[J];if(!K)throw Error(`Unknown platform: ${J}`);this.authUrl=K.token[Z],this.tokenScope=K.tokenScope??void 0,this.pathProfiles=G9[J]??l2,this.endpoints={oneroster:{baseUrl:K.api[Z],authUrl:this.authUrl},edubridge:{baseUrl:K.api[Z],authUrl:this.authUrl},powerpath:{baseUrl:K.api[Z],authUrl:this.authUrl},clr:{baseUrl:K.api[Z],authUrl:this.authUrl},case:{baseUrl:K.api[Z],authUrl:this.authUrl},caliper:{baseUrl:K.caliper[Z],authUrl:this.authUrl},webhooks:{baseUrl:K.caliper[Z],authUrl:this.authUrl},reporting:{baseUrl:K.api[Z],authUrl:this.authUrl},qti:{baseUrl:K.qti[Z],authUrl:this.authUrl}}}else if(sJ($))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(oJ($)){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 K=this.tokenManagers.get(Z);if(!K)K=new m2({tokenUrl:Z,credentials:{clientId:this.auth.clientId,clientSecret:this.auth.clientSecret},scope:this.tokenScope}),this.tokenManagers.set(Z,K);return K}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 F2(){let $=b("DEBUG");return $==="1"||$==="true"}function nJ(){return Math.random().toString(16).slice(2,10)}function C8($){return Q9({scope:$,minLevel:F2()?"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 rJ($){if(typeof $==="string")return $;if($.length===0)throw Error(`Missing env var key: ${$}`);return $[0]}function O2($){return rJ($)}function B9($){if($!=="staging"&&$!=="production")throw Error(`Invalid env "${$}": must be "staging" or "production"`);return $}function _9($,J){let Z=$?.clientId??b(J.clientId),K=$?.clientSecret??b(J.clientSecret);if(!Z)throw Error(`Missing clientId: provide in config or set ${O2(J.clientId)}`);if(!K)throw Error(`Missing clientSecret: provide in config or set ${O2(J.clientSecret)}`);return{clientId:Z,clientSecret:K}}function tJ($,J,Z){if(!Z)return;return _9($,J)}function eJ($){let J=b($.baseUrl),Z=b($.authUrl),K=b($.clientId),H=b($.clientSecret);if(J===void 0&&K===void 0)return`Missing env: provide in config or set ${O2($.env??$.baseUrl)}`;let W=[];if(J===void 0)W.push(O2($.env??$.baseUrl));if(J!==void 0&&Z===void 0)W.push(O2($.authUrl));if(K===void 0)W.push(O2($.clientId));if(H===void 0)W.push(O2($.clientSecret));return`Missing environment variables: ${W.join(", ")}`}var $Z={name:"transport",matches($){return"transport"in $&&!!$.transport},resolve($){return{mode:"transport",transport:$.transport}}},JZ={name:"provider",matches($){return"provider"in $&&!!$.provider},resolve($){return{mode:"provider",provider:$.provider}}},ZZ={name:"env-config",matches($){return"env"in $},resolve($,J,Z){let K=$,H=B9(K.env),W=_9(K.auth,J),F=K.platform??Z.defaultPlatform,_=Z.templates[F];if(!_){let w=Object.keys(Z.templates).join(", ");throw Error(`Unknown platform "${F}": available platforms are ${w}`)}let O=_[H];if(!O){let w=Object.keys(_).join(", ");throw Error(`Unknown env "${H}" for platform "${F}": available environments are ${w}`)}return{mode:"provider",provider:new N2({platform:O.platform,env:O.env,auth:W,timeout:K.timeout})}}},KZ={name:"explicit-config",matches($){return"baseUrl"in $},resolve($,J){let Z=$,K=Z.authUrl??Z.auth?.authUrl,H=!!K,W=tJ(Z.auth,J,H);return{mode:"provider",provider:new N2({baseUrl:Z.baseUrl,authUrl:K,auth:W,timeout:Z.timeout,pathProfile:Z.pathProfile,paths:Z.paths})}}},HZ={name:"env-fallback-platform",matches($,J){if(Object.keys($).length>0)return!1;let Z=J.env?b(J.env):void 0,K=b(J.clientId),H=b(J.clientSecret);return Z!==void 0&&K!==void 0&&H!==void 0},resolve($,J,Z){let K=B9(b(J.env)),H=b(J.clientId),W=b(J.clientSecret),F=Z.defaultPlatform,_=Z.templates[F]?.[K];if(!_)throw Error(`Unknown env "${K}" for platform "${F}"`);return{mode:"provider",provider:new N2({platform:_.platform,env:_.env,auth:{clientId:H,clientSecret:W}})}}},XZ={name:"env-fallback-explicit",matches($,J){if(Object.keys($).length>0)return!1;let Z=b(J.baseUrl),K=b(J.authUrl),H=b(J.clientId),W=b(J.clientSecret);return Z!==void 0&&K!==void 0&&H!==void 0&&W!==void 0},resolve($,J){let Z=b(J.baseUrl),K=b(J.authUrl),H=b(J.clientId),W=b(J.clientSecret);return{mode:"provider",provider:new N2({baseUrl:Z,authUrl:K,auth:{clientId:H,clientSecret:W}})}}},WZ={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.")}},YZ=[$Z,JZ,ZZ,WZ,KZ,HZ,XZ];function F9($,J,Z=u2){for(let K of YZ)if(K.matches($,J))return K.resolve($,J,Z);throw Error(eJ(J))}var M8=3,QZ=[429,503],t8=1000;class A8{config;log;constructor($){let{config:J,logger:Z}=$,K=J.fetch??globalThis.fetch.bind(globalThis),H;if("tokenProvider"in J&&J.tokenProvider)H=J.tokenProvider;else if("auth"in J&&J.auth)H=new m2({tokenUrl:J.auth.authUrl,credentials:{clientId:J.auth.clientId,clientSecret:J.auth.clientSecret},fetch:K});let W=J.baseUrl.endsWith("/")?J.baseUrl:`${J.baseUrl}/`;this.config={baseUrl:W,timeout:J.timeout??30000,fetch:K,tokenProvider:H},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(DJ(Z)&&Z.statusCode===404)return!1;throw Z}}async requestRaw($,J={}){let{method:Z="GET",params:K,body:H,headers:W={}}=J,F=this.buildUrl($,K),_=J.requestId??nJ(),w=Date.now()+this.config.timeout;for(let j=0;j<M8;j++){let A=w-Date.now();if(A<=0)throw this.log.error("Request timeout before attempt",F2()?{requestId:_,path:$}:{path:$}),new s("Request timeout",408);let C=await this.getAccessToken(),q=j===M8-1,z=performance.now();this.log.debug(`→ ${Z} ${F}`,{requestId:_,attempt:j>0?j+1:void 0});let N={"Content-Type":"application/json",Accept:"application/json","X-Request-ID":_,...W};if(C)N.Authorization=`Bearer ${C}`;let I=await this.config.fetch(F,{method:Z,headers:N,body:H?JSON.stringify(H):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:_}),QZ.includes(I.status)&&!q){let y2=I.headers.get("Retry-After"),p=this.parseRetryAfter(y2,j),s8=w-Date.now();if(p>=s8)throw this.log.error("Request timeout during retry backoff",F2()?{requestId:_,path:$}:{path:$}),new s("Request timeout",408);this.log.warn(`Retrying in ${p}ms (attempt ${j+1}/${M8})`,{...F2()&&{requestId:_},status:I.status}),await this.sleep(p);continue}if(!I.ok)return this.handleErrorResponse(I,_);return I}throw this.log.error("Max retries exceeded",F2()?{requestId:_,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):$,K=new URL(Z,this.config.baseUrl);if(J){for(let[H,W]of Object.entries(J))if(W!==void 0)K.searchParams.set(H,String(W))}return K.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 K=J.length>200?J.slice(0,200)+"...":J,H=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:K,error:H}),new s(`Invalid JSON response from ${W}`,$.status,{parseError:H,body:K})}}async handleErrorResponse($,J){let Z,K=await $.text().catch(()=>"");if(K)try{Z=JSON.parse(K)}catch(W){let F=W instanceof Error?W.message:"Unknown parse error";Z={rawBody:K.slice(0,500),parseError:F},this.log.warn("Failed to parse error response as JSON",{...F2()&&{requestId:J},url:$.url,status:$.status,parseError:F,bodyPreview:K.slice(0,200)})}let H=this.extractErrorMessage(Z,$.statusText);if($.status!==404)this.log.error(`Request failed: ${$.status} ${H}`,F2()?{requestId:J}:void 0);switch($.status){case 401:throw this.config.tokenProvider?.invalidate?.(),new $9(H,Z);case 403:throw new J9(H,Z);case 404:throw new Z9(H,Z);case 422:throw new K9(H,Z);default:throw new s(H,$.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 K=Date.parse($);if(!isNaN(K)){let H=K-Date.now();return Math.max(0,H)}return t8*Math.pow(2,J)}}function GZ($){if($ instanceof Date)return $.toISOString();if(typeof $==="boolean")return $?"true":"false";if(typeof $==="number")return String($);return $.replaceAll("'","''")}function t($){let J=GZ($);return typeof $==="string"||$ instanceof Date?`'${J}'`:J}function BZ($,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,K=[];if(Z.ne!==void 0)K.push(`${$}!=${t(Z.ne)}`);if(Z.gt!==void 0)K.push(`${$}>${t(Z.gt)}`);if(Z.gte!==void 0)K.push(`${$}>=${t(Z.gte)}`);if(Z.lt!==void 0)K.push(`${$}<${t(Z.lt)}`);if(Z.lte!==void 0)K.push(`${$}<=${t(Z.lte)}`);if(Z.contains!==void 0)K.push(`${$}~${t(Z.contains)}`);if(Z.in!==void 0&&Z.in.length>0){let H=Z.in.map((F)=>`${$}=${t(F)}`),W=H.join(" OR ");K.push(H.length>1?`(${W})`:W)}if(Z.notIn!==void 0&&Z.notIn.length>0){let H=Z.notIn.map((W)=>`${$}!=${t(W)}`);K.push(H.join(" AND "))}return K}return[]}function _Z($){return"OR"in $&&Array.isArray($.OR)}function O9($){if(_Z($)){let Z=$.OR.map((K)=>O9(K)).filter((K)=>K!==void 0);return Z.length>0?Z.join(" OR "):void 0}let J=[];for(let[Z,K]of Object.entries($))if(K!==void 0)J.push(...BZ(Z,K));return J.length>0?J.join(" AND "):void 0}var e8=100,FZ=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:K,offset:H,limit:W,...F}=this.params,_={...F,filter:Z?O9(Z):void 0,fields:K?.join(","),limit:$};if(this.paginationStyle==="page"){let O=Math.floor(J/$)+1;return{..._,page:O}}return{..._,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,K){if(J===0)return!1;let H=$.hasMore,W=$.total!==void 0&&Z+J<$.total;return H||W||J===K&&$.total===void 0}async*[Symbol.asyncIterator](){let $=this.params.offset??0,J=this.params.limit??e8,Z=!0,K=1,H=0;while(Z){this.log.debug(`Fetching page ${K} (offset: ${$}, limit: ${J})`);let W=await this.fetcher(this.path,{params:this.buildRequestParams(J,$)}),F=this.extractItems(W.data,K);this.log.debug(`Page ${K}: ${F.length} items`);for(let _ of F)if(yield this.transform?this.transform(_):_,H++,this.max!==void 0&&H>=this.max){this.log.debug(`Pagination complete: reached max of ${this.max} items`);return}Z=this.hasMorePages(W,F.length,$,J),$+=F.length,K++}this.log.debug(`Pagination complete: ${H} total items`)}async toArray($={}){let J=$.maxItems??FZ,Z=[];for await(let K 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(K)}return Z}async firstPage(){let $=this.params.limit??e8,J=this.params.offset??0,Z=await this.fetcher(this.path,{params:this.buildRequestParams($,J)}),K=this.extractItems(Z.data,1),H=this.transform?K.map(this.transform):K,W=this.hasMorePages(Z,H.length,J,$);return this.log.debug(`First page: ${H.length} items, total: ${Z.total}, hasMore: ${W}`),{data:H,hasMore:W,total:Z.total,nextOffset:W?J+H.length:void 0}}}function g2($,J){return{path:$,message:J}}function o($,J,Z){let K=$.safeParse(J);if(K.success)return K.data;let H=K.error.issues.map((W)=>({path:W.path.join(".")||"(root)",message:W.message}));throw R2(`Invalid ${Z} data`,H)}function c2($,J){if(typeof $!=="string"||$.trim()==="")throw R2(`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 R2("Invalid list parameters",J)}var x9={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"]},V2="http://purl.imsglobal.org/ctx/caliper/v1p2",D9="QUESTION_RESULT";import{z as L}from"zod/v4";import{z as Y}from"zod/v4";import{z as J2}from"zod/v4";import{z as U}from"zod/v4";import{z as D}from"zod/v4";import{z as V}from"zod/v4";import{z as x}from"zod/v4";import{z as f}from"zod/v4";import{z as G}from"zod/v4";import{z as B}from"zod/v4";import{z as Q}from"zod/v4";import{z as C2}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:K,pagination:H}=Z,F=(J.params?.offset??0)+K.length<H.total;return{data:K,hasMore:F,total:H.total}}}class q9 extends z8{constructor($,J,Z={}){P8(Z);let{max:K,...H}=Z;super({fetcher:(W,F)=>$.requestPaginated(W,F),path:J,params:H,max:K,logger:S})}}function T8($){let J=$.id??`urn:uuid:${crypto.randomUUID()}`,Z=$.metricsId??`urn:uuid:${crypto.randomUUID()}`;return{"@context":V2,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":V2,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 OZ($){return{"@context":V2,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 xZ($){return{"@context":V2,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 DZ($){let{id:J,...Z}=$.generated;return{"@context":V2,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:D9},...$.session===void 0?{}:{session:$.session},...$.extensions===void 0?{}:{extensions:$.extensions}}}var wZ="http://purl.imsglobal.org/ctx/caliper/v1p2",UZ="urn:tag:auto-attach",X2={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"},MZ=new Set(["ActivityEvent","TimeSpentEvent","AssignableEvent","ViewEvent"]);class C9{transformEnvelope($){let{data:J}=$;if(!Array.isArray(J))return $;let Z=J.map((K)=>x2(K)?LZ(K):K);return{...$,data:Z}}}function LZ($){let J={...$};if(!("@context"in J))J["@context"]=wZ;if(J.profile==="TimebackProfile")J.profile="AggregationProfile";if(x2(J.actor)&&J.actor.type==="TimebackUser")J.actor=jZ(J.actor);if(x2(J.object))J.object=VZ(J.object,J.type);if(x2(J.generated)){let Z=J.type;if(Z==="ActivityEvent")J.generated=kZ(J.generated);else if(Z==="TimeSpentEvent")J.generated=qZ(J.generated)}if(!J.session)J.session=UZ;return J}function jZ($){return{type:X2.PERSON,id:$.id}}function VZ($,J){if(!($.type==="TimebackActivityContext"&&J!==void 0&&MZ.has(J)))return{...$};let K={type:X2.ASSIGNABLE_DIGITAL_RESOURCE,id:$.id};if($.name)K.name=$.name;if(x2($.course)){let H={type:X2.COURSE_OFFERING,id:$.course.id};if($.course.name)H.name=$.course.name;K.isPartOf=H}if(x2($.app))K.extensions={app:{name:$.app.name,type:$.app.type??X2.SOFTWARE_APPLICATION}};return K}function A9($,J,Z,K){let W={id:typeof $?.id==="string"?$.id:`urn:uuid:${crypto.randomUUID()}`,type:J};if(Array.isArray($.items))W.items=$.items.map((F)=>{if(!x2(F))return F;let _=typeof F.type==="string"?F.type:"",O=_?_[0].toUpperCase()+_.slice(1):K;return{id:`urn:uuid:${crypto.randomUUID()}`,type:Z,metricType:O,metricValue:F.value}});for(let[F,_]of Object.entries($))if(F!=="id"&&F!=="type"&&F!=="items")W[F]=_;return W}function kZ($){return A9($,X2.ACTIVITY_METRICS_COLLECTION,X2.ACTIVITY_METRIC,"")}function qZ($){return A9($,X2.TIME_SPENT_METRICS_COLLECTION,X2.TIME_SPENT_METRIC,"Active")}function x2($){return typeof $==="object"&&$!==null&&!Array.isArray($)}function CZ($,J=u2){return F9($,x9,J)}var AZ=/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(?:\.\d+)?(?:Z|[+-]\d{2}:\d{2})$/,v=L.string().min(1).regex(AZ,"must be a valid ISO 8601 datetime"),q2=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),Z2=L.enum(["Reading","Language","Vocabulary","Social Studies","Writing","Science","FastMath","Math","None","Other"]).meta({id:"TimebackSubject",description:"Subject area"}),n=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)]).meta({id:"TimebackGrade",description:"Grade level (-1 = Pre-K, 0 = K, 1-12 = grades, 13 = AP)"}),zZ=L.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),PZ=L.enum(["department","school","district","local","state","national"]),z9=L.enum(["administrator","aide","guardian","parent","proctor","relative","student","teacher"]),TZ=L.enum(["administrator","proctor","student","teacher"]),RZ=L.enum(["qti","text","audio","video","interactive","visual","course-material","assessment-bank"]),NZ=L.enum(["qti-test","qti-question","qti-stimulus","qti-test-bank"]),IZ=L.enum(["unit","course","resource-collection"]),bZ=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"]),hZ=L.enum(["easy","medium","hard"]),SZ=L.array(L.object({source:L.string(),learningObjectiveIds:L.array(L.string())})),EZ=L.object({consecutive_failures:L.number().int().min(1).optional(),stagnation_limit:L.number().int().min(1).optional()}).optional(),Q1=L.enum(["powerpath-100","quiz","test-out","placement","unit-test","alpha-read-article"]).nullable(),G1=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()}),vZ=Y.union([Y.url(),Y.string().regex(/^urn:/,"Must be a URL or URN")]),D2=Y.object({id:Y.url(),type:Y.literal("TimebackUser"),email:Y.string()}),N8=Y.object({id:Y.url(),type:Y.literal("TimebackActivityContext"),subject:Z2,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()}),P9=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:v,profile:Y.literal("TimebackProfile"),actor:D2,action:Y.string(),object:N8,edApp:Y.object({id:Y.string(),name:Y.string().optional()}).optional()}),T9=Y.object({type:Y.enum(["totalQuestions","correctQuestions","xpEarned","masteredUnits"]),value:Y.number()}),yZ=Y.object({id:Y.url(),type:Y.literal("TimebackActivityMetricsCollection"),attempt:Y.number().optional(),items:Y.array(T9),extensions:Y.record(Y.string(),Y.unknown()).optional()}),fZ=P9.extend({type:Y.literal("ActivityEvent"),action:Y.literal("Completed"),generated:yZ}),R9=Y.object({type:Y.enum(["active","inactive","waste","unknown","anti-pattern"]),value:Y.number(),subType:Y.string().optional()}),gZ=Y.object({id:Y.url(),type:Y.literal("TimebackTimeSpentMetricsCollection"),items:Y.array(R9)}),dZ=P9.extend({type:Y.literal("TimeSpentEvent"),action:Y.literal("SpentTime"),generated:gZ}),mZ=Y.object({actor:D2,object:N8,metrics:Y.array(T9).min(1,"metrics must contain at least one metric"),eventTime:v.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(),lZ=Y.object({actor:D2,object:N8,metrics:Y.array(R9).min(1,"metrics must contain at least one metric"),eventTime:v.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(),uZ=Y.union([fZ,dZ]),_1=Y.object({sensor:Y.string(),sendTime:v,dataVersion:Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),data:Y.array(uZ)}),cZ=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:vZ,type:Y.string(),name:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),N9=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,D2]),action:Y.string(),object:l,eventTime:v,profile:cZ,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(),w9=Y.object({sensor:X,sendTime:v,dataVersion:Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),data:Y.array(N9).min(1,"data must contain at least one event")}),pZ=Y.object({sensor:X,events:Y.array(N9).min(1,"events must contain at least one event")}),U9=Y.object({limit:Y.number().int().positive().optional(),offset:Y.number().int().min(0).optional(),sensor:X.optional(),startDate:v.optional(),endDate:v.optional(),actorId:X.optional(),actorEmail:Y.email().optional()}).strict(),I9=Y.object({id:Y.string(),name:Y.string().optional(),isPartOf:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),aZ=Y.object({id:Y.string(),attempt:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),startedAtTime:v.optional(),endedAtTime:v.optional(),duration:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),iZ=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()}),sZ=Y.object({actor:Y.union([Y.string(),p2,D2]),object:I9,edApp:l,id:Y.string().optional(),eventTime:v.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),oZ=Y.object({actor:Y.union([Y.string(),p2,D2]),object:I9,edApp:l,generated:aZ.optional(),id:Y.string().optional(),eventTime:v.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),nZ=Y.object({actor:Y.union([Y.string(),p2,D2]),object:Y.string(),generated:iZ,edApp:l,id:Y.string().optional(),eventTime:v.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),O1=J2.object({name:X,targetUrl:J2.url("targetUrl must be a valid URL"),secret:X,active:J2.boolean(),sensor:J2.string().nullable().optional(),description:J2.string().nullable().optional()}).strict(),rZ=J2.enum(["string","number","boolean"]),tZ=J2.enum(["eq","neq","gt","gte","lt","lte","contains","notContains","in","notIn","startsWith","endsWith","regexp"]),x1=J2.object({webhookId:X,filterKey:X,filterValue:X,filterType:rZ,filterOperator:tZ,active:J2.boolean()}).strict(),a2=U.string().uuid(),M9=U.object({title:X,identifier:a2,uri:X}),eZ=U.object({identifier:a2,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()}),$5=U.object({identifier:a2,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()}),J5=U.object({identifier:a2,uri:X,lastChangeDateTime:X,associationType:X,originNodeURI:M9,destinationNodeURI:M9,sequenceNumber:U.number().optional(),extensions:U.unknown().optional()}),Z5=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()}),w1=U.object({CFDocument:eZ,CFItems:U.array($5),CFAssociations:U.array(J5),CFDefinitions:Z5.optional(),extensions:U.unknown().optional()}),L9="https://www.w3.org/ns/credentials/v2",K5=D.array(D.url()).min(3,"@context must include W3C, CLR, and OB context URLs").refine(($)=>$[0]===L9,`First @context entry must be "${L9}"`),b9=D.object({id:D.url(),type:D.literal("Image"),caption:D.string().optional()}),I8=D.object({id:D.url(),type:D.array(D.string()).min(1).refine(($)=>$.includes("Profile"),'type must include "Profile"'),name:D.string().optional(),url:D.string().url().optional(),phone:D.string().optional(),description:D.string().optional(),image:b9.optional(),email:D.string().email().optional()}),h9=D.object({type:X,proofPurpose:X,verificationMethod:X,created:D.iso.datetime(),proofValue:X,cryptosuite:D.string().optional()}),H5=D.object({id:X,type:D.literal("1EdTechJsonSchemaValidator2019")}),X5=D.object({id:D.string().url().optional(),narrative:D.string().optional()}),W5=D.object({id:D.url(),type:D.array(D.string()).min(1).refine(($)=>$.includes("Achievement"),'type must include "Achievement"'),name:X,description:X,criteria:X5,image:b9.optional(),achievementType:D.string().optional(),creator:I8.optional()}),Y5=D.object({type:D.literal("IdentityObject"),identityHash:X,identityType:X,hashed:D.boolean(),salt:D.string().optional()}),Q5=D.enum(["exactMatchOf","isChildOf","isParentOf","isPartOf","isPeerOf","isRelatedTo","precedes","replacedBy"]),G5=D.object({type:D.literal("Association"),associationType:Q5,sourceId:D.url(),targetId:D.url()}),B5=D.object({"@context":D.array(D.string()).min(1),id:D.url(),type:D.array(D.string()).min(1).refine(($)=>$.includes("VerifiableCredential"),'type must include "VerifiableCredential"'),issuer:I8,validFrom:D.iso.datetime(),validUntil:D.string().datetime().optional(),credentialSubject:D.object({id:D.string().optional()}),proof:D.array(h9).optional()}),_5=D.object({id:D.string().url().optional(),type:D.array(D.string()).min(1).refine(($)=>$.includes("ClrSubject"),'type must include "ClrSubject"'),identifier:D.array(Y5).optional(),achievement:D.array(W5).optional(),verifiableCredential:D.array(B5).min(1),association:D.array(G5).optional()}),M1=D.object({"@context":K5,id:D.url(),type:D.array(D.string()).min(1).refine(($)=>$.includes("VerifiableCredential")&&$.includes("ClrCredential"),'type must include "VerifiableCredential" and "ClrCredential"'),issuer:I8,name:X,description:D.string().optional(),validFrom:D.iso.datetime(),validUntil:D.iso.datetime().optional(),credentialSubject:_5,proof:D.array(h9).optional(),credentialSchema:D.array(H5).optional()}),F5=V.object({staging:V.string().meta({description:"Course ID in staging environment"}).optional(),production:V.string().meta({description:"Course ID in production environment"}).optional()}).meta({id:"CourseIds",description:"Environment-specific course IDs (populated by sync)"}),O5=V.enum(["base","hole-filling","optional"]).meta({id:"CourseType",description:"Course classification type"}),x5=V.enum(["draft","testing","published","deactivated"]).meta({id:"PublishStatus",description:"Course publication status"}),D5=V.object({dailyXp:V.number().int().positive().meta({description:"Target XP to earn per day"}).optional(),dailyLessons:V.number().int().positive().meta({description:"Target lessons to complete per day"}).optional(),dailyActiveMinutes:V.number().int().positive().meta({description:"Target active learning minutes per day"}).optional(),dailyAccuracy:V.number().int().min(0).max(100).meta({description:"Target accuracy percentage (0-100)"}).optional(),dailyMasteredUnits:V.number().int().positive().meta({description:"Target units to master per day"}).optional()}).meta({id:"CourseGoals",description:"Daily learning goals for a course"}),w5=V.object({totalXp:V.number().int().positive().meta({description:"Total XP available in the course"}).optional(),totalLessons:V.number().int().positive().meta({description:"Total number of lessons/activities"}).optional(),totalGrades:V.number().int().positive().meta({description:"Total grade levels covered"}).optional()}).meta({id:"CourseMetrics",description:"Aggregate metrics for a course"}),S9=V.object({courseType:O5.optional(),isSupplemental:V.boolean().meta({description:"Whether this is supplemental to a base course"}).optional(),isCustom:V.boolean().meta({description:"Whether this is a custom course for an individual student"}).optional(),publishStatus:x5.optional(),contactEmail:V.email().meta({description:"Contact email for course issues"}).optional(),primaryApp:V.string().meta({description:"Primary application identifier"}).optional(),goals:D5.optional(),metrics:w5.optional()}).meta({id:"CourseMetadata",description:"Course metadata (matches API metadata object)"}),E9=V.object({courseCode:V.string().meta({description:"Course code (e.g., 'MATH101')"}).optional(),level:V.string().meta({description:"Course level (e.g., 'AP', 'Honors')"}).optional(),metadata:S9.optional()}).meta({id:"CourseDefaults",description:"Default properties that apply to all courses unless overridden"}),j9=V.object({level:V.string().meta({description:"Course level for this environment"}).optional(),sensor:V.url().meta({description:"Caliper sensor endpoint URL for this environment"}).optional(),launchUrl:V.url().meta({description:"LTI launch URL for this environment"}).optional(),metadata:S9.optional()}).meta({id:"CourseEnvOverrides",description:"Environment-specific course overrides (non-identity fields)"}),U5=V.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"}),M5=E9.extend({subject:Z2.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:F5.nullable().optional(),sensor:V.url().meta({description:"Caliper sensor endpoint URL for this course"}).optional(),launchUrl:V.url().meta({description:"LTI launch URL for this course"}).optional(),overrides:U5.optional()}).meta({id:"CourseConfig",description:"Configuration for a single course. Must have either grade or courseCode (or both)."}),j1=V.object({$schema:V.string().meta({description:"JSON Schema reference for editor support"}).optional(),name:V.string().min(1,"App name is required").meta({description:"Display name for your app"}),defaults:E9.meta({description:"Default properties applied to all courses"}).optional(),courses:V.array(M5).min(1,"At least one course is required").meta({description:"Courses available in this app"}),sensor:V.url().meta({description:"Default Caliper sensor endpoint URL for all courses"}).optional(),launchUrl:V.url().meta({description:"Default LTI launch URL for all courses"}).optional(),studio:V.object({telemetry:V.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((K)=>K.grade!==void 0).map((K)=>`${K.subject}:${K.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((K)=>K.courseCode!==void 0).map((K)=>K.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"]}),I2=x.union([v,q2]),b2=I2.transform(($)=>$.includes("T")?$:`${$}T00:00:00.000Z`),k1=x.object({id:x.string(),role:x.string(),beginDate:I2.nullable(),endDate:I2.nullable(),metadata:x.object({goals:x.object({dailyXp:x.number().optional()}).optional(),metrics:x.object({totalXp:x.number().optional(),totalLessons:x.number().optional()}).optional()}).optional(),course:x.object({id:x.string(),title:x.string(),subjects:x.array(x.string()).nullable(),grades:x.array(x.string()).nullable()}),school:x.object({id:x.string(),name:x.string()}),testOutSupported:x.boolean(),testOutEligible:x.boolean()}),L5=x.object({activityMetrics:x.object({xpEarned:x.number(),totalQuestions:x.number(),correctQuestions:x.number(),masteredUnits:x.number()}),timeSpentMetrics:x.object({activeSeconds:x.number(),inactiveSeconds:x.number(),wasteSeconds:x.number()}),apps:x.array(x.string())}),j5=x.record(x.string(),x.record(x.string(),L5)),q1=x.object({message:x.string(),enrollmentId:x.string(),startDate:I2,endDate:I2,facts:j5,factsByApp:x.unknown()}),v9=x.object({email:x.email().optional(),studentId:X.optional()}).superRefine(($,J)=>{if(!$.email&&!$.studentId)J.addIssue({code:x.ZodIssueCode.custom,message:"must provide either email or studentId",path:["email"]}),J.addIssue({code:x.ZodIssueCode.custom,message:"must provide either email or studentId",path:["studentId"]})}),C1=x.object({subject:X,grade:X,courseId:X,orgSourcedId:X.optional()}),A1=x.object({userId:X}),z1=x.object({sourcedId:X.optional(),role:TZ.optional(),beginDate:v.optional(),metadata:x.record(x.string(),x.unknown()).optional()}),P1=x.object({fields:x.string().optional(),limit:x.number().int().positive().optional(),offset:x.number().int().nonnegative().optional(),sort:x.string().optional(),orderBy:x.enum(["asc","desc"]).optional(),filter:x.string().optional(),search:x.string().optional(),roles:x.array(z9).min(1),orgSourcedIds:x.array(X).optional()}),T1=v9.extend({startDate:b2,endDate:b2,timezone:x.string().optional()}),R1=v9.extend({weekDate:b2,timezone:x.string().optional()}),N1=x.object({enrollmentId:X,startDate:b2.optional(),endDate:b2.optional(),timezone:x.string().optional()}),b1=f.object({email:f.email()}),h1=f.object({name:X.optional(),timeback_id:X.optional(),grade:X.optional(),subject:X.optional(),all:f.boolean().optional()}),S1=f.object({student_email:f.email(),timeback_id:X.optional(),subject:X.optional(),grade_rank:f.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,K=!!$.subject,H=$.grade_rank!==void 0;if(!Z&&!K)J.addIssue({code:f.ZodIssueCode.custom,message:"must provide either timeback_id or subject",path:["timeback_id"]}),J.addIssue({code:f.ZodIssueCode.custom,message:"must provide either timeback_id or subject",path:["subject"]});if(H&&!K)J.addIssue({code:f.ZodIssueCode.custom,message:"grade_rank requires subject",path:["grade_rank"]});let W=!!$.assessment_line_item_sourced_id,F=!!$.assessment_result_sourced_id;if(W!==F)J.addIssue({code:f.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"]})}),E1=f.object({student_email:f.email(),assignment_id:f.number().int().positive().optional(),timeback_id:X.optional(),subject:X.optional(),grade_rank:f.number().int().min(0).max(12).optional()}).superRefine(($,J)=>{let Z=$.assignment_id!==void 0,K=!!$.timeback_id,H=!!$.subject&&$.grade_rank!==void 0;if(!Z&&!K&&!H)J.addIssue({code:f.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:f.ZodIssueCode.custom,message:"grade_rank requires subject",path:["grade_rank"]})}),g=G.enum(["active","tobedeleted"]),y=G.record(G.string(),G.unknown()).nullable().optional(),k=G.object({sourcedId:X,type:G.string().optional(),href:G.string().optional()}).strict(),W2=G.union([q2,v]).transform(($)=>$.includes("T")?$:`${$}T00:00:00Z`),V5=G.object({roleType:G.enum(["primary","secondary"]),role:z9,org:k,userProfile:G.string().optional(),metadata:y,beginDate:W2.optional(),endDate:W2.optional()}).strict(),y1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:g.optional(),enabledUser:G.boolean(),givenName:X.describe("givenName must be a non-empty string"),familyName:X.describe("familyName must be a non-empty string"),middleName:X.optional(),username:X.optional(),email:G.email(),roles:G.array(V5).min(1,"roles must include at least one role"),userIds:G.array(G.object({type:X,identifier:X}).strict()).optional(),agents:G.array(k).optional(),grades:G.array(n).optional(),identifier:X.optional(),sms:X.optional(),phone:X.optional(),pronouns:X.optional(),password:X.optional(),metadata:y}).strict(),f1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:g.optional(),title:X.describe("title must be a non-empty string"),org:k,courseCode:X.optional(),subjects:G.array(Z2).optional(),grades:G.array(n).optional(),level:X.optional(),metadata:y}).strict(),g1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:g.optional(),title:X.describe("title must be a non-empty string"),terms:G.array(k).min(1,"terms must have at least one item"),course:k,org:k,classCode:X.optional(),classType:G.enum(["homeroom","scheduled"]).optional(),location:X.optional(),grades:G.array(n).optional(),subjects:G.array(Z2).optional(),subjectCodes:G.array(X).optional(),periods:G.array(X).optional(),metadata:y}).strict(),y9=G.enum(["true","false"]),d1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:g.optional(),user:k,class:k,school:k.optional(),role:G.enum(["administrator","proctor","student","teacher"]),primary:y9.optional(),beginDate:q2.optional(),endDate:q2.optional(),metadata:y}).strict(),m1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),status:g,weight:G.number().nullable().optional(),metadata:y}).strict(),l1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),class:k,school:k,category:k,assignDate:W2,dueDate:W2,status:g,description:G.string().optional(),resultValueMin:G.number().nullable().optional(),resultValueMax:G.number().nullable().optional(),scoreScale:k.optional(),metadata:y}).strict(),u1=G.object({sourcedId:X.optional(),lineItem:k,student:k,class:k.optional(),scoreDate:W2,scoreStatus:G.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),score:G.number().nullable().optional(),textScore:G.string().nullable().optional(),status:g,comment:G.string().nullable().optional(),metadata:y}).strict(),c1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),status:g.optional(),type:G.string(),class:k,course:k.nullable().optional(),scoreScaleValue:G.array(G.object({itemValueLHS:X,itemValueRHS:X,value:G.string().optional(),description:G.string().optional()}).strict()),minScore:G.number().optional(),maxScore:G.number().optional(),metadata:y}).strict(),p1=G.object({sourcedId:X.optional(),status:g.optional(),dateLastModified:v.optional(),title:X.describe("title must be a non-empty string"),description:G.string().nullable().optional(),class:k.nullable().optional(),parentAssessmentLineItem:k.nullable().optional(),scoreScale:k.nullable().optional(),resultValueMin:G.number().nullable().optional(),resultValueMax:G.number().nullable().optional(),component:k.nullable().optional(),componentResource:k.nullable().optional(),learningObjectiveSet:G.array(G.object({source:G.string(),learningObjectiveIds:G.array(G.string())})).optional().nullable(),course:k.nullable().optional(),metadata:y}).strict(),k5=G.object({learningObjectiveId:G.string(),score:G.number().optional(),textScore:G.string().optional()}),q5=G.array(G.object({source:G.string(),learningObjectiveResults:G.array(k5)})),a1=G.object({sourcedId:X.optional(),status:g.optional(),dateLastModified:v.optional(),metadata:y,assessmentLineItem:k,student:k,score:G.number().nullable().optional(),textScore:G.string().nullable().optional(),scoreDate:W2,scoreScale:k.nullable().optional(),scorePercentile:G.number().nullable().optional(),scoreStatus:G.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),comment:G.string().nullable().optional(),learningObjectiveSet:q5.nullable().optional(),inProgress:G.string().nullable().optional(),incomplete:G.string().nullable().optional(),late:G.string().nullable().optional(),missing:G.string().nullable().optional()}).strict(),i1=G.object({sourcedId:X.optional(),status:g.optional(),name:X.describe("name must be a non-empty string"),type:PZ,identifier:X.optional(),parent:k.optional(),metadata:y}).strict(),s1=G.object({sourcedId:X.optional(),status:g.optional(),name:X.describe("name must be a non-empty string"),type:G.literal("school").optional(),identifier:X.optional(),parent:k.optional(),metadata:y}).strict(),o1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:g,title:X.describe("title must be a non-empty string"),startDate:W2,endDate:W2,type:G.enum(["gradingPeriod","semester","schoolYear","term"]),schoolYear:X.describe("schoolYear must be a non-empty string"),org:k,parent:k.optional(),children:G.array(k).optional(),metadata:y}).strict(),n1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),courseComponent:k,resource:k,status:g,metadata:y}).strict(),r1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),course:k,status:g,metadata:y}).strict(),t1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string"),role:G.enum(["student","teacher"]),primary:y9.optional(),beginDate:q2.optional(),endDate:q2.optional(),metadata:y}).strict(),e1=G.object({agentSourcedId:X.describe("agentSourcedId must be a non-empty string")}).strict(),$6=G.object({applicationName:X.describe("applicationName must be a non-empty string"),credentials:G.object({username:X.describe("username must be a non-empty string"),password:X.describe("password must be a non-empty string")})}).strict(),J6=G.object({sourcedId:X.describe("sourcedId must be a non-empty string")}).loose(),Y2=G.object({type:RZ,subject:Z2.nullish(),grades:G.array(n).nullish(),language:G.string().nullish(),xp:G.number().nullish(),url:G.url().nullish(),keywords:G.array(G.string()).nullish(),learningObjectiveSet:SZ.nullish(),lessonType:G.string().nullish()}).passthrough(),C5=Y2.extend({type:G.literal("qti"),subType:NZ,questionType:bZ.optional(),difficulty:hZ.optional()}),A5=Y2.extend({type:G.literal("text"),format:G.string(),author:G.string().optional(),pageCount:G.number().optional()}),z5=Y2.extend({type:G.literal("audio"),duration:G.string().regex(/^\d{2}:\d{2}:\d{2}(\.\d{2})?$/).optional(),format:G.string(),speaker:G.string().optional()}),P5=Y2.extend({type:G.literal("video"),duration:G.string().regex(/^\d{2}:\d{2}:\d{2}(\.\d{2})?$/).optional(),captionsAvailable:G.boolean().optional(),format:G.string()}),T5=Y2.extend({type:G.literal("interactive"),launchUrl:G.url().optional(),toolProvider:G.string().optional(),instructionalMethod:G.string().optional(),courseIdOnFail:G.string().nullable().optional(),fail_fast:EZ}),R5=Y2.extend({type:G.literal("visual"),format:G.string(),resolution:G.string().optional()}),N5=Y2.extend({type:G.literal("course-material"),subType:IZ,author:G.string().optional(),format:G.string(),instructionalMethod:G.string().optional()}),I5=Y2.extend({type:G.literal("assessment-bank"),resources:G.array(G.string())}),b5=G.discriminatedUnion("type",[C5,A5,z5,P5,T5,R5,N5,I5]),Z6=G.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:G.array(G.enum(["primary","secondary"])).optional(),importance:G.enum(["primary","secondary"]).optional(),vendorId:G.string().optional(),applicationId:G.string().optional(),status:g.optional(),metadata:b5.nullable().optional()}).strict(),h5=G.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:G.number()}).loose(),K6=G.object({course:G.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:k,status:g,metadata:y}).strict(),courseStructure:G.record(G.string(),h5)}).strict(),S5=G.object({student:k}).loose(),H6=G.array(S5).min(1,"results must have at least one item"),E5=B.enum(["edulastic","mastery-track"]),v5=B.enum(["powerpath-100","quiz","test-out","placement","unit-test","alpha-read-article"]),f9=B.array(n),g9=B.record(B.string(),B.unknown()).optional(),d9=B.object({courseId:X,lessonTitle:X.optional(),launchUrl:X.optional(),toolProvider:E5,unitTitle:X.optional(),courseComponentSourcedId:X.optional(),vendorId:X.optional(),description:X.optional(),resourceMetadata:g9.nullable().optional(),grades:f9}),W6=d9.extend({lessonType:B.literal("test-out"),xp:B.number()}),Y6=d9.extend({lessonType:B.literal("placement"),courseIdOnFail:X.nullable().optional(),xp:B.number().optional()}),V9=B.object({courseId:X,lessonType:v5,lessonTitle:X.optional(),unitTitle:X.optional(),courseComponentSourcedId:X.optional(),resourceMetadata:g9.nullable().optional(),xp:B.number().optional(),grades:f9.optional(),courseIdOnFail:X.nullable().optional()}),Q6=B.discriminatedUnion("testType",[V9.extend({testType:B.literal("qti"),qti:B.object({url:B.url(),title:X.optional(),metadata:B.record(B.string(),B.unknown()).optional()})}),V9.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)})})]),G6=B.object({student:X,lesson:X}),B6=B.object({student:X,lesson:X}),_6=B.object({courseId:X,userId:X,classId:X.optional()}),k2=B.object({type:B.enum(["component","resource"]),id:X}),y5=B.union([B.object({type:B.literal("set-skipped"),payload:B.object({target:k2,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:k2,reference_id:X})}),B.object({type:B.literal("move-item-after"),payload:B.object({target:k2,reference_id:X})}),B.object({type:B.literal("move-item-to-start"),payload:B.object({target:k2})}),B.object({type:B.literal("move-item-to-end"),payload:B.object({target:k2})}),B.object({type:B.literal("change-item-parent"),payload:B.object({target:k2,new_parent_id:X,position:B.enum(["start","end"]).optional()})})]),F6=B.object({operation:y5,reason:X.optional()}),O6=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:zZ,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()})}),x6=B.object({student:X,lesson:X,applicationName:X.optional(),testId:X.optional(),skipCourseEnrollment:B.boolean().optional()}),D6=B.object({student:X,subject:Z2}),w6=B.object({student:X,lesson:X}),U6=B.object({userId:X}),M6=B.object({userId:X,subject:B.enum(["Math","Reading","Language","Science"])}),L6=B.object({student:X,subject:Z2,grade:n,testName:X.optional()}),j6=B.object({testName:X}),f5=B.object({student:X,subject:Z2,grade:n,testName:X.optional()}),V6=B.object({items:B.array(f5)}),k6=B.object({spreadsheetUrl:B.url(),sheet:X}),q6=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()}),C6=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()}),A6=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 $}),z6=B.object({student:X,lesson:X,attempt:B.number().int().positive().optional()}),P6=B.object({student:X,lesson:X}),T6=B.object({student:X,lesson:X}),R6=B.object({student:X,course:X}),N6=B.object({student:X,lesson:X,applicationName:X.optional()}),I6=B.object({student:X,subject:Z2}),b6=B.object({oneRosterSourcedId:X,subject:X}),h6=B.object({courseIds:B.array(X).min(1),rendererId:X,rendererUrl:B.url(),rendererVersion:X.optional(),suppressFeedback:B.boolean().optional(),suppressCorrectResponse:B.boolean().optional()}),S6=B.object({status:B.enum(["active","tobedeleted"]).optional()}),m9=Q.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=Q.enum(["single","multiple","ordered","record"]),h8=Q.enum(["identifier","boolean","integer","float","string","point","pair","directedPair","duration","file","uri"]),g5=Q.enum(["easy","medium","hard"]),d5=Q.enum(["linear","nonlinear"]),m5=Q.enum(["individual","simultaneous"]),S8=Q.enum(["show","hide"]),l9=Q.enum(["test","item","stimulus"]),v6=Q.enum(["QUESTION","LESSON"]),l5=Q.object({value:Q.array(Q.string())}).strict(),u9=Q.object({identifier:X,cardinality:b8,baseType:h8.optional(),correctResponse:l5}).strict(),c9=Q.object({identifier:X,cardinality:b8,baseType:h8.optional()}).strict(),p9=Q.object({identifier:X,cardinality:b8.optional(),baseType:h8,normalMaximum:Q.number().optional(),normalMinimum:Q.number().optional(),defaultValue:Q.object({value:Q.unknown().optional()}).strict().optional()}).strict(),u5=Q.object({outcomeIdentifier:X,variableIdentifier:X}).strict(),a9=Q.object({templateType:Q.enum(["match_correct","map_response"]),responseDeclarationIdentifier:X,outcomeIdentifier:X,correctResponseIdentifier:X,incorrectResponseIdentifier:X,inlineFeedback:u5.optional()}).strict(),c5=Q.object({source:X,learningObjectiveIds:Q.array(Q.string())}).strict(),E8=Q.object({subject:Q.string().optional(),grade:n.optional(),difficulty:g5.optional(),learningObjectiveSet:Q.array(c5).optional()}).loose(),i9=Q.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:Q.string(),title:Q.string()}).strict(),s9=Q.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:Q.string(),class:Q.array(Q.string())}).strict(),o9=Q.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:Q.string(),class:Q.array(Q.string())}).strict(),n9=Q.object({href:X,type:X}).strict(),r9=Q.object({id:X,support:Q.string(),content:Q.string()}).strict(),y6=Q.object({page:Q.number().int().positive().optional(),limit:Q.number().int().positive().optional(),sort:Q.string().optional(),order:Q.enum(["asc","desc"]).optional()}).strict(),p5=Q.object({format:Q.string().pipe(Q.literal("xml")),xml:X,metadata:E8.optional()}).strict(),a5=Q.object({identifier:X,title:X,type:m9,qtiVersion:Q.string().optional(),timeDependent:Q.boolean().optional(),adaptive:Q.boolean().optional(),responseDeclarations:Q.array(u9).optional(),outcomeDeclarations:Q.array(c9).optional(),responseProcessing:a9.optional(),metadata:E8.optional(),modalFeedback:Q.array(i9).optional(),feedbackInline:Q.array(s9).optional(),feedbackBlock:Q.array(o9).optional()}).strict(),f6=Q.union([p5,a5]),g6=Q.object({identifier:X.optional(),title:X,type:m9,qtiVersion:Q.string().optional(),timeDependent:Q.boolean().optional(),adaptive:Q.boolean().optional(),responseDeclarations:Q.array(u9).optional(),outcomeDeclarations:Q.array(c9).optional(),responseProcessing:a9.optional(),metadata:E8.optional(),modalFeedback:Q.array(i9).optional(),feedbackInline:Q.array(s9).optional(),feedbackBlock:Q.array(o9).optional(),rawXml:Q.string(),content:Q.record(Q.string(),Q.unknown())}).strict(),d6=Q.object({identifier:X,response:Q.union([Q.string(),Q.array(Q.string())])}).strict(),t9=Q.object({identifier:X,href:X,sequence:Q.number().int().positive().optional()}).strict(),i5=Q.object({identifier:X,title:X,visible:Q.boolean(),required:Q.boolean().optional(),fixed:Q.boolean().optional(),sequence:Q.number().int().positive(),"qti-assessment-item-ref":Q.array(t9).optional()}).strict(),e9=Q.object({identifier:X,navigationMode:Q.string().pipe(d5),submissionMode:Q.string().pipe(m5),"qti-assessment-section":Q.array(i5)}).strict(),m6=Q.object({items:Q.array(t9).min(1)}).strict(),l6=Q.object({metadata:Q.record(Q.string(),Q.unknown()).optional()}).strict(),u6=Q.object({identifier:X,title:X,qtiVersion:Q.string().optional(),toolName:Q.string().optional(),toolVersion:Q.string().optional(),timeLimit:Q.number().optional(),maxAttempts:Q.number().optional(),toolsEnabled:Q.record(Q.string(),Q.boolean()).optional(),metadata:Q.record(Q.string(),Q.unknown()).optional(),"qti-test-part":Q.array(e9),"qti-outcome-declaration":Q.array(p9).optional()}).strict(),c6=Q.object({identifier:X.optional(),title:X,qtiVersion:Q.string().optional(),toolName:Q.string().optional(),toolVersion:Q.string().optional(),timeLimit:Q.number().optional(),maxAttempts:Q.number().optional(),toolsEnabled:Q.record(Q.string(),Q.boolean()).optional(),metadata:Q.record(Q.string(),Q.unknown()).optional(),"qti-test-part":Q.array(e9),"qti-outcome-declaration":Q.array(p9).optional()}).strict(),p6=Q.object({identifier:X,title:X,label:Q.string().optional(),language:Q.string().optional(),stylesheet:n9.optional(),content:Q.string(),catalogInfo:Q.array(r9).optional(),toolName:Q.string().optional(),toolVersion:Q.string().optional(),metadata:Q.record(Q.string(),Q.unknown()).optional()}).strict(),a6=Q.object({identifier:X.optional(),title:X,label:Q.string().optional(),language:Q.string().optional(),stylesheet:n9.optional(),content:Q.string(),catalogInfo:Q.array(r9).optional(),toolName:Q.string().optional(),toolVersion:Q.string().optional(),metadata:Q.record(Q.string(),Q.unknown()).optional()}).strict(),i6=Q.object({xml:Q.string().optional(),schema:l9,entityId:Q.string().optional()}).strict(),s6=Q.object({xml:Q.array(Q.string()),schema:l9,entityIds:Q.array(Q.string())}).strict(),o6=Q.object({questionId:Q.string().optional(),userId:X,feedback:X,lessonId:X,humanApproved:Q.boolean().optional()}).strict(),s5=C2.enum(["creator_only","same_org","all_orgs"]),o5=C2.string().uuid(),r6=C2.email(),t6=C2.object({sql:X}),e6=C2.object({endpointName:X,sql:X,authMode:s5,description:X.optional()}),$7=C2.object({endpointId:o5});class $${transport;eventTransformer;constructor($,J){this.transport=$,this.eventTransformer=J?.eventTransformer}send($,J){o(pZ,{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(w9,$,"caliper envelope"),S.debug("Sending Caliper envelope",{sensor:$.sensor,eventCount:$.data.length});let J=this.eventTransformer?this.eventTransformer.transformEnvelope($):$,K=(await this.transport.request(this.transport.paths.send,{method:"POST",body:J}))?.jobId;return S.debug("Events sent",{jobId:K??"(none)"}),{jobId:K}}async validate($){let J=this.transport.paths.validate;if(!J)throw Error("validate() is not supported on this platform");o(w9,$,"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(U9,$,"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 K=await this.transport.request(J,{params:Z});return{events:K.events??[],pagination:K.pagination}}stream($={}){let J=this.transport.paths.list;if(!J)throw Error("stream() is not supported on this platform");let{max:Z,...K}=$;o(U9,K,"list events params"),P8($),S.debug("Streaming events",{params:$});let H={};if(K.limit!==void 0)H.limit=K.limit;if(K.offset!==void 0)H.offset=K.offset;if(K.sensor)H.sensor=K.sensor;if(K.startDate)H.startDate=K.startDate;if(K.endDate)H.endDate=K.endDate;if(K.actorId)H.actorId=K.actorId;if(K.actorEmail)H.actorEmail=K.actorEmail;return new q9(this.transport,J,{...H,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($)),K=await this.transport.request(Z);if(!K.event)throw Error(`Event not found: ${$}`);return K.event}sendActivity($,J){o(mZ,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(lZ,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(sZ,J,"question seen event");let Z=OZ(J);return S.debug("Sending AssessmentItemEvent.Started",{eventId:Z.id,actor:J.actor,object:J.object.id}),this.send($,[Z])}sendQuestionAnswered($,J){o(oZ,J,"question answered event");let Z=xZ(J);return S.debug("Sending AssessmentItemEvent.Completed",{eventId:Z.id,actor:J.actor,object:J.object.id}),this.send($,[Z])}sendQuestionGraded($,J){o(nZ,J,"question graded event");let Z=DZ(J);return S.debug("Sending GradeEvent.Graded",{eventId:Z.id,actor:J.actor,object:J.object,scoreGiven:J.generated.scoreGiven}),this.send($,[Z])}}class J${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"),n5(J);let{timeoutMs:Z=30000,pollIntervalMs:K=1000}=J,H=Date.now();S.debug("Waiting for job completion",{jobId:$,timeoutMs:Z,pollIntervalMs:K});while(Date.now()-H<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((F)=>{setTimeout(F,K)})}throw Error(`Job ${$} timed out after ${Z}ms`)}}function n5($){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 R2("Invalid polling options",J)}function r5($=u2){return class{transport;_provider;events;jobs;constructor(Z={}){let K=CZ(Z,$),H;if(K.mode==="transport")this.transport=K.transport,S.info("Client initialized with custom transport");else{let{provider:W}=K,{baseUrl:F,paths:_}=W.getEndpointWithPaths("caliper"),O=W.getTokenProvider("caliper");if(!O)throw Error("Caliper API requires authentication");if(this._provider=W,this.transport=new k9({baseUrl:F,tokenProvider:O,timeout:W.timeout,paths:_}),W.platform===X9)H=new C9;S.info("Client initialized",{platform:W.platform,env:W.env,baseUrl:F})}this.events=new $$(this.transport,{eventTransformer:H}),this.jobs=new J$(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 J7=r5();class K2 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 Z$="/ims/oneroster/rostering/v1p2";function K$($,J){return`${$.replace(/\/+$/,"")}/${J.replace(/^\/+/,"")}`}function t5($,J){return K$($,`${Z$}/courses/${J}`)}function e5($,J){return K$($,`${Z$}/users/${J}`)}function H$($,J){let Z=$.ids?.[J];if(!Z)throw new K2($,J);return Z}function $0($,J,Z){let K=H$(J,Z);return t5($,K)}function J0($){if($.courseCode)return $.courseCode;if($.grade!==void 0)return`${$.subject} G${String($.grade)}`;return $.subject}function v8($,J,Z){let K;try{K=new URL($)}catch{throw new i2($)}let H="grade"in J?`${J.subject}/g${String(J.grade)}`:J.code,W=K.pathname.replace(/\/+$/,"");return K.pathname=`${W}/activities/${H}/${encodeURIComponent(Z)}`,K.toString()}function X$($,J,Z,K,H,W,F){return{id:v8(H,$.course,$.id),type:"TimebackActivityContext",subject:J.subject,app:{name:Z},activity:{name:$.name},course:{id:$0(W,J,K),name:J0(J)},process:F??!0}}function Z0($){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 K0($,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 H0($){return $===void 0?void 0:{pctCompleteApp:$}}function W$($,J){return{courseId:H$($,J)}}function Y$($){if(!$)return;return`urn:uuid:${$}`}function Q$($,J,Z){return{id:e5($,J),type:"TimebackUser",email:Z}}function G$($){let{sensor:J,timebackId:Z,email:K,payload:H,process:W,course:F,appName:_,apiEnv:O,onerosterBaseUrl:w,attempt:j,runId:A}=$,C=Q$(w,Z,K),q=X$(H,F,_,O,J,w,W??H.process),z=Z0(H.metrics),N=H0(H.pctComplete),I=Y$(A),r=W$(F,O),H2=T8({actor:C,object:q,metrics:z,eventTime:H.endedAt,generatedExtensions:N,extensions:r,attempt:j,session:I});return{sensor:J,actor:C,object:q,event:H2,payload:H,course:F,appName:_,apiEnv:O,email:K,timebackId:Z}}function s2($){let{sensor:J,timebackId:Z,email:K,payload:H,course:W,appName:F,apiEnv:_,onerosterBaseUrl:O,runId:w}=$,j=Q$(O,Z,K),A=X$(H,W,F,_,J,O),C=K0(H.elapsedMs,H.pausedMs),q=W$(W,_),z=Y$(w),N=R8({actor:j,object:A,metrics:C,eventTime:H.endedAt,extensions:q,session:z});return{sensor:J,actor:j,object:A,event:N,payload:{id:H.id,name:H.name,course:H.course,startedAt:H.startedAt,endedAt:H.endedAt,elapsedMs:H.elapsedMs,pausedMs:H.pausedMs},course:W,appName:F,apiEnv:_,email:K,timebackId:Z,runId:w}}import*as h from"zod";import*as h2 from"zod";var X0=h2.object({subject:B$,grade:_$}),W0=h2.object({code:E}),w2=h2.union([X0,W0]);class Q2 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 U2($,J){let Z;if("grade"in J)Z=$.filter((K)=>K.subject===J.subject&&K.grade===J.grade);else Z=$.filter((K)=>K.courseCode===J.code);if(Z.length===0)throw new Q2("unknown_course",J);if(Z.length>1)throw new Q2("ambiguous_course",J,Z.length);return Z[0]}var A2=R("handlers:activity:schema"),Y0=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 Q0($){return Math.min(100,Math.max(0,$))}function o2($){if("grade"in $)return`${$.subject} grade ${$.grade}`;return`code "${$.code}"`}var G0=h.object({id:E,name:E,course:w2,runId:h.string().uuid(),startedAt:h.iso.datetime(),endedAt:h.iso.datetime(),elapsedMs:h.number().int().nonnegative(),pausedMs:h.number().int().nonnegative()});function F$($,J,Z){let K=G0.safeParse($);if(!K.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:K.error.flatten()})};let H=K.data,W;try{W=U2(J.courses,H.course)}catch(_){if(_ instanceof Q2){let O=o2(H.course);if(_.code==="unknown_course")return A2.warn("Unknown course selector",{selector:H.course}),{ok:!1,response:M("UNKNOWN_COURSE",`Unknown course: ${O}`,400)};return A2.error("Ambiguous course selector",{selector:H.course}),{ok:!1,response:M("AMBIGUOUS_COURSE_SELECTOR","Ambiguous course selector in timeback.config.json",500)}}throw _}let F=S2(J,W,Z);if(!F){let _=o2(H.course);return A2.error("Missing sensor for course",{selector:H.course}),{ok:!1,response:M("MISSING_SENSOR",`Course "${_}" has no sensor configured.`,500)}}return{ok:!0,payload:H,course:W,sensor:F}}var B0=h.object({id:E,name:E,course:w2,process:h.boolean().optional(),runId:h.string().uuid(),endedAt:h.iso.datetime(),metrics:Y0,pctComplete:h.number().optional().transform(($)=>$===void 0?void 0:Q0($))});function O$($,J,Z){let K=B0.safeParse($);if(!K.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:K.error.flatten()})};let H=K.data,W;try{W=U2(J.courses,H.course)}catch(_){if(_ instanceof Q2){let O=o2(H.course);if(_.code==="unknown_course")return A2.warn("Unknown course selector",{selector:H.course}),{ok:!1,response:M("UNKNOWN_COURSE",`Unknown course: ${O}`,400)};return A2.error("Ambiguous course selector",{selector:H.course}),{ok:!1,response:M("AMBIGUOUS_COURSE_SELECTOR","Ambiguous course selector in timeback.config.json",500)}}throw _}let F=S2(J,W,Z);if(!F){let _=o2(H.course);return A2.error("Missing sensor for course",{selector:H.course}),{ok:!1,response:M("MISSING_SENSOR",`Course "${_}" has no sensor configured.`,500)}}return{ok:!0,payload:H,course:W,sensor:F}}import{getServiceUrlsForEnv as j$}from"@timeback/core";var y8=R("handlers:activity:attempts");async function x$($,J){let Z=[$,J].join("_");return`caliper_${await E2(Z)}`}function _0($,J){return $.find((Z)=>Z.scoreDate===J)}function F0($){let J=0;for(let Z of $){let H=Z.metadata?.attempt;if(typeof H==="number"&&H>J)J=H}return J}function O0($){let J=$.metadata;return typeof J?.attempt==="number"&&J.attempt>=1?J.attempt:1}async function D$($,J,Z,K){let H;try{H=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(H.length===0)return 1;let W=_0(H,K);if(W){let O=O0(W);return y8.debug("Retry detected, reusing attempt number",{lineItemId:J,timebackId:Z,endedAt:K,attempt:O}),O}let F=F0(H),_=F+1;return y8.debug("New attempt computed",{lineItemId:J,timebackId:Z,endedAt:K,maxAttempt:F,nextAttempt:_}),_}var w$=R("handlers:activity:completion");async function M$($){let{client:J,courseId:Z,timebackId:K,pctComplete:H,appName:W}=$;if(H!==100)return;let F=U$(Z),_=`timeback_sdk_${await E2(`mastery-completion_${Z}`)}`,O=`timeback_sdk_${await E2(`mastery-completion_${Z}_${K}`)}`;try{let w=!1;try{await J.oneroster.assessmentLineItems(_).get(),w=!0}catch{}if(!w)await J.oneroster.assessmentLineItems.create({sourcedId:_,title:`${W}: Complete`,status:"active",course:{sourcedId:F.course},componentResource:{sourcedId:F.componentResource},resultValueMin:0,resultValueMax:100,metadata:{appName:W}});let j=new Date().toISOString();await J.oneroster.assessmentResults.upsert(O,{status:"active",assessmentLineItem:{sourcedId:_},student:{sourcedId:K},score:100,scoreDate:j,scoreStatus:"fully graded",metadata:{isMasteryCompletion:!0,completedAt:j,appName:W}}),w$.debug("Created mastery completion entry",{courseId:Z,timebackId:K,lineItemId:_,resultId:O})}catch(w){let j=w instanceof Error?w.message:"Unknown error";w$.error("Failed to create mastery completion entry",{courseId:Z,timebackId:K,lineItemId:_,error:j})}}import{aggregateActivityMetrics as x0}from"@timeback/core/utils";var z2=R("handlers:activity:progress");function D0($,J){let Z=u(J),K=$.overrides?.[Z]?.metadata?.metrics,H=$.metadata?.metrics;return K?.totalLessons??H?.totalLessons}function w0($,J){let Z=$/J*100;return Math.min(100,Math.max(0,Math.round(Z)))}async function L$($){let{client:J,courseId:Z,timebackId:K,payload:H,course:W,env:F}=$;if(H.pctComplete!==void 0)return;let _=H.metrics.masteredUnits;if(typeof _!=="number"||_<=0)return;let O=D0(W,F);if(!O||O<=0){z2.debug("Skipping progress computation: totalLessons not configured",{courseId:Z});return}let w;try{w=(await J.edubridge.enrollments.list({userId:K})).find((N)=>N.course.id===Z)?.id}catch(z){let N=z instanceof Error?z.message:"Unknown error";z2.warn("Failed to fetch enrollments for progress computation",{courseId:Z,timebackId:K,error:N});return}if(!w){z2.warn("Skipping progress computation: enrollment not found for student/course",{courseId:Z,timebackId:K});return}let j=0;try{let z=await J.edubridge.analytics.getEnrollmentFacts({enrollmentId:w});j=x0(z).masteredUnits}catch(z){let N=z instanceof Error?z.message:"Unknown error";z2.warn("Failed to fetch enrollment facts for progress computation",{courseId:Z,timebackId:K,enrollmentId:w,error:N});return}let A=!0;try{if((await J.edubridge.analytics.getWeeklyFacts({studentId:K,weekDate:H.endedAt})).some((I)=>{if(I.activityId!==H.id)return!1;if(I.enrollmentId&&I.enrollmentId===w)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";z2.warn("Failed to fetch weekly facts for retry-safe progress computation",{courseId:Z,timebackId:K,enrollmentId:w,activityId:H.id,error:N}),A=!1}let C=j+(A?_:0),q=w0(C,O);return z2.debug("Computed pctComplete",{courseId:Z,timebackId:K,enrollmentId:w,historicalMasteredUnits:j,currentMasteredUnits:_,shouldIncludeCurrentMasteredUnits:A,totalMastered:C,totalLessons:O,pctComplete:q}),{pctComplete:q}}var f8=R("handlers:activity:submit"),U0={computeProgress:L$,maybeWriteCompletionEntry:M$};function V$($,J){return $.timebackId??G2({email:$.email,client:J})}async function r2($,J,Z=U0){let K=u($.env),{userInfo:H,payload:W,course:F,sensor:_,preview:O=!1,runId:w}=J,j=F.ids?.[K];if(!j)throw new K2(F,K);let A=$.getClient(),C=await V$(H,A),q=W;if(W.pctComplete===void 0){let f2=await Z.computeProgress({client:A,courseId:j,timebackId:C,payload:W,course:F,env:$.env});if(f2)q={...W,pctComplete:f2.pctComplete}}let z;if(!O){let f2=v8(_,q.course,q.id),xJ=await x$(f2,j);z=await D$(A,xJ,C,q.endedAt)}let I=j$(K).oneroster,r=G$({sensor:_,timebackId:C,email:H.email,payload:q,process:q.process,course:F,appName:$.appConfig.name,apiEnv:K,onerosterBaseUrl:I,attempt:z,runId:w}),H2=$.hooks?.beforeActivitySend,y2=H2?await H2(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:j,timebackId:p.timebackId,pctComplete:p.payload.pctComplete,appName:p.appName}),f8.debug("Recorded completion",{courseSelector:W.course,activityId:W.id}),{success:!0}}async function k$($,J){let Z=u($.env),{userInfo:K,payload:H,course:W,sensor:F,runId:_}=J;if(!W.ids?.[Z])throw new K2(W,Z);let w=$.getClient(),j=await V$(K,w),C=j$(Z).oneroster,q=s2({sensor:F,timebackId:j,email:K.email,payload:H,course:W,appName:$.appConfig.name,apiEnv:Z,onerosterBaseUrl:C,runId:_}),z=$.hooks?.beforeTimeSpentSend,N=z?await z(q):q,I=N??q;if(N===null){f8.debug("Hook skipped time-spent send",{activityId:H.id});return}await w.caliper.events.send(I.sensor,[I.event]),f8.debug("Sent time-spent event",{activityId:H.id,elapsedMs:H.elapsedMs})}var v2=R("handlers:activity:submit");async function P2($,J){if($.mode==="custom"){let K=await $.getEmail(J);return K?{email:K}: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 P2($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let K=O$(await J.json(),$.appConfig,$.env);if(!K.ok)return K.response;let{payload:H,course:W,sensor:F}=K,_={id:H.id,name:H.name,course:H.course,process:H.process,endedAt:H.endedAt,metrics:H.metrics,pctComplete:H.pctComplete};try{if((await r2({env:$.env,api:$.api,appConfig:$.appConfig,getClient:$.getClient,hooks:$.hooks},{userInfo:Z,payload:_,course:W,sensor:F,preview:!1,runId:H.runId})).preview)return P({preview:!0});return v2.debug("Submitted activity",{runId:H.runId,activityId:H.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",B2(O));if(O instanceof K2)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 w=O instanceof Error?O.message:"Unknown error";return v2.error("Failed to submit activity",{error:w}),M("INTERNAL_ERROR",w,502)}}}var T2=R("handlers:activity:heartbeat"),e2=new Map,L0=300000;function j0(){let $=Date.now();for(let[J,Z]of e2)if($-Z>L0)e2.delete(J)}function V0($){return j0(),e2.has($)}function q$($){e2.set($,Date.now())}function $8($){return async(J)=>{let Z=await P2($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let K=F$(await J.json(),$.appConfig,$.env);if(!K.ok)return K.response;let{payload:H,course:W,sensor:F}=K,_=new Date(H.startedAt).getTime(),O=`${H.runId}:${_}`;if(V0(O))return T2.debug("Duplicate heartbeat, skipping",{runId:H.runId,windowKey:O}),P({duplicate:!0});let w=u($.env);if(!W.ids?.[w]){let C=new K2(W,w);return T2.warn("Course not synced",{course:H.course,env:w}),M("MISSING_SYNCED_COURSE_ID",C.message,503)}let A=$.getClient();try{let C=Z.timebackId??await G2({email:Z.email,client:A}),z=M0(w).oneroster,N=s2({sensor:F,timebackId:C,email:Z.email,payload:H,course:W,appName:$.appConfig.name,apiEnv:w,onerosterBaseUrl:z,runId:H.runId}),I=$.hooks?.beforeTimeSpentSend,r=I?await I(N):N,H2=r??N;if(r===null)return q$(O),T2.debug("Hook skipped heartbeat send",{runId:H.runId}),new Response(null,{status:204});return await A.caliper.events.send(H2.sensor,[H2.event]),q$(O),T2.debug("Sent heartbeat",{runId:H.runId,activityId:H.id,elapsedMs:H.elapsedMs}),new Response(null,{status:204})}catch(C){if(C instanceof d)return T2.warn("Failed to resolve Timeback user",{code:C.code}),M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",B2(C));let q=C instanceof Error?C.message:"Unknown error";return T2.error("Failed to send heartbeat",{error:q}),M("INTERNAL_ERROR",q,502)}}}async function h$($,J,Z){if(Z.mode!=="sso")return M2.warn("SSO not configured"),P({error:"SSO not configured"},400);return await N$({req:$,env:J,clientId:Z.clientId,issuer:Z.issuer,redirectUri:Z.redirectUri,buildState:Z.buildState})}async function k0($,J,Z,K,H,W,F){try{let _=W.issuer??C$(H),O=b$(J,W.redirectUri);M2.debug("Exchanging auth code for tokens",{issuer:_,clientId:W.clientId});let w=await A$({issuer:_,clientId:W.clientId,clientSecret:W.clientSecret,code:$,redirectUri:O}),j=await z$({issuer:_,accessToken:w.access_token}),A=typeof j.identities==="string"?JSON.parse(j.identities):j.identities;M2.debug("SSO completed, resolving Timeback user",{user:{...j,identities:A}});let C=await _2({env:H,apiCredentials:F.credentials,userInfo:j,client:F.getClient()});M2.debug("Timeback user resolved",{timebackId:C.id});let q={user:C,idp:{tokens:w,userInfo:j},state:Z,req:K,redirect:J8,json:P};return W.onCallbackSuccess(q)}catch(_){let O=_ instanceof Error?_:Error("Unknown error"),w=_ instanceof d?_.code:"token_exchange_failed";if(M2.error("SSO callback failed",{error:O.message,errorCode:w}),W.onCallbackError)return W.onCallbackError(P$(O,w,Z,K));if(_ instanceof d)return P({error:O.message},B2(_));return P({error:O.message},500)}}async function S$($,J,Z,K){if(Z.mode!=="sso")return M2.warn("SSO not configured"),P({error:"SSO not configured"},400);let{url:H,code:W,errorParam:F,state:_}=I$($);if(F)return T$(F,H,_,$,Z.onCallbackError);if(!W)return R$(_,$,Z.onCallbackError);return await k0(W,H,_,$,J,Z,K)}function Z8($){let{env:J,identity:Z}=$;return(K)=>h$(K,J,Z)}function K8($){let{env:J,identity:Z,api:K}=$;return(H)=>S$(H,J,Z,K)}function H8(){return()=>J8("/")}var E$=($)=>{return async(J)=>{return await $.getClient().powerpath.assessments.getAssessmentProgress({student:J.student,lesson:J.lesson,attempt:J.attempt})}};var v$=($)=>{return async(J)=>{return(await $.getClient().powerpath.assessments.getAttempts({student:J.student,lesson:J.lesson})).attempts.map((K)=>({attempt:K.attempt??void 0,score:K.score,startedAt:K.startedAt??void 0,completedAt:K.completedAt??void 0,scoreStatus:K.scoreStatus}))}};var y$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments,K=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson}),H;if(K.lessonType!=="powerpath-100"&&K.finalized!==!0){try{H=await Z.finalStudentAssessmentResponse({student:J.student,lesson:J.lesson})}catch{}K=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson})}let W=K.lessonType==="powerpath-100"?!0:K.finalized===!0;return{...H??{},score:K.score??0,totalQuestions:K.totalQuestions,correctQuestions:K.correctQuestions,accuracy:K.accuracy,finalized:W}}};function f$($){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 g$($,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 d$($){let J=$.questions,Z=J.filter((K)=>K.response!==void 0).map((K)=>K.id);return{questions:J.map(X8),answeredIds:Z,score:$.score??0,finalized:$.finalized,complete:$.finalized}}function q0($){if($.lessonType==="powerpath-100")return $.responseResult.isCorrect;return!1}function C0($){if($.lessonType==="powerpath-100")return $.powerpathScore;return 0}function m$($){let J=C0($);return{correct:q0($),score:J,complete:$.lessonType==="powerpath-100"&&J>=100,questionResult:$.questionResult}}function l$($,J){return Array.from(new Set($.map((Z)=>Z.ids?.[J]).filter((Z)=>Boolean(Z))))}function u$($){return String($.sourcedId??"")}var c$=($)=>{return async(J={})=>{let Z=$.getClient(),K=u($.env),H;if(J.course){let _=U2($.appConfig.courses,J.course).ids?.[K];H=_?[_]:[]}else H=l$($.appConfig.courses,K);let W=[];for(let F of H){let _=await Z.oneroster.courses(F).components({where:{status:"active"}}),O=g8(_);for(let w of O){let j=u$(w);if(!j)continue;let A=await Z.oneroster.courses.componentResources({where:{"courseComponent.sourcedId":j,status:"active"}}),C=g8(A);for(let q of C){if(!q.sourcedId)continue;let z=q.metadata??{},N=f$(q);W.push({id:q.sourcedId,name:q.title??q.sourcedId,type:z.lessonType??"quiz",courseId:F,questionCount:N,metadata:z})}}}return W}};class W8 extends Error{name="LessonTypeMismatchError"}var p$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments;if(J.lessonType==="powerpath-100"){let H=await Z.getNextQuestion({student:J.student,lesson:J.lesson});return X8(H.question)}let K=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson});if(K.lessonType==="powerpath-100"){if(J.lessonType!==void 0)throw new W8(`Expected quiz progress but got powerpath-100 for lesson ${J.lesson}`);let H=await Z.getNextQuestion({student:J.student,lesson:J.lesson});return X8(H.question)}return d$(K)}};var A0=R("lessons:start"),a$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments,K=J.lesson,H=J.student,W=await Z.getAssessmentProgress({student:H,lesson:K});if(J.forceNew){if(W.lessonType!=="powerpath-100"&&W.finalized!==!0)try{await Z.finalStudentAssessmentResponse({student:H,lesson:K})}catch{}try{await Z.createNewAttempt({student:H,lesson:K}),W=await Z.getAssessmentProgress({student:H,lesson:K})}catch(F){A0.warn("Failed to create new lesson attempt",{lessonId:K,error:F})}}else if(W.lessonType!=="powerpath-100"&&W.finalized!==!0&&(W.questions?.length??0)===0)try{await Z.createNewAttempt({student:H,lesson:K}),W=await Z.getAssessmentProgress({student:H,lesson:K})}catch{}return g$(K,W)}};var i$=($)=>{return async(J)=>{let K=await $.getClient().powerpath.assessments.updateStudentQuestionResponse({student:J.student,lesson:J.lesson,question:J.question,responses:{RESPONSE:J.response}});return m$(K)}};function m($){return{list:c$($),start:a$($),next:p$($),submit:i$($),complete:y$($),attempts:v$($),attemptDetails:E$($)}}async function a($,J){let Z=await P2($.identity,J);if(!Z)return null;if(Z.timebackId)return Z.timebackId;return await G2({email:Z.email,client:$.getClient()})}async function s$($,J){let Z=await P2($.identity,J);return Boolean(Z)}var z0=R("handlers:lessons");function c($,J){if($ instanceof d)return M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",B2($));if($ instanceof W8)return M("LESSON_TYPE_MISMATCH",$.message,400);let Z=$ instanceof Error?$.message:"Unknown error";return z0.error(J,{error:Z}),M("INTERNAL_ERROR",Z,500)}import*as i from"zod";var P0=i.object({course:w2.optional()}).partial(),T0=i.object({lessonId:E,forceNew:i.boolean().optional()}),R0=i.object({lessonId:E,lessonType:E.optional()}),N0=i.object({lessonId:E,questionId:E,response:E}),I0=i.object({lessonId:E}),b0=i.object({lessonId:E}),h0=i.object({lessonId:E,attempt:i.number().int().positive()});async function S0($){try{let J=await $.json();if(!J||typeof J!=="object"||Array.isArray(J))return null;return J}catch{return null}}async function L2($,J){let Z=await S0($);if(!Z)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid JSON body",400)};let K=J.safeParse(Z);if(!K.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:K.error.flatten()})};return{ok:!0,payload:K.data}}async function o$($){return await L2($,P0)}async function n$($){return await L2($,T0)}async function r$($){return await L2($,R0)}async function t$($){return await L2($,N0)}async function e$($){return await L2($,I0)}async function $J($){return await L2($,b0)}async function JJ($){return await L2($,h0)}function Y8($){let J=m($);return async(Z)=>{try{let K=await JJ(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.attemptDetails({lesson:K.payload.lessonId,attempt:K.payload.attempt,student:H});return P(W)}catch(K){return c(K,"Failed to fetch lesson attempt details")}}}function Q8($){let J=m($);return async(Z)=>{try{let K=await $J(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.attempts({lesson:K.payload.lessonId,student:H});return P({attempts:W})}catch(K){return c(K,"Failed to list lesson attempts")}}}function G8($){let J=m($);return async(Z)=>{try{let K=await e$(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.complete({lesson:K.payload.lessonId,student:H});return P(W)}catch(K){return c(K,"Failed to complete lesson")}}}function B8($){let J=m($);return async(Z)=>{try{if(!await s$($,Z))return M("UNAUTHORIZED","Unauthorized",401);let K=await o$(Z);if(!K.ok)return K.response;let H=await J.list({course:K.payload.course});return P({lessons:H})}catch(K){return c(K,"Failed to list lessons")}}}function _8($){let J=m($);return async(Z)=>{try{let K=await r$(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.next({lesson:K.payload.lessonId,lessonType:K.payload.lessonType,student:H});return P(W)}catch(K){return c(K,"Failed to fetch next lesson question")}}}function F8($){let J=m($);return async(Z)=>{try{let K=await n$(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.start({lesson:K.payload.lessonId,student:H,forceNew:K.payload.forceNew});return P(W)}catch(K){return c(K,"Failed to start lesson")}}}function O8($){let J=m($);return async(Z)=>{try{let K=await t$(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.submit({lesson:K.payload.lessonId,question:K.payload.questionId,response:K.payload.response,student:H});return P(W)}catch(K){return c(K,"Failed to submit lesson answer")}}}function ZJ($,J){let Z=new Map;for(let K of $){let H=K.ids?.[J];if(H)Z.set(H,K)}return Z}function KJ($,J){return $.map((Z)=>{let K=J.get(Z.course.id);return{id:Z.course.id,code:K?.courseCode??Z.course.id,name:Z.course.title}})}function HJ($){return $.map((J)=>J.metadata?.goals).find(Boolean)}function XJ($){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((K,H)=>{return K+(H.activityMetrics?.xpEarned??0)},0)},0)}async function x8($,J,Z,K){let H=await $.edubridge.enrollments.list({userId:J.id}),W=ZJ(Z.courses,K),F=KJ(H,W),_=HJ(H),{start:O,end:w}=XJ(new Date),[j,A]=await Promise.all([$.edubridge.analytics.getActivity({studentId:J.id,startDate:O.toISOString(),endDate:w.toISOString()}),$.edubridge.analytics.getActivity({studentId:J.id,startDate:"2000-01-01",endDate:w.toISOString()})]);return{id:J.id,email:J.email,name:J.name,school:J.school,grade:J.grade,courses:F.length?F:void 0,goals:_,xp:{today:d8(j),all:d8(A)}}}var m8=R("handlers:user");async function E0($,J){if($.mode==="custom"){let K=await $.getEmail(J);if(!K)return;return{sub:K,email:K}}let Z=await $.getUser(J);if(!Z)return;return{sub:Z.id,email:Z.email}}function v0($){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 D8($){return async(J)=>{try{let Z=await E0($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let K=u($.env),H=$.getClient();try{let W=await _2({env:$.env,apiCredentials:$.api,userInfo:Z,client:H}),F=await x8(H,W,$.appConfig,K);return P(F)}catch(W){if(W instanceof d)return v0(W);let F=W instanceof Error?W.message:"Unknown error";return m8.error("Failed to build user profile",{error:F}),M("INTERNAL_ERROR",F,502)}}catch(Z){let K=Z instanceof Error?Z.message:"Unknown error";return m8.error("Unhandled error in user handler",{error:K}),M("INTERNAL_ERROR",K,500)}}}var l8=R("handlers:user:verify");async function y0($,J){if($.mode==="custom"){let K=await $.getEmail(J);if(!K)return;return{sub:K,email:K}}let Z=await $.getUser(J);if(!Z)return;return{sub:Z.id,email:Z.email}}function w8($){return async(J)=>{try{let Z=await y0($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let K=$.getClient();try{let H=await _2({env:$.env,apiCredentials:$.api,userInfo:Z,client:K});return P({verified:!0,timebackId:H.id})}catch(H){if(H instanceof d){if(l8.warn("Timeback user resolution failed",{code:H.code}),H.code==="timeback_user_ambiguous")return M("USER_AMBIGUOUS","Timeback user resolution ambiguous",409);if(H.code==="timeback_user_not_found")return P({verified:!1});return M("USER_RESOLUTION_FAILED",H.message,502)}let W=H instanceof Error?H.message:"Unknown error";return l8.error("Failed to verify user",{error:W}),M("INTERNAL_ERROR",W,502)}}catch(Z){let K=Z instanceof Error?Z.message:"Unknown error";return l8.error("Unhandled error in verify handler",{error:K}),M("INTERNAL_ERROR",K,500)}}}import*as T from"zod";var f0=T.object({email:T.email(),timebackId:T.string().optional()}),g0=T.object({id:E,name:E,course:w2}),d0=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),m0=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()}),l0=T.uuid().optional(),WJ=T.object({user:f0,activity:g0,metrics:d0,time:m0.optional(),runId:l0});var YJ=R("activity:record");class QJ 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 u0($){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 c0($,J){let Z=$.time,K=Z.activeMs??0,H=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 F=new Date(J),_=K+H;W=new Date(F.getTime()-_).toISOString()}else W=J;return{id:$.activity.id,name:$.activity.name,course:$.activity.course,startedAt:W,endedAt:J,elapsedMs:K,pausedMs:H}}function p0($,J,Z){let K;try{K=U2($.courses,J)}catch(W){if(W instanceof Q2)throw new u8(W.code==="unknown_course"?`Unknown course: ${JSON.stringify(J)}`:"Ambiguous course selector in timeback.config.json");throw W}let H=S2($,K,Z);if(!H)throw new u8(`Course "${JSON.stringify(J)}" has no sensor configured.`);return{course:K,sensor:H}}var a0={recordCompletion:r2,sendTimeSpent:k$};function c8($,J=a0){return async(Z)=>{let K=WJ.safeParse(Z);if(!K.success)throw YJ.warn("Invalid record input",{error:K.error.flatten()}),new QJ("Invalid activity record input",K.error.flatten());let H=K.data,{course:W,sensor:F}=p0($.appConfig,H.activity.course,$.env),_=u0(H),O={env:$.env,api:$.api,appConfig:$.appConfig,getClient:$.getClient,hooks:$.hooks};if(await J.recordCompletion(O,{userInfo:{email:H.user.email,timebackId:H.user.timebackId},payload:_,course:W,sensor:F,preview:!1,runId:H.runId}),H.time){let w=c0(H,_.endedAt);await J.sendTimeSpent(O,{userInfo:{email:H.user.email,timebackId:H.user.timebackId},payload:w,course:W,sensor:F,runId:H.runId})}YJ.debug("Recorded activity",{activityId:H.activity.id,hasTime:!!H.time})}}var U8=R("user:verify");function p8($){return async(J)=>{U8.debug("Verifying user by email");let Z=$.getClient();try{let K=await G2({email:J,client:Z});return U8.debug("User verified",{timebackId:K}),{verified:!0,timebackId:K}}catch(K){if(K instanceof d){if(K.code==="timeback_user_not_found")return U8.debug("User not found in Timeback"),{verified:!1};throw K}let H=K instanceof Error?K.message:"Unknown error";throw U8.error("Failed to verify user",{error:H}),K}}}var a8=R("user:getProfile");function i8($){return async(J)=>{a8.debug("Building user profile by email");let Z=$.getClient(),K=u($.env);try{let H=await _2({env:$.env,apiCredentials:$.api,userInfo:{sub:J,email:J},client:Z}),W=await x8(Z,H,$.appConfig,K);return a8.debug("User profile built",{userId:W.id}),W}catch(H){if(H instanceof d)throw H;let W=H instanceof Error?H.message:"Unknown error";throw a8.error("Failed to build user profile",{error:W}),H}}}async function s0($){let{loadConfig:J}=await import("./chunk-44nb8dx1.js");return await J($)}function o0($){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 FJ($){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.");GJ({logger:$.logger,logLevel:$.logLevel});let J=BJ($.env),Z=await s0({configPath:$.configPath});if(!Z.success)throw Error(`Failed to load timeback config: ${Z.error}`);let K=Z.config,H,W=()=>{if(!H)H=new i0({env:u(J),auth:{clientId:$.api.clientId,clientSecret:$.api.clientSecret}});return H},F=$.hooks,_=(j,A)=>_J(j,F,A),O={env:J,identity:$.identity,appConfig:{name:K.name,sensor:K.sensor,launchUrl:K.launchUrl,courses:o0(K.courses)},api:$.api,getClient:W,hooks:F},w={env:J,identity:$.identity,api:{credentials:$.api,getClient:W}};return{config:$,handle:{activity:{heartbeat:_("activity.heartbeat",$8(O)),submit:_("activity.submit",t2(O))},identity:{signIn:_("identity.signIn",Z8(w)),callback:_("identity.callback",K8(w)),signOut:H8()},lessons:{list:_("lessons.list",B8(O)),start:_("lessons.start",F8(O)),next:_("lessons.next",_8(O)),submit:_("lessons.submit",O8(O)),complete:_("lessons.complete",G8(O)),attempts:_("lessons.attempts",Q8(O)),attemptDetails:_("lessons.attemptDetails",Y8(O))},user:{me:_("user.me",D8(O)),verify:_("user.verify",w8(O))}},activity:{record:c8({...O})},lessons:m({...O}),user:{verify:p8({...O}),getProfile:i8({...O})},get api(){return W()},close(){if(H)H.close(),H=void 0}}}import{TimebackClient as zH}from"@timeback/core";import{ApiError as TH,ForbiddenError as RH,isApiError as NH,NotFoundError as IH,UnauthorizedError as bH,ValidationError as hH}from"@timeback/core";export{r0 as toNativeHandler,NH as isApiError,OJ as createTimebackIdentity,FJ as createTimeback,hH as ValidationError,bH as UnauthorizedError,zH as TimebackClient,n0 as ROUTES,IH as NotFoundError,RH as ForbiddenError,TH as ApiError};
1
+ import"./chunk-ae6bkfs5.js";import{a as E,b as B$,c as _$}from"./chunk-5qkmnsbm.js";import{e as P$,f as T$,g as R$,h as N$,i as I$,j as b$,k as OJ}from"./chunk-6nybr8hs.js";import{l as r0}from"./server/adapters/native.js";import{m as d,n as B2,o as _2,p as G2,q as n2,r as S2,s as _J}from"./chunk-bh0ewh45.js";import{t as GJ,u as R,v as M2,w as C$,y as A$,z as z$}from"./chunk-nw6brg34.js";import"./chunk-3pf3cpfk.js";import{G as g8,H as U$,I as E2,J as u,K as BJ,L as P,M,N as J8,Q as n0}from"./chunk-dbcsf5tk.js";import{R as L8,U as j8}from"./chunk-1s709w5e.js";import{TimebackClient as i0}from"@timeback/core";import{getServiceUrlsForEnv as M0}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 K9 extends s{name="ValidationError";constructor($="Validation Error",J){super($,422,J)}}class H9 extends s{name="InputValidationError";issues;constructor($,J){let Z={imsx_codeMajor:"failure",imsx_severity:"error",imsx_description:$,imsx_error_details:J.map((K)=>({path:K.path,message:K.message}))};super($,400,Z);this.issues=J}}function R2($,J){return new H9($,J)}function DJ($){if(!($ instanceof Error))return!1;return"statusCode"in $&&"response"in $}var wJ="BEYOND_AI",X9="LEARNWITH_AI",W9=wJ,UJ={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"},MJ={staging:"https://api.staging.alpha-1edtech.ai",production:"https://api.alpha-1edtech.ai"},LJ={staging:"https://caliper.staging.alpha-1edtech.ai",production:"https://caliper.alpha-1edtech.ai"},jJ={staging:"https://qti.alpha-1edtech.ai/api",production:"https://qti.alpha-1edtech.ai/api"},VJ={staging:"https://platform.dev.timeback.com/auth/1.0/token",production:"https://platform.timeback.com/auth/1.0/token"},kJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},qJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},CJ={staging:"https://platform.dev.timeback.com",production:"https://platform.timeback.com"},AJ={BEYOND_AI:{token:UJ,tokenScope:void 0,api:MJ,caliper:LJ,qti:jJ},LEARNWITH_AI:{token:VJ,tokenScope:"https://purl.imsglobal.org/spec/caliper/v1p2/scope/events.write",api:kJ,caliper:qJ,qti:CJ}},zJ={tsc:"tsc",nativePreview:"@typescript/native-preview"},J1={package:zJ.nativePreview,bin:"tsgo"},e=null,V8=!1,k8=!1;function o8($){let Z=$.replace(/[.+?^${}()|[\]\\]/g,"\\$&").replace(/\*/g,".*");return new RegExp(`^${Z}$`)}function PJ(){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((K)=>K.trim()).filter(Boolean);for(let K of J)if(K.startsWith("-"))e.push({regex:o8(K.slice(1)),exclude:!0});else e.push({regex:o8(K),exclude:!1});if(!e.some((K)=>!K.exclude)&&e.length>0)V8=!0}function TJ($){if(PJ(),!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 RJ(){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 NJ($){switch($){case"debug":return $2.blue;case"info":return $2.cyan;case"warn":return $2.yellow;case"error":return $2.red}}function IJ($){switch($){case"debug":return console.debug;case"info":return console.info;case"warn":return console.warn;case"error":return console.error}}function bJ($){if(q8)return q8($,{depth:null,colors:!0,breakLength:80,compact:!1});return JSON.stringify($,null,2)}var hJ=($)=>{let J=NJ($.level),Z=IJ($.level),K=$.level.toUpperCase().padEnd(5),H=$.timestamp.toISOString().replace(/\.\d{3}Z$/,""),W=`${$2.dim}[${H}]${$2.reset}`,F=`${J}${K}${$2.reset}`,_=$.scope?`${$2.bold}[${$.scope}]${$2.reset} `:"",O=`${W} ${F} ${_}${$.message}`;if($.context&&Object.keys($.context).length>0)Z(O,bJ($.context));else Z(O)},SJ={debug:"[DEBUG]",info:"[INFO]",warn:"[WARN]",error:"[ERROR]"};function EJ($){return Object.entries($).map(([J,Z])=>`${J}=${vJ(Z)}`).join(" ")}function vJ($){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 yJ=($)=>{let J=[];if(J.push($.timestamp.toISOString()),J.push(SJ[$.level]),$.scope)J.push(`[${$.scope}]`);if(J.push($.message),$.context&&Object.keys($.context).length>0)J.push(EJ($.context));console.log(J.join(" "))},fJ=($)=>{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))},gJ={debug:"color: gray",info:"color: #0ea5e9",warn:"color: #f59e0b",error:"color: #ef4444; font-weight: bold"},dJ={debug:"log",info:"info",warn:"warn",error:"error"},mJ=($)=>{let J=dJ[$.level],Z=gJ[$.level],H=`%c${$.scope?`[${$.scope}]`:""} ${$.message}`;if($.context&&Object.keys($.context).length>0)console[J](H,Z,$.context);else console[J](H,Z)},n8=["debug","info","warn","error"];function lJ($){switch($){case"terminal":return hJ;case"ci":return yJ;case"production":return fJ;case"browser":return mJ;case"test":return()=>{}}}function uJ(){if(typeof process<"u"&&process.env?.DEBUG)return"debug";return"info"}function cJ($,J){return n8.indexOf($)>=n8.indexOf(J)}class d2{scope;minLevel;environment;formatter;defaultContext;constructor($={}){this.scope=$.scope,this.minLevel=$.minLevel??uJ(),this.defaultContext=$.defaultContext??{},this.environment=$.environment??RJ(),this.formatter=lJ(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"&&!TJ(this.scope))return;if(!cJ($,this.minLevel))return;let K={level:$,message:J,scope:this.scope,context:Z||Object.keys(this.defaultContext).length>0?{...this.defaultContext,...Z}:void 0,timestamp:new Date};this.formatter(K)}}function Q9($={}){return new d2($)}function pJ(){try{let $=typeof process>"u"?void 0:process.env.DEBUG;return $==="1"||$==="true"}catch{return!1}}var j2=Q9({scope:"auth",minLevel:pJ()?"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 j2.debug("Using cached token"),this.accessToken;if(this.pendingRequest)return j2.debug("Waiting for in-flight token request"),this.pendingRequest;this.pendingRequest=this.fetchToken();try{return await this.pendingRequest}finally{this.pendingRequest=null}}async fetchToken(){j2.debug("Fetching new access token...");let{clientId:$,clientSecret:J}=this.config.credentials,Z=btoa(`${$}:${J}`),K=performance.now(),H=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()-K);if(!H.ok)throw j2.error(`Token request failed: ${H.status} ${H.statusText}`),Error(`Failed to obtain access token: ${H.status} ${H.statusText}`);let F=await H.json();return this.accessToken=F.access_token,this.tokenExpiry=Date.now()+(F.expires_in-60)*1000,j2.debug(`Token acquired (${W}ms, expires in ${F.expires_in}s)`),this.accessToken}invalidate(){j2.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"}},aJ={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:aJ};function iJ($){return"env"in $&&!("baseUrl"in $)&&!("services"in $)}function sJ($){return"baseUrl"in $&&!("services"in $)}function oJ($){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 N2{platform;env;auth;timeout;_endpoints;_authUrl;_tokenScope;_pathProfiles;_tokenManagers=new Map;constructor($){if(this.timeout=$.timeout??30000,iJ($)){this.auth=$.auth;let J=$.platform??W9,Z=$.env;this.platform=J,this.env=Z;let K=AJ[J];if(!K)throw Error(`Unknown platform: ${J}`);this._authUrl=K.token[Z],this._tokenScope=K.tokenScope??void 0,this._pathProfiles=G9[J]??l2,this._endpoints={oneroster:{baseUrl:K.api[Z],authUrl:this._authUrl},edubridge:{baseUrl:K.api[Z],authUrl:this._authUrl},powerpath:{baseUrl:K.api[Z],authUrl:this._authUrl},clr:{baseUrl:K.api[Z],authUrl:this._authUrl},case:{baseUrl:K.api[Z],authUrl:this._authUrl},caliper:{baseUrl:K.caliper[Z],authUrl:this._authUrl},webhooks:{baseUrl:K.caliper[Z],authUrl:this._authUrl},reporting:{baseUrl:K.api[Z],authUrl:this._authUrl},qti:{baseUrl:K.qti[Z],authUrl:this._authUrl}}}else if(sJ($))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(oJ($)){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 K=this._tokenManagers.get(Z);if(!K)K=new m2({tokenUrl:Z,credentials:{clientId:this.auth.clientId,clientSecret:this.auth.clientSecret},scope:this._tokenScope}),this._tokenManagers.set(Z,K);return K}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 F2(){let $=b("DEBUG");return $==="1"||$==="true"}function nJ(){return Math.random().toString(16).slice(2,10)}function C8($){return Q9({scope:$,minLevel:F2()?"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 rJ($){if(typeof $==="string")return $;if($.length===0)throw Error(`Missing env var key: ${$}`);return $[0]}function O2($){return rJ($)}function B9($){if($!=="staging"&&$!=="production")throw Error(`Invalid env "${$}": must be "staging" or "production"`);return $}function _9($,J){let Z=$?.clientId??b(J.clientId),K=$?.clientSecret??b(J.clientSecret);if(!Z)throw Error(`Missing clientId: provide in config or set ${O2(J.clientId)}`);if(!K)throw Error(`Missing clientSecret: provide in config or set ${O2(J.clientSecret)}`);return{clientId:Z,clientSecret:K}}function tJ($,J,Z){if(!Z)return;return _9($,J)}function eJ($){let J=b($.baseUrl),Z=b($.authUrl),K=b($.clientId),H=b($.clientSecret);if(J===void 0&&K===void 0)return`Missing env: provide in config or set ${O2($.env??$.baseUrl)}`;let W=[];if(J===void 0)W.push(O2($.env??$.baseUrl));if(J!==void 0&&Z===void 0)W.push(O2($.authUrl));if(K===void 0)W.push(O2($.clientId));if(H===void 0)W.push(O2($.clientSecret));return`Missing environment variables: ${W.join(", ")}`}var $Z={name:"transport",matches($){return"transport"in $&&!!$.transport},resolve($){return{mode:"transport",transport:$.transport}}},JZ={name:"provider",matches($){return"provider"in $&&!!$.provider},resolve($){return{mode:"provider",provider:$.provider}}},ZZ={name:"env-config",matches($){return"env"in $},resolve($,J,Z){let K=$,H=B9(K.env),W=_9(K.auth,J),F=K.platform??Z.defaultPlatform,_=Z.templates[F];if(!_){let w=Object.keys(Z.templates).join(", ");throw Error(`Unknown platform "${F}": available platforms are ${w}`)}let O=_[H];if(!O){let w=Object.keys(_).join(", ");throw Error(`Unknown env "${H}" for platform "${F}": available environments are ${w}`)}return{mode:"provider",provider:new N2({platform:O.platform,env:O.env,auth:W,timeout:K.timeout})}}},KZ={name:"explicit-config",matches($){return"baseUrl"in $},resolve($,J){let Z=$,K=Z.authUrl??Z.auth?.authUrl,H=!!K,W=tJ(Z.auth,J,H);return{mode:"provider",provider:new N2({baseUrl:Z.baseUrl,authUrl:K,auth:W,timeout:Z.timeout,pathProfile:Z.pathProfile,paths:Z.paths})}}},HZ={name:"env-fallback-platform",matches($,J){if(Object.keys($).length>0)return!1;let Z=J.env?b(J.env):void 0,K=b(J.clientId),H=b(J.clientSecret);return Z!==void 0&&K!==void 0&&H!==void 0},resolve($,J,Z){let K=B9(b(J.env)),H=b(J.clientId),W=b(J.clientSecret),F=Z.defaultPlatform,_=Z.templates[F]?.[K];if(!_)throw Error(`Unknown env "${K}" for platform "${F}"`);return{mode:"provider",provider:new N2({platform:_.platform,env:_.env,auth:{clientId:H,clientSecret:W}})}}},XZ={name:"env-fallback-explicit",matches($,J){if(Object.keys($).length>0)return!1;let Z=b(J.baseUrl),K=b(J.authUrl),H=b(J.clientId),W=b(J.clientSecret);return Z!==void 0&&K!==void 0&&H!==void 0&&W!==void 0},resolve($,J){let Z=b(J.baseUrl),K=b(J.authUrl),H=b(J.clientId),W=b(J.clientSecret);return{mode:"provider",provider:new N2({baseUrl:Z,authUrl:K,auth:{clientId:H,clientSecret:W}})}}},WZ={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.")}},YZ=[$Z,JZ,ZZ,WZ,KZ,HZ,XZ];function F9($,J,Z=u2){for(let K of YZ)if(K.matches($,J))return K.resolve($,J,Z);throw Error(eJ(J))}var M8=3,QZ=[429,503],t8=1000;class A8{config;log;constructor($){let{config:J,logger:Z}=$,K=J.fetch??globalThis.fetch.bind(globalThis),H;if("tokenProvider"in J&&J.tokenProvider)H=J.tokenProvider;else if("auth"in J&&J.auth)H=new m2({tokenUrl:J.auth.authUrl,credentials:{clientId:J.auth.clientId,clientSecret:J.auth.clientSecret},fetch:K});let W=J.baseUrl.endsWith("/")?J.baseUrl:`${J.baseUrl}/`;this.config={baseUrl:W,timeout:J.timeout??30000,fetch:K,tokenProvider:H},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(DJ(Z)&&Z.statusCode===404)return!1;throw Z}}async requestRaw($,J={}){let{method:Z="GET",params:K,body:H,headers:W={}}=J,F=this.buildUrl($,K),_=J.requestId??nJ(),w=Date.now()+this.config.timeout;for(let j=0;j<M8;j++){let A=w-Date.now();if(A<=0)throw this.log.error("Request timeout before attempt",F2()?{requestId:_,path:$}:{path:$}),new s("Request timeout",408);let C=await this.getAccessToken(),q=j===M8-1,z=performance.now();this.log.debug(`→ ${Z} ${F}`,{requestId:_,attempt:j>0?j+1:void 0});let N={"Content-Type":"application/json",Accept:"application/json","X-Request-ID":_,...W};if(C)N.Authorization=`Bearer ${C}`;let I=await this.config.fetch(F,{method:Z,headers:N,body:H?JSON.stringify(H):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:_}),QZ.includes(I.status)&&!q){let y2=I.headers.get("Retry-After"),p=this.parseRetryAfter(y2,j),s8=w-Date.now();if(p>=s8)throw this.log.error("Request timeout during retry backoff",F2()?{requestId:_,path:$}:{path:$}),new s("Request timeout",408);this.log.warn(`Retrying in ${p}ms (attempt ${j+1}/${M8})`,{...F2()&&{requestId:_},status:I.status}),await this.sleep(p);continue}if(!I.ok)return this.handleErrorResponse(I,_);return I}throw this.log.error("Max retries exceeded",F2()?{requestId:_,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):$,K=new URL(Z,this.config.baseUrl);if(J){for(let[H,W]of Object.entries(J))if(W!==void 0)K.searchParams.set(H,String(W))}return K.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 K=J.length>200?J.slice(0,200)+"...":J,H=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:K,error:H}),new s(`Invalid JSON response from ${W}`,$.status,{parseError:H,body:K})}}async handleErrorResponse($,J){let Z,K=await $.text().catch(()=>"");if(K)try{Z=JSON.parse(K)}catch(W){let F=W instanceof Error?W.message:"Unknown parse error";Z={rawBody:K.slice(0,500),parseError:F},this.log.warn("Failed to parse error response as JSON",{...F2()&&{requestId:J},url:$.url,status:$.status,parseError:F,bodyPreview:K.slice(0,200)})}let H=this.extractErrorMessage(Z,$.statusText);if($.status!==404)this.log.error(`Request failed: ${$.status} ${H}`,F2()?{requestId:J}:void 0);switch($.status){case 401:throw this.config.tokenProvider?.invalidate?.(),new $9(H,Z);case 403:throw new J9(H,Z);case 404:throw new Z9(H,Z);case 422:throw new K9(H,Z);default:throw new s(H,$.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 K=Date.parse($);if(!isNaN(K)){let H=K-Date.now();return Math.max(0,H)}return t8*Math.pow(2,J)}}function GZ($){if($ instanceof Date)return $.toISOString();if(typeof $==="boolean")return $?"true":"false";if(typeof $==="number")return String($);return $.replaceAll("'","''")}function t($){let J=GZ($);return typeof $==="string"||$ instanceof Date?`'${J}'`:J}function BZ($,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,K=[];if(Z.ne!==void 0)K.push(`${$}!=${t(Z.ne)}`);if(Z.gt!==void 0)K.push(`${$}>${t(Z.gt)}`);if(Z.gte!==void 0)K.push(`${$}>=${t(Z.gte)}`);if(Z.lt!==void 0)K.push(`${$}<${t(Z.lt)}`);if(Z.lte!==void 0)K.push(`${$}<=${t(Z.lte)}`);if(Z.contains!==void 0)K.push(`${$}~${t(Z.contains)}`);if(Z.in!==void 0&&Z.in.length>0){let H=Z.in.map((F)=>`${$}=${t(F)}`),W=H.join(" OR ");K.push(H.length>1?`(${W})`:W)}if(Z.notIn!==void 0&&Z.notIn.length>0){let H=Z.notIn.map((W)=>`${$}!=${t(W)}`);K.push(H.join(" AND "))}return K}return[]}function _Z($){return"OR"in $&&Array.isArray($.OR)}function O9($){if(_Z($)){let Z=$.OR.map((K)=>O9(K)).filter((K)=>K!==void 0);return Z.length>0?Z.join(" OR "):void 0}let J=[];for(let[Z,K]of Object.entries($))if(K!==void 0)J.push(...BZ(Z,K));return J.length>0?J.join(" AND "):void 0}var e8=100,FZ=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:K,offset:H,limit:W,...F}=this.params,_={...F,filter:Z?O9(Z):void 0,fields:K?.join(","),limit:$};if(this.paginationStyle==="page"){let O=Math.floor(J/$)+1;return{..._,page:O}}return{..._,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,K){if(J===0)return!1;let H=$.hasMore,W=$.total!==void 0&&Z+J<$.total;return H||W||J===K&&$.total===void 0}async*[Symbol.asyncIterator](){let $=this.params.offset??0,J=this.params.limit??e8,Z=!0,K=1,H=0;while(Z){this.log.debug(`Fetching page ${K} (offset: ${$}, limit: ${J})`);let W=await this.fetcher(this.path,{params:this.buildRequestParams(J,$)}),F=this.extractItems(W.data,K);this.log.debug(`Page ${K}: ${F.length} items`);for(let _ of F)if(yield this.transform?this.transform(_):_,H++,this.max!==void 0&&H>=this.max){this.log.debug(`Pagination complete: reached max of ${this.max} items`);return}Z=this.hasMorePages(W,F.length,$,J),$+=F.length,K++}this.log.debug(`Pagination complete: ${H} total items`)}async toArray($={}){let J=$.maxItems??FZ,Z=[];for await(let K 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(K)}return Z}async firstPage(){let $=this.params.limit??e8,J=this.params.offset??0,Z=await this.fetcher(this.path,{params:this.buildRequestParams($,J)}),K=this.extractItems(Z.data,1),H=this.transform?K.map(this.transform):K,W=this.hasMorePages(Z,H.length,J,$);return this.log.debug(`First page: ${H.length} items, total: ${Z.total}, hasMore: ${W}`),{data:H,hasMore:W,total:Z.total,nextOffset:W?J+H.length:void 0}}}function g2($,J){return{path:$,message:J}}function o($,J,Z){let K=$.safeParse(J);if(K.success)return K.data;let H=K.error.issues.map((W)=>({path:W.path.join(".")||"(root)",message:W.message}));throw R2(`Invalid ${Z} data`,H)}function c2($,J){if(typeof $!=="string"||$.trim()==="")throw R2(`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 R2("Invalid list parameters",J)}var x9={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"]},V2="http://purl.imsglobal.org/ctx/caliper/v1p2",D9="QUESTION_RESULT";import{z as L}from"zod/v4";import{z as Y}from"zod/v4";import{z as J2}from"zod/v4";import{z as U}from"zod/v4";import{z as D}from"zod/v4";import{z as V}from"zod/v4";import{z as x}from"zod/v4";import{z as f}from"zod/v4";import{z as G}from"zod/v4";import{z as B}from"zod/v4";import{z as Q}from"zod/v4";import{z as C2}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:K,pagination:H}=Z,F=(J.params?.offset??0)+K.length<H.total;return{data:K,hasMore:F,total:H.total}}}class q9 extends z8{constructor($,J,Z={}){P8(Z);let{max:K,...H}=Z;super({fetcher:(W,F)=>$.requestPaginated(W,F),path:J,params:H,max:K,logger:S})}}function T8($){let J=$.id??`urn:uuid:${crypto.randomUUID()}`,Z=$.metricsId??`urn:uuid:${crypto.randomUUID()}`;return{"@context":V2,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":V2,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 OZ($){return{"@context":V2,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 xZ($){return{"@context":V2,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 DZ($){let{id:J,...Z}=$.generated;return{"@context":V2,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:D9},...$.session===void 0?{}:{session:$.session},...$.extensions===void 0?{}:{extensions:$.extensions}}}var wZ="http://purl.imsglobal.org/ctx/caliper/v1p2",UZ="urn:tag:auto-attach",X2={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"},MZ=new Set(["ActivityEvent","TimeSpentEvent","AssignableEvent","ViewEvent"]);class C9{transformEnvelope($){let{data:J}=$;if(!Array.isArray(J))return $;let Z=J.map((K)=>x2(K)?LZ(K):K);return{...$,data:Z}}}function LZ($){let J={...$};if(!("@context"in J))J["@context"]=wZ;if(J.profile==="TimebackProfile")J.profile="AggregationProfile";if(x2(J.actor)&&J.actor.type==="TimebackUser")J.actor=jZ(J.actor);if(x2(J.object))J.object=VZ(J.object,J.type);if(x2(J.generated)){let Z=J.type;if(Z==="ActivityEvent")J.generated=kZ(J.generated);else if(Z==="TimeSpentEvent")J.generated=qZ(J.generated)}if(!J.session)J.session=UZ;return J}function jZ($){return{type:X2.PERSON,id:$.id}}function VZ($,J){if(!($.type==="TimebackActivityContext"&&J!==void 0&&MZ.has(J)))return{...$};let K={type:X2.ASSIGNABLE_DIGITAL_RESOURCE,id:$.id};if($.name)K.name=$.name;if(x2($.course)){let H={type:X2.COURSE_OFFERING,id:$.course.id};if($.course.name)H.name=$.course.name;K.isPartOf=H}if(x2($.app))K.extensions={app:{name:$.app.name,type:$.app.type??X2.SOFTWARE_APPLICATION}};return K}function A9($,J,Z,K){let W={id:typeof $?.id==="string"?$.id:`urn:uuid:${crypto.randomUUID()}`,type:J};if(Array.isArray($.items))W.items=$.items.map((F)=>{if(!x2(F))return F;let _=typeof F.type==="string"?F.type:"",O=_?_[0].toUpperCase()+_.slice(1):K;return{id:`urn:uuid:${crypto.randomUUID()}`,type:Z,metricType:O,metricValue:F.value}});for(let[F,_]of Object.entries($))if(F!=="id"&&F!=="type"&&F!=="items")W[F]=_;return W}function kZ($){return A9($,X2.ACTIVITY_METRICS_COLLECTION,X2.ACTIVITY_METRIC,"")}function qZ($){return A9($,X2.TIME_SPENT_METRICS_COLLECTION,X2.TIME_SPENT_METRIC,"Active")}function x2($){return typeof $==="object"&&$!==null&&!Array.isArray($)}function CZ($,J=u2){return F9($,x9,J)}var AZ=/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(?:\.\d+)?(?:Z|[+-]\d{2}:\d{2})$/,v=L.string().min(1).regex(AZ,"must be a valid ISO 8601 datetime"),q2=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),Z2=L.enum(["Reading","Language","Vocabulary","Social Studies","Writing","Science","FastMath","Math","None","Other"]).meta({id:"TimebackSubject",description:"Subject area"}),n=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)]).meta({id:"TimebackGrade",description:"Grade level (-1 = Pre-K, 0 = K, 1-12 = grades, 13 = AP)"}),zZ=L.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),PZ=L.enum(["department","school","district","local","state","national"]),z9=L.enum(["administrator","aide","guardian","parent","proctor","relative","student","teacher"]),TZ=L.enum(["administrator","proctor","student","teacher"]),RZ=L.enum(["qti","text","audio","video","interactive","visual","course-material","assessment-bank"]),NZ=L.enum(["qti-test","qti-question","qti-stimulus","qti-test-bank"]),IZ=L.enum(["unit","course","resource-collection"]),bZ=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"]),hZ=L.enum(["easy","medium","hard"]),SZ=L.array(L.object({source:L.string(),learningObjectiveIds:L.array(L.string())})),EZ=L.object({consecutive_failures:L.number().int().min(1).optional(),stagnation_limit:L.number().int().min(1).optional()}).optional(),Q1=L.enum(["powerpath-100","quiz","test-out","placement","unit-test","alpha-read-article"]).nullable(),G1=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()}),vZ=Y.union([Y.url(),Y.string().regex(/^urn:/,"Must be a URL or URN")]),D2=Y.object({id:Y.url(),type:Y.literal("TimebackUser"),email:Y.string()}),N8=Y.object({id:Y.url(),type:Y.literal("TimebackActivityContext"),subject:Z2,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()}),P9=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:v,profile:Y.literal("TimebackProfile"),actor:D2,action:Y.string(),object:N8,edApp:Y.object({id:Y.string(),name:Y.string().optional()}).optional()}),T9=Y.object({type:Y.enum(["totalQuestions","correctQuestions","xpEarned","masteredUnits"]),value:Y.number()}),yZ=Y.object({id:Y.url(),type:Y.literal("TimebackActivityMetricsCollection"),attempt:Y.number().optional(),items:Y.array(T9),extensions:Y.record(Y.string(),Y.unknown()).optional()}),fZ=P9.extend({type:Y.literal("ActivityEvent"),action:Y.literal("Completed"),generated:yZ}),R9=Y.object({type:Y.enum(["active","inactive","waste","unknown","anti-pattern"]),value:Y.number(),subType:Y.string().optional()}),gZ=Y.object({id:Y.url(),type:Y.literal("TimebackTimeSpentMetricsCollection"),items:Y.array(R9)}),dZ=P9.extend({type:Y.literal("TimeSpentEvent"),action:Y.literal("SpentTime"),generated:gZ}),mZ=Y.object({actor:D2,object:N8,metrics:Y.array(T9).min(1,"metrics must contain at least one metric"),eventTime:v.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(),lZ=Y.object({actor:D2,object:N8,metrics:Y.array(R9).min(1,"metrics must contain at least one metric"),eventTime:v.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(),uZ=Y.union([fZ,dZ]),_1=Y.object({sensor:Y.string(),sendTime:v,dataVersion:Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),data:Y.array(uZ)}),cZ=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:vZ,type:Y.string(),name:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),N9=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,D2]),action:Y.string(),object:l,eventTime:v,profile:cZ,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(),w9=Y.object({sensor:X,sendTime:v,dataVersion:Y.literal("http://purl.imsglobal.org/ctx/caliper/v1p2"),data:Y.array(N9).min(1,"data must contain at least one event")}),pZ=Y.object({sensor:X,events:Y.array(N9).min(1,"events must contain at least one event")}),U9=Y.object({limit:Y.number().int().positive().optional(),offset:Y.number().int().min(0).optional(),sensor:X.optional(),startDate:v.optional(),endDate:v.optional(),actorId:X.optional(),actorEmail:Y.email().optional()}).strict(),I9=Y.object({id:Y.string(),name:Y.string().optional(),isPartOf:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),aZ=Y.object({id:Y.string(),attempt:Y.union([Y.string(),Y.record(Y.string(),Y.unknown())]).optional(),startedAtTime:v.optional(),endedAtTime:v.optional(),duration:Y.string().optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}),iZ=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()}),sZ=Y.object({actor:Y.union([Y.string(),p2,D2]),object:I9,edApp:l,id:Y.string().optional(),eventTime:v.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),oZ=Y.object({actor:Y.union([Y.string(),p2,D2]),object:I9,edApp:l,generated:aZ.optional(),id:Y.string().optional(),eventTime:v.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),nZ=Y.object({actor:Y.union([Y.string(),p2,D2]),object:Y.string(),generated:iZ,edApp:l,id:Y.string().optional(),eventTime:v.optional(),session:l.optional(),extensions:Y.record(Y.string(),Y.unknown()).optional()}).strict(),O1=J2.object({name:X,targetUrl:J2.url("targetUrl must be a valid URL"),secret:X,active:J2.boolean(),sensor:J2.string().nullable().optional(),description:J2.string().nullable().optional()}).strict(),rZ=J2.enum(["string","number","boolean"]),tZ=J2.enum(["eq","neq","gt","gte","lt","lte","contains","notContains","in","notIn","startsWith","endsWith","regexp"]),x1=J2.object({webhookId:X,filterKey:X,filterValue:X,filterType:rZ,filterOperator:tZ,active:J2.boolean()}).strict(),a2=U.string().uuid(),M9=U.object({title:X,identifier:a2,uri:X}),eZ=U.object({identifier:a2,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()}),$5=U.object({identifier:a2,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()}),J5=U.object({identifier:a2,uri:X,lastChangeDateTime:X,associationType:X,originNodeURI:M9,destinationNodeURI:M9,sequenceNumber:U.number().optional(),extensions:U.unknown().optional()}),Z5=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()}),w1=U.object({CFDocument:eZ,CFItems:U.array($5),CFAssociations:U.array(J5),CFDefinitions:Z5.optional(),extensions:U.unknown().optional()}),L9="https://www.w3.org/ns/credentials/v2",K5=D.array(D.url()).min(3,"@context must include W3C, CLR, and OB context URLs").refine(($)=>$[0]===L9,`First @context entry must be "${L9}"`),b9=D.object({id:D.url(),type:D.literal("Image"),caption:D.string().optional()}),I8=D.object({id:D.url(),type:D.array(D.string()).min(1).refine(($)=>$.includes("Profile"),'type must include "Profile"'),name:D.string().optional(),url:D.string().url().optional(),phone:D.string().optional(),description:D.string().optional(),image:b9.optional(),email:D.string().email().optional()}),h9=D.object({type:X,proofPurpose:X,verificationMethod:X,created:D.iso.datetime(),proofValue:X,cryptosuite:D.string().optional()}),H5=D.object({id:X,type:D.literal("1EdTechJsonSchemaValidator2019")}),X5=D.object({id:D.string().url().optional(),narrative:D.string().optional()}),W5=D.object({id:D.url(),type:D.array(D.string()).min(1).refine(($)=>$.includes("Achievement"),'type must include "Achievement"'),name:X,description:X,criteria:X5,image:b9.optional(),achievementType:D.string().optional(),creator:I8.optional()}),Y5=D.object({type:D.literal("IdentityObject"),identityHash:X,identityType:X,hashed:D.boolean(),salt:D.string().optional()}),Q5=D.enum(["exactMatchOf","isChildOf","isParentOf","isPartOf","isPeerOf","isRelatedTo","precedes","replacedBy"]),G5=D.object({type:D.literal("Association"),associationType:Q5,sourceId:D.url(),targetId:D.url()}),B5=D.object({"@context":D.array(D.string()).min(1),id:D.url(),type:D.array(D.string()).min(1).refine(($)=>$.includes("VerifiableCredential"),'type must include "VerifiableCredential"'),issuer:I8,validFrom:D.iso.datetime(),validUntil:D.string().datetime().optional(),credentialSubject:D.object({id:D.string().optional()}),proof:D.array(h9).optional()}),_5=D.object({id:D.string().url().optional(),type:D.array(D.string()).min(1).refine(($)=>$.includes("ClrSubject"),'type must include "ClrSubject"'),identifier:D.array(Y5).optional(),achievement:D.array(W5).optional(),verifiableCredential:D.array(B5).min(1),association:D.array(G5).optional()}),M1=D.object({"@context":K5,id:D.url(),type:D.array(D.string()).min(1).refine(($)=>$.includes("VerifiableCredential")&&$.includes("ClrCredential"),'type must include "VerifiableCredential" and "ClrCredential"'),issuer:I8,name:X,description:D.string().optional(),validFrom:D.iso.datetime(),validUntil:D.iso.datetime().optional(),credentialSubject:_5,proof:D.array(h9).optional(),credentialSchema:D.array(H5).optional()}),F5=V.object({staging:V.string().meta({description:"Course ID in staging environment"}).optional(),production:V.string().meta({description:"Course ID in production environment"}).optional()}).meta({id:"CourseIds",description:"Environment-specific course IDs (populated by sync)"}),O5=V.enum(["base","hole-filling","optional"]).meta({id:"CourseType",description:"Course classification type"}),x5=V.enum(["draft","testing","published","deactivated"]).meta({id:"PublishStatus",description:"Course publication status"}),D5=V.object({dailyXp:V.number().int().positive().meta({description:"Target XP to earn per day"}).optional(),dailyLessons:V.number().int().positive().meta({description:"Target lessons to complete per day"}).optional(),dailyActiveMinutes:V.number().int().positive().meta({description:"Target active learning minutes per day"}).optional(),dailyAccuracy:V.number().int().min(0).max(100).meta({description:"Target accuracy percentage (0-100)"}).optional(),dailyMasteredUnits:V.number().int().positive().meta({description:"Target units to master per day"}).optional()}).meta({id:"CourseGoals",description:"Daily learning goals for a course"}),w5=V.object({totalXp:V.number().int().positive().meta({description:"Total XP available in the course"}).optional(),totalLessons:V.number().int().positive().meta({description:"Total number of lessons/activities"}).optional(),totalGrades:V.number().int().positive().meta({description:"Total grade levels covered"}).optional()}).meta({id:"CourseMetrics",description:"Aggregate metrics for a course"}),S9=V.object({courseType:O5.optional(),isSupplemental:V.boolean().meta({description:"Whether this is supplemental to a base course"}).optional(),isCustom:V.boolean().meta({description:"Whether this is a custom course for an individual student"}).optional(),publishStatus:x5.optional(),contactEmail:V.email().meta({description:"Contact email for course issues"}).optional(),primaryApp:V.string().meta({description:"Primary application identifier"}).optional(),goals:D5.optional(),metrics:w5.optional()}).meta({id:"CourseMetadata",description:"Course metadata (matches API metadata object)"}),E9=V.object({courseCode:V.string().meta({description:"Course code (e.g., 'MATH101')"}).optional(),level:V.string().meta({description:"Course level (e.g., 'AP', 'Honors')"}).optional(),metadata:S9.optional()}).meta({id:"CourseDefaults",description:"Default properties that apply to all courses unless overridden"}),j9=V.object({level:V.string().meta({description:"Course level for this environment"}).optional(),sensor:V.url().meta({description:"Caliper sensor endpoint URL for this environment"}).optional(),launchUrl:V.url().meta({description:"LTI launch URL for this environment"}).optional(),metadata:S9.optional()}).meta({id:"CourseEnvOverrides",description:"Environment-specific course overrides (non-identity fields)"}),U5=V.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"}),M5=E9.extend({subject:Z2.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:F5.nullable().optional(),sensor:V.url().meta({description:"Caliper sensor endpoint URL for this course"}).optional(),launchUrl:V.url().meta({description:"LTI launch URL for this course"}).optional(),overrides:U5.optional()}).meta({id:"CourseConfig",description:"Configuration for a single course. Must have either grade or courseCode (or both)."}),j1=V.object({$schema:V.string().meta({description:"JSON Schema reference for editor support"}).optional(),name:V.string().min(1,"App name is required").meta({description:"Display name for your app"}),defaults:E9.meta({description:"Default properties applied to all courses"}).optional(),courses:V.array(M5).min(1,"At least one course is required").meta({description:"Courses available in this app"}),sensor:V.url().meta({description:"Default Caliper sensor endpoint URL for all courses"}).optional(),launchUrl:V.url().meta({description:"Default LTI launch URL for all courses"}).optional(),studio:V.object({telemetry:V.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((K)=>K.grade!==void 0).map((K)=>`${K.subject}:${K.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((K)=>K.courseCode!==void 0).map((K)=>K.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"]}),I2=x.union([v,q2]),b2=I2.transform(($)=>$.includes("T")?$:`${$}T00:00:00.000Z`),k1=x.object({id:x.string(),role:x.string(),beginDate:I2.nullable(),endDate:I2.nullable(),metadata:x.object({goals:x.object({dailyXp:x.number().optional()}).optional(),metrics:x.object({totalXp:x.number().optional(),totalLessons:x.number().optional()}).optional()}).optional(),course:x.object({id:x.string(),title:x.string(),subjects:x.array(x.string()).nullable(),grades:x.array(x.string()).nullable()}),school:x.object({id:x.string(),name:x.string()}),testOutSupported:x.boolean(),testOutEligible:x.boolean()}),L5=x.object({activityMetrics:x.object({xpEarned:x.number(),totalQuestions:x.number(),correctQuestions:x.number(),masteredUnits:x.number()}),timeSpentMetrics:x.object({activeSeconds:x.number(),inactiveSeconds:x.number(),wasteSeconds:x.number()}),apps:x.array(x.string())}),j5=x.record(x.string(),x.record(x.string(),L5)),q1=x.object({message:x.string(),enrollmentId:x.string(),startDate:I2,endDate:I2,facts:j5,factsByApp:x.unknown()}),v9=x.object({email:x.email().optional(),studentId:X.optional()}).superRefine(($,J)=>{if(!$.email&&!$.studentId)J.addIssue({code:x.ZodIssueCode.custom,message:"must provide either email or studentId",path:["email"]}),J.addIssue({code:x.ZodIssueCode.custom,message:"must provide either email or studentId",path:["studentId"]})}),C1=x.object({subject:X,grade:X,courseId:X,orgSourcedId:X.optional()}),A1=x.object({userId:X}),z1=x.object({sourcedId:X.optional(),role:TZ.optional(),beginDate:v.optional(),metadata:x.record(x.string(),x.unknown()).optional()}),P1=x.object({fields:x.string().optional(),limit:x.number().int().positive().optional(),offset:x.number().int().nonnegative().optional(),sort:x.string().optional(),orderBy:x.enum(["asc","desc"]).optional(),filter:x.string().optional(),search:x.string().optional(),roles:x.array(z9).min(1),orgSourcedIds:x.array(X).optional()}),T1=v9.extend({startDate:b2,endDate:b2,timezone:x.string().optional()}),R1=v9.extend({weekDate:b2,timezone:x.string().optional()}),N1=x.object({enrollmentId:X,startDate:b2.optional(),endDate:b2.optional(),timezone:x.string().optional()}),b1=f.object({email:f.email()}),h1=f.object({name:X.optional(),timeback_id:X.optional(),grade:X.optional(),subject:X.optional(),all:f.boolean().optional()}),S1=f.object({student_email:f.email(),timeback_id:X.optional(),subject:X.optional(),grade_rank:f.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,K=!!$.subject,H=$.grade_rank!==void 0;if(!Z&&!K)J.addIssue({code:f.ZodIssueCode.custom,message:"must provide either timeback_id or subject",path:["timeback_id"]}),J.addIssue({code:f.ZodIssueCode.custom,message:"must provide either timeback_id or subject",path:["subject"]});if(H&&!K)J.addIssue({code:f.ZodIssueCode.custom,message:"grade_rank requires subject",path:["grade_rank"]});let W=!!$.assessment_line_item_sourced_id,F=!!$.assessment_result_sourced_id;if(W!==F)J.addIssue({code:f.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"]})}),E1=f.object({student_email:f.email(),assignment_id:f.number().int().positive().optional(),timeback_id:X.optional(),subject:X.optional(),grade_rank:f.number().int().min(0).max(12).optional()}).superRefine(($,J)=>{let Z=$.assignment_id!==void 0,K=!!$.timeback_id,H=!!$.subject&&$.grade_rank!==void 0;if(!Z&&!K&&!H)J.addIssue({code:f.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:f.ZodIssueCode.custom,message:"grade_rank requires subject",path:["grade_rank"]})}),g=G.enum(["active","tobedeleted"]),y=G.record(G.string(),G.unknown()).nullable().optional(),k=G.object({sourcedId:X,type:G.string().optional(),href:G.string().optional()}).strict(),W2=G.union([q2,v]).transform(($)=>$.includes("T")?$:`${$}T00:00:00Z`),V5=G.object({roleType:G.enum(["primary","secondary"]),role:z9,org:k,userProfile:G.string().optional(),metadata:y,beginDate:W2.optional(),endDate:W2.optional()}).strict(),y1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:g.optional(),enabledUser:G.boolean(),givenName:X.describe("givenName must be a non-empty string"),familyName:X.describe("familyName must be a non-empty string"),middleName:X.optional(),username:X.optional(),email:G.email(),roles:G.array(V5).min(1,"roles must include at least one role"),userIds:G.array(G.object({type:X,identifier:X}).strict()).optional(),agents:G.array(k).optional(),grades:G.array(n).optional(),identifier:X.optional(),sms:X.optional(),phone:X.optional(),pronouns:X.optional(),password:X.optional(),metadata:y}).strict(),f1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:g.optional(),title:X.describe("title must be a non-empty string"),org:k,courseCode:X.optional(),subjects:G.array(Z2).optional(),grades:G.array(n).optional(),level:X.optional(),metadata:y}).strict(),g1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:g.optional(),title:X.describe("title must be a non-empty string"),terms:G.array(k).min(1,"terms must have at least one item"),course:k,org:k,classCode:X.optional(),classType:G.enum(["homeroom","scheduled"]).optional(),location:X.optional(),grades:G.array(n).optional(),subjects:G.array(Z2).optional(),subjectCodes:G.array(X).optional(),periods:G.array(X).optional(),metadata:y}).strict(),y9=G.enum(["true","false"]),d1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string").optional(),status:g.optional(),user:k,class:k,school:k.optional(),role:G.enum(["administrator","proctor","student","teacher"]),primary:y9.optional(),beginDate:q2.optional(),endDate:q2.optional(),metadata:y}).strict(),m1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),status:g,weight:G.number().nullable().optional(),metadata:y}).strict(),l1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),class:k,school:k,category:k,assignDate:W2,dueDate:W2,status:g,description:G.string().optional(),resultValueMin:G.number().nullable().optional(),resultValueMax:G.number().nullable().optional(),scoreScale:k.optional(),metadata:y}).strict(),u1=G.object({sourcedId:X.optional(),lineItem:k,student:k,class:k.optional(),scoreDate:W2,scoreStatus:G.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),score:G.number().nullable().optional(),textScore:G.string().nullable().optional(),status:g,comment:G.string().nullable().optional(),metadata:y}).strict(),c1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),status:g.optional(),type:G.string(),class:k,course:k.nullable().optional(),scoreScaleValue:G.array(G.object({itemValueLHS:X,itemValueRHS:X,value:G.string().optional(),description:G.string().optional()}).strict()),minScore:G.number().optional(),maxScore:G.number().optional(),metadata:y}).strict(),p1=G.object({sourcedId:X.optional(),status:g.optional(),dateLastModified:v.optional(),title:X.describe("title must be a non-empty string"),description:G.string().nullable().optional(),class:k.nullable().optional(),parentAssessmentLineItem:k.nullable().optional(),scoreScale:k.nullable().optional(),resultValueMin:G.number().nullable().optional(),resultValueMax:G.number().nullable().optional(),component:k.nullable().optional(),componentResource:k.nullable().optional(),learningObjectiveSet:G.array(G.object({source:G.string(),learningObjectiveIds:G.array(G.string())})).optional().nullable(),course:k.nullable().optional(),metadata:y}).strict(),k5=G.object({learningObjectiveId:G.string(),score:G.number().optional(),textScore:G.string().optional()}),q5=G.array(G.object({source:G.string(),learningObjectiveResults:G.array(k5)})),a1=G.object({sourcedId:X.optional(),status:g.optional(),dateLastModified:v.optional(),metadata:y,assessmentLineItem:k,student:k,score:G.number().nullable().optional(),textScore:G.string().nullable().optional(),scoreDate:W2,scoreScale:k.nullable().optional(),scorePercentile:G.number().nullable().optional(),scoreStatus:G.enum(["exempt","fully graded","not submitted","partially graded","submitted"]),comment:G.string().nullable().optional(),learningObjectiveSet:q5.nullable().optional(),inProgress:G.string().nullable().optional(),incomplete:G.string().nullable().optional(),late:G.string().nullable().optional(),missing:G.string().nullable().optional()}).strict(),i1=G.object({sourcedId:X.optional(),status:g.optional(),name:X.describe("name must be a non-empty string"),type:PZ,identifier:X.optional(),parent:k.optional(),metadata:y}).strict(),s1=G.object({sourcedId:X.optional(),status:g.optional(),name:X.describe("name must be a non-empty string"),type:G.literal("school").optional(),identifier:X.optional(),parent:k.optional(),metadata:y}).strict(),o1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string"),status:g,title:X.describe("title must be a non-empty string"),startDate:W2,endDate:W2,type:G.enum(["gradingPeriod","semester","schoolYear","term"]),schoolYear:X.describe("schoolYear must be a non-empty string"),org:k,parent:k.optional(),children:G.array(k).optional(),metadata:y}).strict(),n1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),courseComponent:k,resource:k,status:g,metadata:y}).strict(),r1=G.object({sourcedId:X.optional(),title:X.describe("title must be a non-empty string"),course:k,status:g,metadata:y}).strict(),t1=G.object({sourcedId:X.describe("sourcedId must be a non-empty string"),role:G.enum(["student","teacher"]),primary:y9.optional(),beginDate:q2.optional(),endDate:q2.optional(),metadata:y}).strict(),e1=G.object({agentSourcedId:X.describe("agentSourcedId must be a non-empty string")}).strict(),$6=G.object({applicationName:X.describe("applicationName must be a non-empty string"),credentials:G.object({username:X.describe("username must be a non-empty string"),password:X.describe("password must be a non-empty string")})}).strict(),J6=G.object({sourcedId:X.describe("sourcedId must be a non-empty string")}).loose(),Y2=G.object({type:RZ,subject:Z2.nullish(),grades:G.array(n).nullish(),language:G.string().nullish(),xp:G.number().nullish(),url:G.url().nullish(),keywords:G.array(G.string()).nullish(),learningObjectiveSet:SZ.nullish(),lessonType:G.string().nullish()}).passthrough(),C5=Y2.extend({type:G.literal("qti"),subType:NZ,questionType:bZ.optional(),difficulty:hZ.optional()}),A5=Y2.extend({type:G.literal("text"),format:G.string(),author:G.string().optional(),pageCount:G.number().optional()}),z5=Y2.extend({type:G.literal("audio"),duration:G.string().regex(/^\d{2}:\d{2}:\d{2}(\.\d{2})?$/).optional(),format:G.string(),speaker:G.string().optional()}),P5=Y2.extend({type:G.literal("video"),duration:G.string().regex(/^\d{2}:\d{2}:\d{2}(\.\d{2})?$/).optional(),captionsAvailable:G.boolean().optional(),format:G.string()}),T5=Y2.extend({type:G.literal("interactive"),launchUrl:G.url().optional(),toolProvider:G.string().optional(),instructionalMethod:G.string().optional(),courseIdOnFail:G.string().nullable().optional(),fail_fast:EZ}),R5=Y2.extend({type:G.literal("visual"),format:G.string(),resolution:G.string().optional()}),N5=Y2.extend({type:G.literal("course-material"),subType:IZ,author:G.string().optional(),format:G.string(),instructionalMethod:G.string().optional()}),I5=Y2.extend({type:G.literal("assessment-bank"),resources:G.array(G.string())}),b5=G.discriminatedUnion("type",[C5,A5,z5,P5,T5,R5,N5,I5]),Z6=G.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:G.array(G.enum(["primary","secondary"])).optional(),importance:G.enum(["primary","secondary"]).optional(),vendorId:G.string().optional(),applicationId:G.string().optional(),status:g.optional(),metadata:b5.nullable().optional()}).strict(),h5=G.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:G.number()}).loose(),K6=G.object({course:G.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:k,status:g,metadata:y}).strict(),courseStructure:G.record(G.string(),h5)}).strict(),S5=G.object({student:k}).loose(),H6=G.array(S5).min(1,"results must have at least one item"),E5=B.enum(["edulastic","mastery-track"]),v5=B.enum(["powerpath-100","quiz","test-out","placement","unit-test","alpha-read-article"]),f9=B.array(n),g9=B.record(B.string(),B.unknown()).optional(),d9=B.object({courseId:X,lessonTitle:X.optional(),launchUrl:X.optional(),toolProvider:E5,unitTitle:X.optional(),courseComponentSourcedId:X.optional(),vendorId:X.optional(),description:X.optional(),resourceMetadata:g9.nullable().optional(),grades:f9}),W6=d9.extend({lessonType:B.literal("test-out"),xp:B.number()}),Y6=d9.extend({lessonType:B.literal("placement"),courseIdOnFail:X.nullable().optional(),xp:B.number().optional()}),V9=B.object({courseId:X,lessonType:v5,lessonTitle:X.optional(),unitTitle:X.optional(),courseComponentSourcedId:X.optional(),resourceMetadata:g9.nullable().optional(),xp:B.number().optional(),grades:f9.optional(),courseIdOnFail:X.nullable().optional()}),Q6=B.discriminatedUnion("testType",[V9.extend({testType:B.literal("qti"),qti:B.object({url:B.url(),title:X.optional(),metadata:B.record(B.string(),B.unknown()).optional()})}),V9.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)})})]),G6=B.object({student:X,lesson:X}),B6=B.object({student:X,lesson:X}),_6=B.object({courseId:X,userId:X,classId:X.optional()}),k2=B.object({type:B.enum(["component","resource"]),id:X}),y5=B.union([B.object({type:B.literal("set-skipped"),payload:B.object({target:k2,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:k2,reference_id:X})}),B.object({type:B.literal("move-item-after"),payload:B.object({target:k2,reference_id:X})}),B.object({type:B.literal("move-item-to-start"),payload:B.object({target:k2})}),B.object({type:B.literal("move-item-to-end"),payload:B.object({target:k2})}),B.object({type:B.literal("change-item-parent"),payload:B.object({target:k2,new_parent_id:X,position:B.enum(["start","end"]).optional()})})]),F6=B.object({operation:y5,reason:X.optional()}),O6=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:zZ,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()})}),x6=B.object({student:X,lesson:X,applicationName:X.optional(),testId:X.optional(),skipCourseEnrollment:B.boolean().optional()}),D6=B.object({student:X,subject:Z2}),w6=B.object({student:X,lesson:X}),U6=B.object({userId:X}),M6=B.object({userId:X,subject:B.enum(["Math","Reading","Language","Science"])}),L6=B.object({student:X,subject:Z2,grade:n,testName:X.optional()}),j6=B.object({testName:X}),f5=B.object({student:X,subject:Z2,grade:n,testName:X.optional()}),V6=B.object({items:B.array(f5)}),k6=B.object({spreadsheetUrl:B.url(),sheet:X}),q6=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()}),C6=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()}),A6=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 $}),z6=B.object({student:X,lesson:X,attempt:B.number().int().positive().optional()}),P6=B.object({student:X,lesson:X}),T6=B.object({student:X,lesson:X}),R6=B.object({student:X,course:X}),N6=B.object({student:X,lesson:X,applicationName:X.optional()}),I6=B.object({student:X,subject:Z2}),b6=B.object({oneRosterSourcedId:X,subject:X}),h6=B.object({courseIds:B.array(X).min(1),rendererId:X,rendererUrl:B.url(),rendererVersion:X.optional(),suppressFeedback:B.boolean().optional(),suppressCorrectResponse:B.boolean().optional()}),S6=B.object({status:B.enum(["active","tobedeleted"]).optional()}),m9=Q.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=Q.enum(["single","multiple","ordered","record"]),h8=Q.enum(["identifier","boolean","integer","float","string","point","pair","directedPair","duration","file","uri"]),g5=Q.enum(["easy","medium","hard"]),d5=Q.enum(["linear","nonlinear"]),m5=Q.enum(["individual","simultaneous"]),S8=Q.enum(["show","hide"]),l9=Q.enum(["test","item","stimulus"]),v6=Q.enum(["QUESTION","LESSON"]),l5=Q.object({value:Q.array(Q.string())}).strict(),u9=Q.object({identifier:X,cardinality:b8,baseType:h8.optional(),correctResponse:l5}).strict(),c9=Q.object({identifier:X,cardinality:b8,baseType:h8.optional()}).strict(),p9=Q.object({identifier:X,cardinality:b8.optional(),baseType:h8,normalMaximum:Q.number().optional(),normalMinimum:Q.number().optional(),defaultValue:Q.object({value:Q.unknown().optional()}).strict().optional()}).strict(),u5=Q.object({outcomeIdentifier:X,variableIdentifier:X}).strict(),a9=Q.object({templateType:Q.enum(["match_correct","map_response"]),responseDeclarationIdentifier:X,outcomeIdentifier:X,correctResponseIdentifier:X,incorrectResponseIdentifier:X,inlineFeedback:u5.optional()}).strict(),c5=Q.object({source:X,learningObjectiveIds:Q.array(Q.string())}).strict(),E8=Q.object({subject:Q.string().optional(),grade:n.optional(),difficulty:g5.optional(),learningObjectiveSet:Q.array(c5).optional()}).loose(),i9=Q.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:Q.string(),title:Q.string()}).strict(),s9=Q.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:Q.string(),class:Q.array(Q.string())}).strict(),o9=Q.object({outcomeIdentifier:X,identifier:X,showHide:S8,content:Q.string(),class:Q.array(Q.string())}).strict(),n9=Q.object({href:X,type:X}).strict(),r9=Q.object({id:X,support:Q.string(),content:Q.string()}).strict(),y6=Q.object({page:Q.number().int().positive().optional(),limit:Q.number().int().positive().optional(),sort:Q.string().optional(),order:Q.enum(["asc","desc"]).optional()}).strict(),p5=Q.object({format:Q.string().pipe(Q.literal("xml")),xml:X,metadata:E8.optional()}).strict(),a5=Q.object({identifier:X,title:X,type:m9,qtiVersion:Q.string().optional(),timeDependent:Q.boolean().optional(),adaptive:Q.boolean().optional(),responseDeclarations:Q.array(u9).optional(),outcomeDeclarations:Q.array(c9).optional(),responseProcessing:a9.optional(),metadata:E8.optional(),modalFeedback:Q.array(i9).optional(),feedbackInline:Q.array(s9).optional(),feedbackBlock:Q.array(o9).optional()}).strict(),f6=Q.union([p5,a5]),g6=Q.object({identifier:X.optional(),title:X,type:m9,qtiVersion:Q.string().optional(),timeDependent:Q.boolean().optional(),adaptive:Q.boolean().optional(),responseDeclarations:Q.array(u9).optional(),outcomeDeclarations:Q.array(c9).optional(),responseProcessing:a9.optional(),metadata:E8.optional(),modalFeedback:Q.array(i9).optional(),feedbackInline:Q.array(s9).optional(),feedbackBlock:Q.array(o9).optional(),rawXml:Q.string(),content:Q.record(Q.string(),Q.unknown())}).strict(),d6=Q.object({identifier:X,response:Q.union([Q.string(),Q.array(Q.string())])}).strict(),t9=Q.object({identifier:X,href:X,sequence:Q.number().int().positive().optional()}).strict(),i5=Q.object({identifier:X,title:X,visible:Q.boolean(),required:Q.boolean().optional(),fixed:Q.boolean().optional(),sequence:Q.number().int().positive(),"qti-assessment-item-ref":Q.array(t9).optional()}).strict(),e9=Q.object({identifier:X,navigationMode:Q.string().pipe(d5),submissionMode:Q.string().pipe(m5),"qti-assessment-section":Q.array(i5)}).strict(),m6=Q.object({items:Q.array(t9).min(1)}).strict(),l6=Q.object({metadata:Q.record(Q.string(),Q.unknown()).optional()}).strict(),u6=Q.object({identifier:X,title:X,qtiVersion:Q.string().optional(),toolName:Q.string().optional(),toolVersion:Q.string().optional(),timeLimit:Q.number().optional(),maxAttempts:Q.number().optional(),toolsEnabled:Q.record(Q.string(),Q.boolean()).optional(),metadata:Q.record(Q.string(),Q.unknown()).optional(),"qti-test-part":Q.array(e9),"qti-outcome-declaration":Q.array(p9).optional()}).strict(),c6=Q.object({identifier:X.optional(),title:X,qtiVersion:Q.string().optional(),toolName:Q.string().optional(),toolVersion:Q.string().optional(),timeLimit:Q.number().optional(),maxAttempts:Q.number().optional(),toolsEnabled:Q.record(Q.string(),Q.boolean()).optional(),metadata:Q.record(Q.string(),Q.unknown()).optional(),"qti-test-part":Q.array(e9),"qti-outcome-declaration":Q.array(p9).optional()}).strict(),p6=Q.object({identifier:X,title:X,label:Q.string().optional(),language:Q.string().optional(),stylesheet:n9.optional(),content:Q.string(),catalogInfo:Q.array(r9).optional(),toolName:Q.string().optional(),toolVersion:Q.string().optional(),metadata:Q.record(Q.string(),Q.unknown()).optional()}).strict(),a6=Q.object({identifier:X.optional(),title:X,label:Q.string().optional(),language:Q.string().optional(),stylesheet:n9.optional(),content:Q.string(),catalogInfo:Q.array(r9).optional(),toolName:Q.string().optional(),toolVersion:Q.string().optional(),metadata:Q.record(Q.string(),Q.unknown()).optional()}).strict(),i6=Q.object({xml:Q.string().optional(),schema:l9,entityId:Q.string().optional()}).strict(),s6=Q.object({xml:Q.array(Q.string()),schema:l9,entityIds:Q.array(Q.string())}).strict(),o6=Q.object({questionId:Q.string().optional(),userId:X,feedback:X,lessonId:X,humanApproved:Q.boolean().optional()}).strict(),s5=C2.enum(["creator_only","same_org","all_orgs"]),o5=C2.string().uuid(),r6=C2.email(),t6=C2.object({sql:X}),e6=C2.object({endpointName:X,sql:X,authMode:s5,description:X.optional()}),$7=C2.object({endpointId:o5});class $${transport;eventTransformer;constructor($,J){this.transport=$,this.eventTransformer=J?.eventTransformer}send($,J){o(pZ,{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(w9,$,"caliper envelope"),S.debug("Sending Caliper envelope",{sensor:$.sensor,eventCount:$.data.length});let J=this.eventTransformer?this.eventTransformer.transformEnvelope($):$,K=(await this.transport.request(this.transport.paths.send,{method:"POST",body:J}))?.jobId;return S.debug("Events sent",{jobId:K??"(none)"}),{jobId:K}}async validate($){let J=this.transport.paths.validate;if(!J)throw Error("validate() is not supported on this platform");o(w9,$,"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(U9,$,"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 K=await this.transport.request(J,{params:Z});return{events:K.events??[],pagination:K.pagination}}stream($={}){let J=this.transport.paths.list;if(!J)throw Error("stream() is not supported on this platform");let{max:Z,...K}=$;o(U9,K,"list events params"),P8($),S.debug("Streaming events",{params:$});let H={};if(K.limit!==void 0)H.limit=K.limit;if(K.offset!==void 0)H.offset=K.offset;if(K.sensor)H.sensor=K.sensor;if(K.startDate)H.startDate=K.startDate;if(K.endDate)H.endDate=K.endDate;if(K.actorId)H.actorId=K.actorId;if(K.actorEmail)H.actorEmail=K.actorEmail;return new q9(this.transport,J,{...H,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($)),K=await this.transport.request(Z);if(!K.event)throw Error(`Event not found: ${$}`);return K.event}sendActivity($,J){o(mZ,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(lZ,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(sZ,J,"question seen event");let Z=OZ(J);return S.debug("Sending AssessmentItemEvent.Started",{eventId:Z.id,actor:J.actor,object:J.object.id}),this.send($,[Z])}sendQuestionAnswered($,J){o(oZ,J,"question answered event");let Z=xZ(J);return S.debug("Sending AssessmentItemEvent.Completed",{eventId:Z.id,actor:J.actor,object:J.object.id}),this.send($,[Z])}sendQuestionGraded($,J){o(nZ,J,"question graded event");let Z=DZ(J);return S.debug("Sending GradeEvent.Graded",{eventId:Z.id,actor:J.actor,object:J.object,scoreGiven:J.generated.scoreGiven}),this.send($,[Z])}}class J${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"),n5(J);let{timeoutMs:Z=30000,pollIntervalMs:K=1000}=J,H=Date.now();S.debug("Waiting for job completion",{jobId:$,timeoutMs:Z,pollIntervalMs:K});while(Date.now()-H<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((F)=>{setTimeout(F,K)})}throw Error(`Job ${$} timed out after ${Z}ms`)}}function n5($){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 R2("Invalid polling options",J)}function r5($=u2){return class{transport;_provider;events;jobs;constructor(Z={}){let K=CZ(Z,$),H;if(K.mode==="transport")this.transport=K.transport,S.info("Client initialized with custom transport");else{let{provider:W}=K,{baseUrl:F,paths:_}=W.getEndpointWithPaths("caliper"),O=W.getTokenProvider("caliper");if(!O)throw Error("Caliper API requires authentication");if(this._provider=W,this.transport=new k9({baseUrl:F,tokenProvider:O,timeout:W.timeout,paths:_}),W.platform===X9)H=new C9;S.info("Client initialized",{platform:W.platform,env:W.env,baseUrl:F})}this.events=new $$(this.transport,{eventTransformer:H}),this.jobs=new J$(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 J7=r5();class K2 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 Z$="/ims/oneroster/rostering/v1p2";function K$($,J){return`${$.replace(/\/+$/,"")}/${J.replace(/^\/+/,"")}`}function t5($,J){return K$($,`${Z$}/courses/${J}`)}function e5($,J){return K$($,`${Z$}/users/${J}`)}function H$($,J){let Z=$.ids?.[J];if(!Z)throw new K2($,J);return Z}function $0($,J,Z){let K=H$(J,Z);return t5($,K)}function J0($){if($.courseCode)return $.courseCode;if($.grade!==void 0)return`${$.subject} G${String($.grade)}`;return $.subject}function v8($,J,Z){let K;try{K=new URL($)}catch{throw new i2($)}let H="grade"in J?`${J.subject}/g${String(J.grade)}`:J.code,W=K.pathname.replace(/\/+$/,"");return K.pathname=`${W}/activities/${H}/${encodeURIComponent(Z)}`,K.toString()}function X$($,J,Z,K,H,W,F){return{id:v8(H,$.course,$.id),type:"TimebackActivityContext",subject:J.subject,app:{name:Z},activity:{name:$.name},course:{id:$0(W,J,K),name:J0(J)},process:F??!0}}function Z0($){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 K0($,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 H0($){return $===void 0?void 0:{pctCompleteApp:$}}function W$($,J){return{courseId:H$($,J)}}function Y$($){if(!$)return;return`urn:uuid:${$}`}function Q$($,J,Z){return{id:e5($,J),type:"TimebackUser",email:Z}}function G$($){let{sensor:J,timebackId:Z,email:K,payload:H,process:W,course:F,appName:_,apiEnv:O,onerosterBaseUrl:w,attempt:j,runId:A}=$,C=Q$(w,Z,K),q=X$(H,F,_,O,J,w,W??H.process),z=Z0(H.metrics),N=H0(H.pctComplete),I=Y$(A),r=W$(F,O),H2=T8({actor:C,object:q,metrics:z,eventTime:H.endedAt,generatedExtensions:N,extensions:r,attempt:j,session:I});return{sensor:J,actor:C,object:q,event:H2,payload:H,course:F,appName:_,apiEnv:O,email:K,timebackId:Z}}function s2($){let{sensor:J,timebackId:Z,email:K,payload:H,course:W,appName:F,apiEnv:_,onerosterBaseUrl:O,runId:w}=$,j=Q$(O,Z,K),A=X$(H,W,F,_,J,O),C=K0(H.elapsedMs,H.pausedMs),q=W$(W,_),z=Y$(w),N=R8({actor:j,object:A,metrics:C,eventTime:H.endedAt,extensions:q,session:z});return{sensor:J,actor:j,object:A,event:N,payload:{id:H.id,name:H.name,course:H.course,startedAt:H.startedAt,endedAt:H.endedAt,elapsedMs:H.elapsedMs,pausedMs:H.pausedMs},course:W,appName:F,apiEnv:_,email:K,timebackId:Z,runId:w}}import*as h from"zod";import*as h2 from"zod";var X0=h2.object({subject:B$,grade:_$}),W0=h2.object({code:E}),w2=h2.union([X0,W0]);class Q2 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 U2($,J){let Z;if("grade"in J)Z=$.filter((K)=>K.subject===J.subject&&K.grade===J.grade);else Z=$.filter((K)=>K.courseCode===J.code);if(Z.length===0)throw new Q2("unknown_course",J);if(Z.length>1)throw new Q2("ambiguous_course",J,Z.length);return Z[0]}var A2=R("handlers:activity:schema"),Y0=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 Q0($){return Math.min(100,Math.max(0,$))}function o2($){if("grade"in $)return`${$.subject} grade ${$.grade}`;return`code "${$.code}"`}var G0=h.object({id:E,name:E,course:w2,runId:h.string().uuid(),startedAt:h.iso.datetime(),endedAt:h.iso.datetime(),elapsedMs:h.number().int().nonnegative(),pausedMs:h.number().int().nonnegative()});function F$($,J,Z){let K=G0.safeParse($);if(!K.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:K.error.flatten()})};let H=K.data,W;try{W=U2(J.courses,H.course)}catch(_){if(_ instanceof Q2){let O=o2(H.course);if(_.code==="unknown_course")return A2.warn("Unknown course selector",{selector:H.course}),{ok:!1,response:M("UNKNOWN_COURSE",`Unknown course: ${O}`,400)};return A2.error("Ambiguous course selector",{selector:H.course}),{ok:!1,response:M("AMBIGUOUS_COURSE_SELECTOR","Ambiguous course selector in timeback.config.json",500)}}throw _}let F=S2(J,W,Z);if(!F){let _=o2(H.course);return A2.error("Missing sensor for course",{selector:H.course}),{ok:!1,response:M("MISSING_SENSOR",`Course "${_}" has no sensor configured.`,500)}}return{ok:!0,payload:H,course:W,sensor:F}}var B0=h.object({id:E,name:E,course:w2,process:h.boolean().optional(),runId:h.string().uuid(),endedAt:h.iso.datetime(),metrics:Y0,pctComplete:h.number().optional().transform(($)=>$===void 0?void 0:Q0($))});function O$($,J,Z){let K=B0.safeParse($);if(!K.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:K.error.flatten()})};let H=K.data,W;try{W=U2(J.courses,H.course)}catch(_){if(_ instanceof Q2){let O=o2(H.course);if(_.code==="unknown_course")return A2.warn("Unknown course selector",{selector:H.course}),{ok:!1,response:M("UNKNOWN_COURSE",`Unknown course: ${O}`,400)};return A2.error("Ambiguous course selector",{selector:H.course}),{ok:!1,response:M("AMBIGUOUS_COURSE_SELECTOR","Ambiguous course selector in timeback.config.json",500)}}throw _}let F=S2(J,W,Z);if(!F){let _=o2(H.course);return A2.error("Missing sensor for course",{selector:H.course}),{ok:!1,response:M("MISSING_SENSOR",`Course "${_}" has no sensor configured.`,500)}}return{ok:!0,payload:H,course:W,sensor:F}}import{getServiceUrlsForEnv as j$}from"@timeback/core";var y8=R("handlers:activity:attempts");async function x$($,J){let Z=[$,J].join("_");return`caliper_${await E2(Z)}`}function _0($,J){return $.find((Z)=>Z.scoreDate===J)}function F0($){let J=0;for(let Z of $){let H=Z.metadata?.attempt;if(typeof H==="number"&&H>J)J=H}return J}function O0($){let J=$.metadata;return typeof J?.attempt==="number"&&J.attempt>=1?J.attempt:1}async function D$($,J,Z,K){let H;try{H=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(H.length===0)return 1;let W=_0(H,K);if(W){let O=O0(W);return y8.debug("Retry detected, reusing attempt number",{lineItemId:J,timebackId:Z,endedAt:K,attempt:O}),O}let F=F0(H),_=F+1;return y8.debug("New attempt computed",{lineItemId:J,timebackId:Z,endedAt:K,maxAttempt:F,nextAttempt:_}),_}var w$=R("handlers:activity:completion");async function M$($){let{client:J,courseId:Z,timebackId:K,pctComplete:H,appName:W}=$;if(H!==100)return;let F=U$(Z),_=`timeback_sdk_${await E2(`mastery-completion_${Z}`)}`,O=`timeback_sdk_${await E2(`mastery-completion_${Z}_${K}`)}`;try{let w=!1;try{await J.oneroster.assessmentLineItems(_).get(),w=!0}catch{}if(!w)await J.oneroster.assessmentLineItems.create({sourcedId:_,title:`${W}: Complete`,status:"active",course:{sourcedId:F.course},componentResource:{sourcedId:F.componentResource},resultValueMin:0,resultValueMax:100,metadata:{appName:W}});let j=new Date().toISOString();await J.oneroster.assessmentResults.upsert(O,{status:"active",assessmentLineItem:{sourcedId:_},student:{sourcedId:K},score:100,scoreDate:j,scoreStatus:"fully graded",metadata:{isMasteryCompletion:!0,completedAt:j,appName:W}}),w$.debug("Created mastery completion entry",{courseId:Z,timebackId:K,lineItemId:_,resultId:O})}catch(w){let j=w instanceof Error?w.message:"Unknown error";w$.error("Failed to create mastery completion entry",{courseId:Z,timebackId:K,lineItemId:_,error:j})}}import{aggregateActivityMetrics as x0}from"@timeback/core/utils";var z2=R("handlers:activity:progress");function D0($,J){let Z=u(J),K=$.overrides?.[Z]?.metadata?.metrics,H=$.metadata?.metrics;return K?.totalLessons??H?.totalLessons}function w0($,J){let Z=$/J*100;return Math.min(100,Math.max(0,Math.round(Z)))}async function L$($){let{client:J,courseId:Z,timebackId:K,payload:H,course:W,env:F}=$;if(H.pctComplete!==void 0)return;let _=H.metrics.masteredUnits;if(typeof _!=="number"||_<=0)return;let O=D0(W,F);if(!O||O<=0){z2.debug("Skipping progress computation: totalLessons not configured",{courseId:Z});return}let w;try{w=(await J.edubridge.enrollments.list({userId:K})).find((N)=>N.course.id===Z)?.id}catch(z){let N=z instanceof Error?z.message:"Unknown error";z2.warn("Failed to fetch enrollments for progress computation",{courseId:Z,timebackId:K,error:N});return}if(!w){z2.warn("Skipping progress computation: enrollment not found for student/course",{courseId:Z,timebackId:K});return}let j=0;try{let z=await J.edubridge.analytics.getEnrollmentFacts({enrollmentId:w});j=x0(z).masteredUnits}catch(z){let N=z instanceof Error?z.message:"Unknown error";z2.warn("Failed to fetch enrollment facts for progress computation",{courseId:Z,timebackId:K,enrollmentId:w,error:N});return}let A=!0;try{if((await J.edubridge.analytics.getWeeklyFacts({studentId:K,weekDate:H.endedAt})).some((I)=>{if(I.activityId!==H.id)return!1;if(I.enrollmentId&&I.enrollmentId===w)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";z2.warn("Failed to fetch weekly facts for retry-safe progress computation",{courseId:Z,timebackId:K,enrollmentId:w,activityId:H.id,error:N}),A=!1}let C=j+(A?_:0),q=w0(C,O);return z2.debug("Computed pctComplete",{courseId:Z,timebackId:K,enrollmentId:w,historicalMasteredUnits:j,currentMasteredUnits:_,shouldIncludeCurrentMasteredUnits:A,totalMastered:C,totalLessons:O,pctComplete:q}),{pctComplete:q}}var f8=R("handlers:activity:submit"),U0={computeProgress:L$,maybeWriteCompletionEntry:M$};function V$($,J){return $.timebackId??G2({email:$.email,client:J})}async function r2($,J,Z=U0){let K=u($.env),{userInfo:H,payload:W,course:F,sensor:_,preview:O=!1,runId:w}=J,j=F.ids?.[K];if(!j)throw new K2(F,K);let A=$.getClient(),C=await V$(H,A),q=W;if(W.pctComplete===void 0){let f2=await Z.computeProgress({client:A,courseId:j,timebackId:C,payload:W,course:F,env:$.env});if(f2)q={...W,pctComplete:f2.pctComplete}}let z;if(!O){let f2=v8(_,q.course,q.id),xJ=await x$(f2,j);z=await D$(A,xJ,C,q.endedAt)}let I=j$(K).oneroster,r=G$({sensor:_,timebackId:C,email:H.email,payload:q,process:q.process,course:F,appName:$.appConfig.name,apiEnv:K,onerosterBaseUrl:I,attempt:z,runId:w}),H2=$.hooks?.beforeActivitySend,y2=H2?await H2(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:j,timebackId:p.timebackId,pctComplete:p.payload.pctComplete,appName:p.appName}),f8.debug("Recorded completion",{courseSelector:W.course,activityId:W.id}),{success:!0}}async function k$($,J){let Z=u($.env),{userInfo:K,payload:H,course:W,sensor:F,runId:_}=J;if(!W.ids?.[Z])throw new K2(W,Z);let w=$.getClient(),j=await V$(K,w),C=j$(Z).oneroster,q=s2({sensor:F,timebackId:j,email:K.email,payload:H,course:W,appName:$.appConfig.name,apiEnv:Z,onerosterBaseUrl:C,runId:_}),z=$.hooks?.beforeTimeSpentSend,N=z?await z(q):q,I=N??q;if(N===null){f8.debug("Hook skipped time-spent send",{activityId:H.id});return}await w.caliper.events.send(I.sensor,[I.event]),f8.debug("Sent time-spent event",{activityId:H.id,elapsedMs:H.elapsedMs})}var v2=R("handlers:activity:submit");async function P2($,J){if($.mode==="custom"){let K=await $.getEmail(J);return K?{email:K}: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 P2($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let K=O$(await J.json(),$.appConfig,$.env);if(!K.ok)return K.response;let{payload:H,course:W,sensor:F}=K,_={id:H.id,name:H.name,course:H.course,process:H.process,endedAt:H.endedAt,metrics:H.metrics,pctComplete:H.pctComplete};try{if((await r2({env:$.env,api:$.api,appConfig:$.appConfig,getClient:$.getClient,hooks:$.hooks},{userInfo:Z,payload:_,course:W,sensor:F,preview:!1,runId:H.runId})).preview)return P({preview:!0});return v2.debug("Submitted activity",{runId:H.runId,activityId:H.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",B2(O));if(O instanceof K2)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 w=O instanceof Error?O.message:"Unknown error";return v2.error("Failed to submit activity",{error:w}),M("INTERNAL_ERROR",w,502)}}}var T2=R("handlers:activity:heartbeat"),e2=new Map,L0=300000;function j0(){let $=Date.now();for(let[J,Z]of e2)if($-Z>L0)e2.delete(J)}function V0($){return j0(),e2.has($)}function q$($){e2.set($,Date.now())}function $8($){return async(J)=>{let Z=await P2($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let K=F$(await J.json(),$.appConfig,$.env);if(!K.ok)return K.response;let{payload:H,course:W,sensor:F}=K,_=new Date(H.startedAt).getTime(),O=`${H.runId}:${_}`;if(V0(O))return T2.debug("Duplicate heartbeat, skipping",{runId:H.runId,windowKey:O}),P({duplicate:!0});let w=u($.env);if(!W.ids?.[w]){let C=new K2(W,w);return T2.warn("Course not synced",{course:H.course,env:w}),M("MISSING_SYNCED_COURSE_ID",C.message,503)}let A=$.getClient();try{let C=Z.timebackId??await G2({email:Z.email,client:A}),z=M0(w).oneroster,N=s2({sensor:F,timebackId:C,email:Z.email,payload:H,course:W,appName:$.appConfig.name,apiEnv:w,onerosterBaseUrl:z,runId:H.runId}),I=$.hooks?.beforeTimeSpentSend,r=I?await I(N):N,H2=r??N;if(r===null)return q$(O),T2.debug("Hook skipped heartbeat send",{runId:H.runId}),new Response(null,{status:204});return await A.caliper.events.send(H2.sensor,[H2.event]),q$(O),T2.debug("Sent heartbeat",{runId:H.runId,activityId:H.id,elapsedMs:H.elapsedMs}),new Response(null,{status:204})}catch(C){if(C instanceof d)return T2.warn("Failed to resolve Timeback user",{code:C.code}),M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",B2(C));let q=C instanceof Error?C.message:"Unknown error";return T2.error("Failed to send heartbeat",{error:q}),M("INTERNAL_ERROR",q,502)}}}async function h$($,J,Z){if(Z.mode!=="sso")return M2.warn("SSO not configured"),P({error:"SSO not configured"},400);return await N$({req:$,env:J,clientId:Z.clientId,issuer:Z.issuer,redirectUri:Z.redirectUri,buildState:Z.buildState})}async function k0($,J,Z,K,H,W,F){try{let _=W.issuer??C$(H),O=b$(J,W.redirectUri);M2.debug("Exchanging auth code for tokens",{issuer:_,clientId:W.clientId});let w=await A$({issuer:_,clientId:W.clientId,clientSecret:W.clientSecret,code:$,redirectUri:O}),j=await z$({issuer:_,accessToken:w.access_token}),A=typeof j.identities==="string"?JSON.parse(j.identities):j.identities;M2.debug("SSO completed, resolving Timeback user",{user:{...j,identities:A}});let C=await _2({env:H,apiCredentials:F.credentials,userInfo:j,client:F.getClient()});M2.debug("Timeback user resolved",{timebackId:C.id});let q={user:C,idp:{tokens:w,userInfo:j},state:Z,req:K,redirect:J8,json:P};return W.onCallbackSuccess(q)}catch(_){let O=_ instanceof Error?_:Error("Unknown error"),w=_ instanceof d?_.code:"token_exchange_failed";if(M2.error("SSO callback failed",{error:O.message,errorCode:w}),W.onCallbackError)return W.onCallbackError(P$(O,w,Z,K));if(_ instanceof d)return P({error:O.message},B2(_));return P({error:O.message},500)}}async function S$($,J,Z,K){if(Z.mode!=="sso")return M2.warn("SSO not configured"),P({error:"SSO not configured"},400);let{url:H,code:W,errorParam:F,state:_}=I$($);if(F)return T$(F,H,_,$,Z.onCallbackError);if(!W)return R$(_,$,Z.onCallbackError);return await k0(W,H,_,$,J,Z,K)}function Z8($){let{env:J,identity:Z}=$;return(K)=>h$(K,J,Z)}function K8($){let{env:J,identity:Z,api:K}=$;return(H)=>S$(H,J,Z,K)}function H8(){return()=>J8("/")}var E$=($)=>{return async(J)=>{return await $.getClient().powerpath.assessments.getAssessmentProgress({student:J.student,lesson:J.lesson,attempt:J.attempt})}};var v$=($)=>{return async(J)=>{return(await $.getClient().powerpath.assessments.getAttempts({student:J.student,lesson:J.lesson})).attempts.map((K)=>({attempt:K.attempt??void 0,score:K.score,startedAt:K.startedAt??void 0,completedAt:K.completedAt??void 0,scoreStatus:K.scoreStatus}))}};var y$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments,K=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson}),H;if(K.lessonType!=="powerpath-100"&&K.finalized!==!0){try{H=await Z.finalStudentAssessmentResponse({student:J.student,lesson:J.lesson})}catch{}K=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson})}let W=K.lessonType==="powerpath-100"?!0:K.finalized===!0;return{...H??{},score:K.score??0,totalQuestions:K.totalQuestions,correctQuestions:K.correctQuestions,accuracy:K.accuracy,finalized:W}}};function f$($){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 g$($,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 d$($){let J=$.questions,Z=J.filter((K)=>K.response!==void 0).map((K)=>K.id);return{questions:J.map(X8),answeredIds:Z,score:$.score??0,finalized:$.finalized,complete:$.finalized}}function q0($){if($.lessonType==="powerpath-100")return $.responseResult.isCorrect;return!1}function C0($){if($.lessonType==="powerpath-100")return $.powerpathScore;return 0}function m$($){let J=C0($);return{correct:q0($),score:J,complete:$.lessonType==="powerpath-100"&&J>=100,questionResult:$.questionResult}}function l$($,J){return Array.from(new Set($.map((Z)=>Z.ids?.[J]).filter((Z)=>Boolean(Z))))}function u$($){return String($.sourcedId??"")}var c$=($)=>{return async(J={})=>{let Z=$.getClient(),K=u($.env),H;if(J.course){let _=U2($.appConfig.courses,J.course).ids?.[K];H=_?[_]:[]}else H=l$($.appConfig.courses,K);let W=[];for(let F of H){let _=await Z.oneroster.courses(F).components({where:{status:"active"}}),O=g8(_);for(let w of O){let j=u$(w);if(!j)continue;let A=await Z.oneroster.courses.componentResources({where:{"courseComponent.sourcedId":j,status:"active"}}),C=g8(A);for(let q of C){if(!q.sourcedId)continue;let z=q.metadata??{},N=f$(q);W.push({id:q.sourcedId,name:q.title??q.sourcedId,type:z.lessonType??"quiz",courseId:F,questionCount:N,metadata:z})}}}return W}};class W8 extends Error{name="LessonTypeMismatchError"}var p$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments;if(J.lessonType==="powerpath-100"){let H=await Z.getNextQuestion({student:J.student,lesson:J.lesson});return X8(H.question)}let K=await Z.getAssessmentProgress({student:J.student,lesson:J.lesson});if(K.lessonType==="powerpath-100"){if(J.lessonType!==void 0)throw new W8(`Expected quiz progress but got powerpath-100 for lesson ${J.lesson}`);let H=await Z.getNextQuestion({student:J.student,lesson:J.lesson});return X8(H.question)}return d$(K)}};var A0=R("lessons:start"),a$=($)=>{return async(J)=>{let Z=$.getClient().powerpath.assessments,K=J.lesson,H=J.student,W=await Z.getAssessmentProgress({student:H,lesson:K});if(J.forceNew){if(W.lessonType!=="powerpath-100"&&W.finalized!==!0)try{await Z.finalStudentAssessmentResponse({student:H,lesson:K})}catch{}try{await Z.createNewAttempt({student:H,lesson:K}),W=await Z.getAssessmentProgress({student:H,lesson:K})}catch(F){A0.warn("Failed to create new lesson attempt",{lessonId:K,error:F})}}else if(W.lessonType!=="powerpath-100"&&W.finalized!==!0&&(W.questions?.length??0)===0)try{await Z.createNewAttempt({student:H,lesson:K}),W=await Z.getAssessmentProgress({student:H,lesson:K})}catch{}return g$(K,W)}};var i$=($)=>{return async(J)=>{let K=await $.getClient().powerpath.assessments.updateStudentQuestionResponse({student:J.student,lesson:J.lesson,question:J.question,responses:{RESPONSE:J.response}});return m$(K)}};function m($){return{list:c$($),start:a$($),next:p$($),submit:i$($),complete:y$($),attempts:v$($),attemptDetails:E$($)}}async function a($,J){let Z=await P2($.identity,J);if(!Z)return null;if(Z.timebackId)return Z.timebackId;return await G2({email:Z.email,client:$.getClient()})}async function s$($,J){let Z=await P2($.identity,J);return Boolean(Z)}var z0=R("handlers:lessons");function c($,J){if($ instanceof d)return M("USER_RESOLUTION_FAILED","Unable to resolve Timeback identity",B2($));if($ instanceof W8)return M("LESSON_TYPE_MISMATCH",$.message,400);let Z=$ instanceof Error?$.message:"Unknown error";return z0.error(J,{error:Z}),M("INTERNAL_ERROR",Z,500)}import*as i from"zod";var P0=i.object({course:w2.optional()}).partial(),T0=i.object({lessonId:E,forceNew:i.boolean().optional()}),R0=i.object({lessonId:E,lessonType:E.optional()}),N0=i.object({lessonId:E,questionId:E,response:E}),I0=i.object({lessonId:E}),b0=i.object({lessonId:E}),h0=i.object({lessonId:E,attempt:i.number().int().positive()});async function S0($){try{let J=await $.json();if(!J||typeof J!=="object"||Array.isArray(J))return null;return J}catch{return null}}async function L2($,J){let Z=await S0($);if(!Z)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid JSON body",400)};let K=J.safeParse(Z);if(!K.success)return{ok:!1,response:M("INVALID_PAYLOAD","Invalid payload",400,{details:K.error.flatten()})};return{ok:!0,payload:K.data}}async function o$($){return await L2($,P0)}async function n$($){return await L2($,T0)}async function r$($){return await L2($,R0)}async function t$($){return await L2($,N0)}async function e$($){return await L2($,I0)}async function $J($){return await L2($,b0)}async function JJ($){return await L2($,h0)}function Y8($){let J=m($);return async(Z)=>{try{let K=await JJ(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.attemptDetails({lesson:K.payload.lessonId,attempt:K.payload.attempt,student:H});return P(W)}catch(K){return c(K,"Failed to fetch lesson attempt details")}}}function Q8($){let J=m($);return async(Z)=>{try{let K=await $J(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.attempts({lesson:K.payload.lessonId,student:H});return P({attempts:W})}catch(K){return c(K,"Failed to list lesson attempts")}}}function G8($){let J=m($);return async(Z)=>{try{let K=await e$(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.complete({lesson:K.payload.lessonId,student:H});return P(W)}catch(K){return c(K,"Failed to complete lesson")}}}function B8($){let J=m($);return async(Z)=>{try{if(!await s$($,Z))return M("UNAUTHORIZED","Unauthorized",401);let K=await o$(Z);if(!K.ok)return K.response;let H=await J.list({course:K.payload.course});return P({lessons:H})}catch(K){return c(K,"Failed to list lessons")}}}function _8($){let J=m($);return async(Z)=>{try{let K=await r$(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.next({lesson:K.payload.lessonId,lessonType:K.payload.lessonType,student:H});return P(W)}catch(K){return c(K,"Failed to fetch next lesson question")}}}function F8($){let J=m($);return async(Z)=>{try{let K=await n$(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.start({lesson:K.payload.lessonId,student:H,forceNew:K.payload.forceNew});return P(W)}catch(K){return c(K,"Failed to start lesson")}}}function O8($){let J=m($);return async(Z)=>{try{let K=await t$(Z);if(!K.ok)return K.response;let H=await a($,Z);if(!H)return M("UNAUTHORIZED","Unauthorized",401);let W=await J.submit({lesson:K.payload.lessonId,question:K.payload.questionId,response:K.payload.response,student:H});return P(W)}catch(K){return c(K,"Failed to submit lesson answer")}}}function ZJ($,J){let Z=new Map;for(let K of $){let H=K.ids?.[J];if(H)Z.set(H,K)}return Z}function KJ($,J){return $.map((Z)=>{let K=J.get(Z.course.id);return{id:Z.course.id,code:K?.courseCode??Z.course.id,name:Z.course.title}})}function HJ($){return $.map((J)=>J.metadata?.goals).find(Boolean)}function XJ($){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((K,H)=>{return K+(H.activityMetrics?.xpEarned??0)},0)},0)}async function x8($,J,Z,K){let H=await $.edubridge.enrollments.list({userId:J.id}),W=ZJ(Z.courses,K),F=KJ(H,W),_=HJ(H),{start:O,end:w}=XJ(new Date),[j,A]=await Promise.all([$.edubridge.analytics.getActivity({studentId:J.id,startDate:O.toISOString(),endDate:w.toISOString()}),$.edubridge.analytics.getActivity({studentId:J.id,startDate:"2000-01-01",endDate:w.toISOString()})]);return{id:J.id,email:J.email,name:J.name,school:J.school,grade:J.grade,courses:F.length?F:void 0,goals:_,xp:{today:d8(j),all:d8(A)}}}var m8=R("handlers:user");async function E0($,J){if($.mode==="custom"){let K=await $.getEmail(J);if(!K)return;return{sub:K,email:K}}let Z=await $.getUser(J);if(!Z)return;return{sub:Z.id,email:Z.email}}function v0($){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 D8($){return async(J)=>{try{let Z=await E0($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let K=u($.env),H=$.getClient();try{let W=await _2({env:$.env,apiCredentials:$.api,userInfo:Z,client:H}),F=await x8(H,W,$.appConfig,K);return P(F)}catch(W){if(W instanceof d)return v0(W);let F=W instanceof Error?W.message:"Unknown error";return m8.error("Failed to build user profile",{error:F}),M("INTERNAL_ERROR",F,502)}}catch(Z){let K=Z instanceof Error?Z.message:"Unknown error";return m8.error("Unhandled error in user handler",{error:K}),M("INTERNAL_ERROR",K,500)}}}var l8=R("handlers:user:verify");async function y0($,J){if($.mode==="custom"){let K=await $.getEmail(J);if(!K)return;return{sub:K,email:K}}let Z=await $.getUser(J);if(!Z)return;return{sub:Z.id,email:Z.email}}function w8($){return async(J)=>{try{let Z=await y0($.identity,J);if(!Z)return M("UNAUTHORIZED","Unauthorized",401);let K=$.getClient();try{let H=await _2({env:$.env,apiCredentials:$.api,userInfo:Z,client:K});return P({verified:!0,timebackId:H.id})}catch(H){if(H instanceof d){if(l8.warn("Timeback user resolution failed",{code:H.code}),H.code==="timeback_user_ambiguous")return M("USER_AMBIGUOUS","Timeback user resolution ambiguous",409);if(H.code==="timeback_user_not_found")return P({verified:!1});return M("USER_RESOLUTION_FAILED",H.message,502)}let W=H instanceof Error?H.message:"Unknown error";return l8.error("Failed to verify user",{error:W}),M("INTERNAL_ERROR",W,502)}}catch(Z){let K=Z instanceof Error?Z.message:"Unknown error";return l8.error("Unhandled error in verify handler",{error:K}),M("INTERNAL_ERROR",K,500)}}}import*as T from"zod";var f0=T.object({email:T.email(),timebackId:T.string().optional()}),g0=T.object({id:E,name:E,course:w2}),d0=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),m0=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()}),l0=T.uuid().optional(),WJ=T.object({user:f0,activity:g0,metrics:d0,time:m0.optional(),runId:l0});var YJ=R("activity:record");class QJ 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 u0($){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 c0($,J){let Z=$.time,K=Z.activeMs??0,H=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 F=new Date(J),_=K+H;W=new Date(F.getTime()-_).toISOString()}else W=J;return{id:$.activity.id,name:$.activity.name,course:$.activity.course,startedAt:W,endedAt:J,elapsedMs:K,pausedMs:H}}function p0($,J,Z){let K;try{K=U2($.courses,J)}catch(W){if(W instanceof Q2)throw new u8(W.code==="unknown_course"?`Unknown course: ${JSON.stringify(J)}`:"Ambiguous course selector in timeback.config.json");throw W}let H=S2($,K,Z);if(!H)throw new u8(`Course "${JSON.stringify(J)}" has no sensor configured.`);return{course:K,sensor:H}}var a0={recordCompletion:r2,sendTimeSpent:k$};function c8($,J=a0){return async(Z)=>{let K=WJ.safeParse(Z);if(!K.success)throw YJ.warn("Invalid record input",{error:K.error.flatten()}),new QJ("Invalid activity record input",K.error.flatten());let H=K.data,{course:W,sensor:F}=p0($.appConfig,H.activity.course,$.env),_=u0(H),O={env:$.env,api:$.api,appConfig:$.appConfig,getClient:$.getClient,hooks:$.hooks};if(await J.recordCompletion(O,{userInfo:{email:H.user.email,timebackId:H.user.timebackId},payload:_,course:W,sensor:F,preview:!1,runId:H.runId}),H.time){let w=c0(H,_.endedAt);await J.sendTimeSpent(O,{userInfo:{email:H.user.email,timebackId:H.user.timebackId},payload:w,course:W,sensor:F,runId:H.runId})}YJ.debug("Recorded activity",{activityId:H.activity.id,hasTime:!!H.time})}}var U8=R("user:verify");function p8($){return async(J)=>{U8.debug("Verifying user by email");let Z=$.getClient();try{let K=await G2({email:J,client:Z});return U8.debug("User verified",{timebackId:K}),{verified:!0,timebackId:K}}catch(K){if(K instanceof d){if(K.code==="timeback_user_not_found")return U8.debug("User not found in Timeback"),{verified:!1};throw K}let H=K instanceof Error?K.message:"Unknown error";throw U8.error("Failed to verify user",{error:H}),K}}}var a8=R("user:getProfile");function i8($){return async(J)=>{a8.debug("Building user profile by email");let Z=$.getClient(),K=u($.env);try{let H=await _2({env:$.env,apiCredentials:$.api,userInfo:{sub:J,email:J},client:Z}),W=await x8(Z,H,$.appConfig,K);return a8.debug("User profile built",{userId:W.id}),W}catch(H){if(H instanceof d)throw H;let W=H instanceof Error?H.message:"Unknown error";throw a8.error("Failed to build user profile",{error:W}),H}}}async function s0($){let{loadConfig:J}=await import("./chunk-44nb8dx1.js");return await J($)}function o0($){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 FJ($){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.");GJ({logger:$.logger,logLevel:$.logLevel});let J=BJ($.env),Z=await s0({configPath:$.configPath});if(!Z.success)throw Error(`Failed to load timeback config: ${Z.error}`);let K=Z.config,H,W=()=>{if(!H)H=new i0({env:u(J),auth:{clientId:$.api.clientId,clientSecret:$.api.clientSecret}});return H},F=$.hooks,_=(j,A)=>_J(j,F,A),O={env:J,identity:$.identity,appConfig:{name:K.name,sensor:K.sensor,launchUrl:K.launchUrl,courses:o0(K.courses)},api:$.api,getClient:W,hooks:F},w={env:J,identity:$.identity,api:{credentials:$.api,getClient:W}};return{config:$,handle:{activity:{heartbeat:_("activity.heartbeat",$8(O)),submit:_("activity.submit",t2(O))},identity:{signIn:_("identity.signIn",Z8(w)),callback:_("identity.callback",K8(w)),signOut:H8()},lessons:{list:_("lessons.list",B8(O)),start:_("lessons.start",F8(O)),next:_("lessons.next",_8(O)),submit:_("lessons.submit",O8(O)),complete:_("lessons.complete",G8(O)),attempts:_("lessons.attempts",Q8(O)),attemptDetails:_("lessons.attemptDetails",Y8(O))},user:{me:_("user.me",D8(O)),verify:_("user.verify",w8(O))}},activity:{record:c8({...O})},lessons:m({...O}),user:{verify:p8({...O}),getProfile:i8({...O})},get api(){return W()},close(){if(H)H.close(),H=void 0}}}import{TimebackClient as zH}from"@timeback/core";import{ApiError as TH,ForbiddenError as RH,isApiError as NH,NotFoundError as IH,UnauthorizedError as bH,ValidationError as hH}from"@timeback/core";export{r0 as toNativeHandler,NH as isApiError,OJ as createTimebackIdentity,FJ as createTimeback,hH as ValidationError,bH as UnauthorizedError,zH as TimebackClient,n0 as ROUTES,IH as NotFoundError,RH as ForbiddenError,TH as ApiError};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@timeback/sdk",
3
- "version": "0.2.1",
3
+ "version": "0.2.2-beta.20260320221119",
4
4
  "description": "Timeback SDK for frontend and backend integration",
5
5
  "type": "module",
6
6
  "exports": {