@vue-skuilder/platform-ui 0.2.7 → 0.2.8

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 (93) hide show
  1. package/dist/assets/{About-BzH1hc19.js → About-zjTECQNT.js} +2 -2
  2. package/dist/assets/{About-BzH1hc19.js.map → About-zjTECQNT.js.map} +1 -1
  3. package/dist/assets/{AdminDashboard-k9H7Nin1.js → AdminDashboard-_zq4nONY.js} +2 -2
  4. package/dist/assets/{AdminDashboard-k9H7Nin1.js.map → AdminDashboard-_zq4nONY.js.map} +1 -1
  5. package/dist/assets/{ClassroomCtrlPanel-DxSSolim.js → ClassroomCtrlPanel-DCbpbIEB.js} +2 -2
  6. package/dist/assets/{ClassroomCtrlPanel-DxSSolim.js.map → ClassroomCtrlPanel-DCbpbIEB.js.map} +1 -1
  7. package/dist/assets/{Classrooms-jR_9dXWG.js → Classrooms-BUI7d6V7.js} +2 -2
  8. package/dist/assets/{Classrooms-jR_9dXWG.js.map → Classrooms-BUI7d6V7.js.map} +1 -1
  9. package/dist/assets/{CourseRouter-CKNJNcOy.js → CourseRouter-D3ww8C0T.js} +2 -2
  10. package/dist/assets/{CourseRouter-CKNJNcOy.js.map → CourseRouter-D3ww8C0T.js.map} +1 -1
  11. package/dist/assets/{CourseWare-BTFRjgBR-DGbHJAQw.js → CourseWare-BTFRjgBR-BKjh4u_F.js} +3 -3
  12. package/dist/assets/{CourseWare-BTFRjgBR-DGbHJAQw.js.map → CourseWare-BTFRjgBR-BKjh4u_F.js.map} +1 -1
  13. package/dist/assets/{Courses-C2SpD8O_.js → Courses-BUtV1A6B.js} +2 -2
  14. package/dist/assets/{Courses-C2SpD8O_.js.map → Courses-BUtV1A6B.js.map} +1 -1
  15. package/dist/assets/{EloModeration-BKouyD0q.js → EloModeration-6WAH9JnJ.js} +2 -2
  16. package/dist/assets/{EloModeration-BKouyD0q.js.map → EloModeration-6WAH9JnJ.js.map} +1 -1
  17. package/dist/assets/{JoinCode-oDXdI9Bl.js → JoinCode-jR454PDt.js} +2 -2
  18. package/dist/assets/{JoinCode-oDXdI9Bl.js.map → JoinCode-jR454PDt.js.map} +1 -1
  19. package/dist/assets/{NewCourseDialog-CMR8fR_w.js → NewCourseDialog-CItEIHPl.js} +2 -2
  20. package/dist/assets/{NewCourseDialog-CMR8fR_w.js.map → NewCourseDialog-CItEIHPl.js.map} +1 -1
  21. package/dist/assets/{ReleaseNotes-gSZkUGbN.js → ReleaseNotes-P12l56VV.js} +2 -2
  22. package/dist/assets/{ReleaseNotes-gSZkUGbN.js.map → ReleaseNotes-P12l56VV.js.map} +1 -1
  23. package/dist/assets/{RequestPasswordReset-DTpIE_EE.js → RequestPasswordReset-DIIQQm5q.js} +2 -2
  24. package/dist/assets/{RequestPasswordReset-DTpIE_EE.js.map → RequestPasswordReset-DIIQQm5q.js.map} +1 -1
  25. package/dist/assets/{ResetPassword-CWoI5c6A.js → ResetPassword-q5BbSDfD.js} +2 -2
  26. package/dist/assets/{ResetPassword-CWoI5c6A.js.map → ResetPassword-q5BbSDfD.js.map} +1 -1
  27. package/dist/assets/{Study-CcNjIsiJ.js → Study-B1IGZg7q.js} +2 -2
  28. package/dist/assets/{Study-CcNjIsiJ.js.map → Study-B1IGZg7q.js.map} +1 -1
  29. package/dist/assets/{TagInformation-Dm3b5YAO.js → TagInformation-Brkhk5Uc.js} +2 -2
  30. package/dist/assets/{TagInformation-Dm3b5YAO.js.map → TagInformation-Brkhk5Uc.js.map} +1 -1
  31. package/dist/assets/{User-eYKSN_Xf.js → User-xO6M7dY_.js} +2 -2
  32. package/dist/assets/{User-eYKSN_Xf.js.map → User-xO6M7dY_.js.map} +1 -1
  33. package/dist/assets/{UserStats-BfKRqF6i.js → UserStats-B1LVmIde.js} +2 -2
  34. package/dist/assets/{UserStats-BfKRqF6i.js.map → UserStats-B1LVmIde.js.map} +1 -1
  35. package/dist/assets/{VerifyEmail-CNih7343.js → VerifyEmail-AhuAGEz9.js} +2 -2
  36. package/dist/assets/{VerifyEmail-CNih7343.js.map → VerifyEmail-AhuAGEz9.js.map} +1 -1
  37. package/dist/assets/{chess-E2uOAyQS-COz_dOo5.js → chess-E2uOAyQS-BOg8AjuS.js} +2 -2
  38. package/dist/assets/{chess-E2uOAyQS-COz_dOo5.js.map → chess-E2uOAyQS-BOg8AjuS.js.map} +1 -1
  39. package/dist/assets/chess-Lfm7Fsvs-gHEeTsK2.js +1 -0
  40. package/dist/assets/{common-ui.es-B-2t-QrP.js → common-ui.es-BcsR0aYn.js} +8 -8
  41. package/dist/assets/{common-ui.es-B-2t-QrP.js.map → common-ui.es-BcsR0aYn.js.map} +1 -1
  42. package/dist/assets/common-ui.es-DeiLeCAz.js +1 -0
  43. package/dist/assets/{dist-DvMZVJ8K.js → dist-DHNWv7su.js} +4 -4
  44. package/dist/assets/{dist-DvMZVJ8K.js.map → dist-DHNWv7su.js.map} +1 -1
  45. package/dist/assets/dist-DTgyj0s-.js +1 -0
  46. package/dist/assets/{dist-CWQK_nTV.js → dist-DlWUPaK5.js} +3 -3
  47. package/dist/assets/{dist-CWQK_nTV.js.map → dist-DlWUPaK5.js.map} +1 -1
  48. package/dist/assets/dist-aM_puP0T.js +1 -0
  49. package/dist/assets/edit-ui.es-B6B148Iu.js +1 -0
  50. package/dist/assets/edit-ui.es-Ddfco4Pk.js +36 -0
  51. package/dist/assets/{edit-ui.es-Bj4rF6z9.js.map → edit-ui.es-Ddfco4Pk.js.map} +1 -1
  52. package/dist/assets/{french-Dk7YG8Td-fk5CTUzg.js → french-Dk7YG8Td-CYgzVCIJ.js} +2 -2
  53. package/dist/assets/{french-Dk7YG8Td-fk5CTUzg.js.map → french-Dk7YG8Td-CYgzVCIJ.js.map} +1 -1
  54. package/dist/assets/french-evUMlbbq-Ck2dkMEb.js +1 -0
  55. package/dist/assets/{index-DVJEAUly.js → index-DqcURlBg.js} +4 -4
  56. package/dist/assets/{index-DVJEAUly.js.map → index-DqcURlBg.js.map} +1 -1
  57. package/dist/assets/{math-B4HbgYf6-DjrrGh69.js → math-B4HbgYf6-vUjsDv91.js} +2 -2
  58. package/dist/assets/{math-B4HbgYf6-DjrrGh69.js.map → math-B4HbgYf6-vUjsDv91.js.map} +1 -1
  59. package/dist/assets/math-DYni7rRl-ClYSEmZE.js +1 -0
  60. package/dist/assets/piano-BN5Btq91-B--Hu9NY.js +1 -0
  61. package/dist/assets/{piano-DF1g6yaX-CbhFxxTH.js → piano-DF1g6yaX-vK-4k-Zr.js} +2 -2
  62. package/dist/assets/{piano-DF1g6yaX-CbhFxxTH.js.map → piano-DF1g6yaX-vK-4k-Zr.js.map} +1 -1
  63. package/dist/assets/{pitch-CgGJFkZ1-Bi5eK_21.js → pitch-CgGJFkZ1-Cjyh79lG.js} +2 -2
  64. package/dist/assets/{pitch-CgGJFkZ1-Bi5eK_21.js.map → pitch-CgGJFkZ1-Cjyh79lG.js.map} +1 -1
  65. package/dist/assets/pitch-Dn0iNqiS-CyQaBUyI.js +1 -0
  66. package/dist/assets/{server-6eXVwi9x.js → server-DSSG5Zqs.js} +2 -2
  67. package/dist/assets/{server-6eXVwi9x.js.map → server-DSSG5Zqs.js.map} +1 -1
  68. package/dist/assets/{sightsing-BFQ7HRig-Fn9kCUeR.js → sightsing-BFQ7HRig-DSg6WCvP.js} +2 -2
  69. package/dist/assets/{sightsing-BFQ7HRig-Fn9kCUeR.js.map → sightsing-BFQ7HRig-DSg6WCvP.js.map} +1 -1
  70. package/dist/assets/sightsing-CJX3k0Fd-CzhjJG5U.js +1 -0
  71. package/dist/assets/{typing-BevtfWlp-Dq6FH0BZ.js → typing-BevtfWlp-BZPlnLfL.js} +2 -2
  72. package/dist/assets/{typing-BevtfWlp-Dq6FH0BZ.js.map → typing-BevtfWlp-BZPlnLfL.js.map} +1 -1
  73. package/dist/assets/typing-k-ojjO7G-CrlVRJ6T.js +1 -0
  74. package/dist/assets/{word-work-BOnRlZgd-BYFf0Knb.js → word-work-BOnRlZgd-0cck95wD.js} +2 -2
  75. package/dist/assets/{word-work-BOnRlZgd-BYFf0Knb.js.map → word-work-BOnRlZgd-0cck95wD.js.map} +1 -1
  76. package/dist/assets/word-work-Dd6tIKrt-JBcDjC3D.js +1 -0
  77. package/dist/index.html +3 -3
  78. package/dist/sw.js +1 -1
  79. package/dist/sw.js.map +1 -1
  80. package/package.json +7 -7
  81. package/dist/assets/chess-Lfm7Fsvs-B9yA4auL.js +0 -1
  82. package/dist/assets/common-ui.es-B5d1EX2n.js +0 -1
  83. package/dist/assets/dist-CLkujmTs.js +0 -1
  84. package/dist/assets/dist-Dzx6egJo.js +0 -1
  85. package/dist/assets/edit-ui.es-B5aacNFP.js +0 -1
  86. package/dist/assets/edit-ui.es-Bj4rF6z9.js +0 -36
  87. package/dist/assets/french-evUMlbbq-bQfCXRur.js +0 -1
  88. package/dist/assets/math-DYni7rRl-BBPYLFvc.js +0 -1
  89. package/dist/assets/piano-BN5Btq91-C0-aF2s_.js +0 -1
  90. package/dist/assets/pitch-Dn0iNqiS-BxOmiYoN.js +0 -1
  91. package/dist/assets/sightsing-CJX3k0Fd-HEow72a8.js +0 -1
  92. package/dist/assets/typing-k-ojjO7G-BTgit7vr.js +0 -1
  93. package/dist/assets/word-work-Dd6tIKrt-C8n1ubqB.js +0 -1
@@ -0,0 +1 @@
1
+ import"./vue.runtime.esm-bundler-C3q8JS9f.js";import"./common-ui.es-BcsR0aYn.js";import"./dist-leWgtXBf.js";import{T as e,m as t}from"./MarkdownRenderer-kStoDRNE-CwICngq1.js";import"./dist-DHNWv7su.js";export{e as MarkdownRenderer,t as piniaPlugin};
@@ -11,7 +11,7 @@ Please double-check your map/reduce function.`),guardedConsole(`error`,t,n)}}fun
11
11
  `)===0?e.substr(1,e.length):e}).forEach(function(e){var n=e.split(`:`),r=n.shift().trim();if(r){var a=n.join(`:`).trim();try{t.append(r,a)}catch(e){console.warn(`Response `+e.message)}}}),t}Body.call(Request.prototype);function Response(e,t){if(!(this instanceof Response))throw TypeError(`Please use the "new" operator, this DOM object constructor cannot be called as a function.`);if(t||(t={}),this.type=`default`,this.status=t.status===void 0?200:t.status,this.status<200||this.status>599)throw RangeError(`Failed to construct 'Response': The status provided (0) is outside the range [200, 599].`);this.ok=this.status>=200&&this.status<300,this.statusText=t.statusText===void 0?``:``+t.statusText,this.headers=new Headers(t.headers),this.url=t.url||``,this._initBody(e)}Body.call(Response.prototype),Response.prototype.clone=function(){return new Response(this._bodyInit,{status:this.status,statusText:this.statusText,headers:new Headers(this.headers),url:this.url})},Response.error=function(){var e=new Response(null,{status:200,statusText:``});return e.ok=!1,e.status=0,e.type=`error`,e};var c=[301,302,303,307,308];Response.redirect=function(e,t){if(c.indexOf(t)===-1)throw RangeError(`Invalid status code`);return new Response(null,{status:t,headers:{location:e}})},t.DOMException=n.DOMException;try{new t.DOMException}catch{t.DOMException=function(e,t){this.message=e,this.name=t,this.stack=Error(e).stack},t.DOMException.prototype=Object.create(Error.prototype),t.DOMException.prototype.constructor=t.DOMException}function fetch(e,a){return new Promise(function(o,s){var c=new Request(e,a);if(c.signal&&c.signal.aborted)return s(new t.DOMException(`Aborted`,`AbortError`));var l=new XMLHttpRequest;function abortXhr(){l.abort()}l.onload=function(){var e={statusText:l.statusText,headers:parseHeaders(l.getAllResponseHeaders()||``)};c.url.indexOf(`file://`)===0&&(l.status<200||l.status>599)?e.status=200:e.status=l.status,e.url=`responseURL`in l?l.responseURL:e.headers.get(`X-Request-URL`);var t=`response`in l?l.response:l.responseText;setTimeout(function(){o(new Response(t,e))},0)},l.onerror=function(){setTimeout(function(){s(TypeError(`Network request failed`))},0)},l.ontimeout=function(){setTimeout(function(){s(TypeError(`Network request timed out`))},0)},l.onabort=function(){setTimeout(function(){s(new t.DOMException(`Aborted`,`AbortError`))},0)};function fixUrl(e){try{return e===``&&n.location.href?n.location.href:e}catch{return e}}if(l.open(c.method,fixUrl(c.url),!0),c.credentials===`include`?l.withCredentials=!0:c.credentials===`omit`&&(l.withCredentials=!1),`responseType`in l&&(r.blob?l.responseType=`blob`:r.arrayBuffer&&(l.responseType=`arraybuffer`)),a&&typeof a.headers==`object`&&!(a.headers instanceof Headers||n.Headers&&a.headers instanceof n.Headers)){var u=[];Object.getOwnPropertyNames(a.headers).forEach(function(e){u.push(normalizeName(e)),l.setRequestHeader(e,normalizeValue(a.headers[e]))}),c.headers.forEach(function(e,t){u.indexOf(t)===-1&&l.setRequestHeader(t,e)})}else c.headers.forEach(function(e,t){l.setRequestHeader(t,e)});c.signal&&(c.signal.addEventListener(`abort`,abortXhr),l.onreadystatechange=function(){l.readyState===4&&c.signal.removeEventListener(`abort`,abortXhr)}),l.send(c._bodyInit===void 0?null:c._bodyInit)})}return fetch.polyfill=!0,n.fetch||(n.fetch=fetch,n.Headers=Headers,n.Request=Request,n.Response=Response),t.Headers=Headers,t.Request=Request,t.Response=Response,t.fetch=fetch,t})({})})(r),r.fetch.ponyfill=!0,delete r.fetch.polyfill;var a=n.fetch?n:r;e=a.fetch,e.default=a.fetch,e.fetch=a.fetch,e.Headers=a.Headers,e.Request=a.Request,e.Response=a.Response,t.exports=e})),require_browser=__commonJSMin(((e,t)=>{var n=t.exports={},r,a;function defaultSetTimout(){throw Error(`setTimeout has not been defined`)}function defaultClearTimeout(){throw Error(`clearTimeout has not been defined`)}(function(){try{r=typeof setTimeout==`function`?setTimeout:defaultSetTimout}catch{r=defaultSetTimout}try{a=typeof clearTimeout==`function`?clearTimeout:defaultClearTimeout}catch{a=defaultClearTimeout}})();function runTimeout(e){if(r===setTimeout)return setTimeout(e,0);if((r===defaultSetTimout||!r)&&setTimeout)return r=setTimeout,setTimeout(e,0);try{return r(e,0)}catch{try{return r.call(null,e,0)}catch{return r.call(this,e,0)}}}function runClearTimeout(e){if(a===clearTimeout)return clearTimeout(e);if((a===defaultClearTimeout||!a)&&clearTimeout)return a=clearTimeout,clearTimeout(e);try{return a(e)}catch{try{return a.call(null,e)}catch{return a.call(this,e)}}}var o=[],s=!1,c,l=-1;function cleanUpNextTick(){!s||!c||(s=!1,c.length?o=c.concat(o):l=-1,o.length&&drainQueue())}function drainQueue(){if(!s){var e=runTimeout(cleanUpNextTick);s=!0;for(var t=o.length;t;){for(c=o,o=[];++l<t;)c&&c[l].run();l=-1,t=o.length}c=null,s=!1,runClearTimeout(e)}}n.nextTick=function(e){var t=Array(arguments.length-1);if(arguments.length>1)for(var n=1;n<arguments.length;n++)t[n-1]=arguments[n];o.push(new Item(e,t)),o.length===1&&!s&&runTimeout(drainQueue)};function Item(e,t){this.fun=e,this.array=t}Item.prototype.run=function(){this.fun.apply(null,this.array)},n.title=`browser`,n.browser=!0,n.env={},n.argv=[],n.version=``,n.versions={};function noop(){}n.on=noop,n.addListener=noop,n.once=noop,n.off=noop,n.removeListener=noop,n.removeAllListeners=noop,n.emit=noop,n.prependListener=noop,n.prependOnceListener=noop,n.listeners=function(e){return[]},n.binding=function(e){throw Error(`process.binding is not supported`)},n.cwd=function(){return`/`},n.chdir=function(e){throw Error(`process.chdir is not supported`)},n.umask=function(){return 0}})),import___vite_browser_external=__toESM(require___vite_browser_external(),1),import_browser_ponyfill=__toESM(require_browser_ponyfill(),1),import_browser=__toESM(require_browser(),1),_SessionController2,__defProp=Object.defineProperty,__getOwnPropNames=Object.getOwnPropertyNames,__glob=e=>t=>{var n=e[t];if(n)return n();throw Error(`Module not found in bundle: `+t)},__esm=(e,t)=>function __init(){return e&&(t=(0,e[__getOwnPropNames(e)[0]])(e=0)),t},__export=(e,t)=>{for(var n in t)__defProp(e,n,{get:t[n],enumerable:!0})},init_adminDB=__esm({"src/core/interfaces/adminDB.ts"(){}}),init_classroomDB=__esm({"src/core/interfaces/classroomDB.ts"(){}}),init_SyncStrategy=__esm({"src/impl/common/SyncStrategy.ts"(){}}),isDevelopment,logger,init_logger=__esm({"src/util/logger.ts"(){isDevelopment=typeof process<`u`&&!1,logger={debug:(e,...t)=>{isDevelopment&&console.debug(`[DB:DEBUG] ${e}`,...t)},info:(e,...t)=>{console.info(`[DB:INFO] ${e}`,...t)},warn:(e,...t)=>{console.warn(`[DB:WARN] ${e}`,...t)},error:(e,...t)=>{console.error(`[DB:ERROR] ${e}`,...t)},log:(e,...t)=>{isDevelopment&&console.log(`[DB:LOG] ${e}`,...t)}}}}),GuestUsername,log,DocType,DocTypePrefixes,init_types_legacy=__esm({"src/core/types/types-legacy.ts"(){init_logger(),GuestUsername=`sk-guest-`,log=e=>{logger.log(e)},DocType=(e=>(e.DISPLAYABLE_DATA=`DISPLAYABLE_DATA`,e.CARD=`CARD`,e.DATASHAPE=`DATASHAPE`,e.QUESTIONTYPE=`QUESTION`,e.VIEW=`VIEW`,e.PEDAGOGY=`PEDAGOGY`,e.CARDRECORD=`CARDRECORD`,e.SCHEDULED_CARD=`SCHEDULED_CARD`,e.TAG=`TAG`,e.NAVIGATION_STRATEGY=`NAVIGATION_STRATEGY`,e.STRATEGY_STATE=`STRATEGY_STATE`,e.USER_OUTCOME=`USER_OUTCOME`,e.STRATEGY_LEARNING_STATE=`STRATEGY_LEARNING_STATE`,e))(DocType||{}),DocTypePrefixes={CARD:`c`,DISPLAYABLE_DATA:`dd`,TAG:`TAG`,CARDRECORD:`cardH`,SCHEDULED_CARD:`card_review_`,DATASHAPE:`DATASHAPE`,QUESTION:`QUESTION`,VIEW:`VIEW`,PEDAGOGY:`PEDAGOGY`,NAVIGATION_STRATEGY:`NAVIGATION_STRATEGY`,STRATEGY_STATE:`STRATEGY_STATE`,USER_OUTCOME:`USER_OUTCOME`,STRATEGY_LEARNING_STATE:`STRATEGY_LEARNING_STATE`}}});function areQuestionRecords(e){return isQuestionRecord(e.records[0])}function isQuestionRecord(e){return e.userAnswer!==void 0}function getCardHistoryID(e,t){return`${DocTypePrefixes.CARDRECORD}-${e}-${t}`}function parseCardHistoryID(e){let t=e.split(`-`),n=``;if(n+=t.length===3?``:`
12
12
  given ID has incorrect number of '-' characters`,n+=t[0]===DocTypePrefixes.CARDRECORD?``:`
13
13
  given ID does not start with ${DocTypePrefixes.CARDRECORD}`,t.length===3&&t[0]===DocTypePrefixes.CARDRECORD)return{courseID:t[1],cardID:t[2]};throw Error(`parseCardHistory Error:`+n)}function docIsDeleted(e){return e?.error===`not_found`&&e?.reason===`deleted`}var init_util=__esm({"src/core/util/index.ts"(){init_types_legacy()}}),pouchdb_setup_default,init_pouchdb_setup=__esm({"src/impl/couch/pouchdb-setup.ts"(){PouchDB.plugin(plugin$1),PouchDB.plugin(plugin),PouchDB.debug!==void 0&&PouchDB.debug.disable(),PouchDB.defaults({}),pouchdb_setup_default=PouchDB}});function getAppDataDirectory(){return ENV.LOCAL_STORAGE_PREFIX?import___vite_browser_external.join(import___vite_browser_external.homedir(),`.tuilder`,ENV.LOCAL_STORAGE_PREFIX):import___vite_browser_external.join(import___vite_browser_external.homedir(),`.tuilder`)}async function ensureAppDataDirectory(){let e=getAppDataDirectory();try{await import___vite_browser_external.promises.mkdir(e,{recursive:!0})}catch(t){if(t.code!==`EEXIST`)throw Error(`Failed to create app data directory ${e}: ${t.message}`)}return e}function getDbPath(e){return import___vite_browser_external.join(getAppDataDirectory(),e)}async function initializeDataDirectory(){await ensureAppDataDirectory(),logger.info(`PouchDB data directory initialized: ${getAppDataDirectory()}`)}var init_dataDirectory=__esm({"src/util/dataDirectory.ts"(){init_logger(),init_factory()}});function hexEncode(e){let t,n=``;for(let r=0;r<e.length;r++)t=e.charCodeAt(r).toString(16),n+=(`000`+t).slice(3);return n}function filterAllDocsByPrefix(e,t,n){let r={startkey:t,endkey:t+`￰`,include_docs:!0};return n&&Object.assign(r,n),e.allDocs(r)}function getStartAndEndKeys(e){return{startkey:e,endkey:e+`￰`}}function updateGuestAccountExpirationDate(e){let t=hooks.utc().add(2,`months`).toISOString(),n=`GuestAccountExpirationDate`;e.get(n).then(r=>e.put({_id:n,_rev:r._rev,date:t})).catch(()=>e.put({_id:n,date:t}))}function getLocalUserDB(e){let t=`userdb-${e}`;return typeof window>`u`?new pouchdb_setup_default(getDbPath(t),{}):new pouchdb_setup_default(t,{})}function scheduleCardReviewLocal(e,t){let n=hooks.utc();logger.info(`Scheduling for review in: ${t.time.diff(n,`h`)/24} days`),e.put({_id:DocTypePrefixes.SCHEDULED_CARD+t.time.format(REVIEW_TIME_FORMAT),cardId:t.card_id,reviewTime:t.time.toISOString(),courseId:t.course_id,scheduledAt:n.toISOString(),scheduledFor:t.scheduledFor,schedulingAgentId:t.schedulingAgentId})}async function removeScheduledCardReviewLocal(e,t){let n=await e.get(t);e.remove(n).then(e=>{e.ok&&log2(`Removed Review Doc: ${t}`)}).catch(e=>{log2(`Failed to remove Review Doc: ${t},
14
- ${JSON.stringify(e)}`)})}var REVIEW_TIME_FORMAT,log2,init_userDBHelpers=__esm({"src/impl/common/userDBHelpers.ts"(){init_core(),init_logger(),init_pouchdb_setup(),init_dataDirectory(),REVIEW_TIME_FORMAT=`YYYY-MM-DD--kk:mm:ss-SSS`,log2=e=>{logger.info(e)}}}),Loggable,init_Loggable=__esm({"src/util/Loggable.ts"(){Loggable=class{log(...e){console.log(`LOG-${this._className}@${new Date}:`,...e)}error(...e){console.error(`ERROR-${this._className}@${new Date}:`,...e)}}}}),UpdateQueue,init_updateQueue=__esm({"src/impl/couch/updateQueue.ts"(){init_Loggable(),init_logger(),UpdateQueue=class extends Loggable{update(e,t){return logger.debug(`Update requested on doc: ${e}`),this.pendingUpdates[e]?this.pendingUpdates[e].push(t):this.pendingUpdates[e]=[t],this.applyUpdates(e)}constructor(e,t){super(),_defineProperty(this,`_className`,`UpdateQueue`),_defineProperty(this,`pendingUpdates`,{}),_defineProperty(this,`inprogressUpdates`,{}),_defineProperty(this,`readDB`,void 0),_defineProperty(this,`writeDB`,void 0),this.readDB=e,this.writeDB=t||e,logger.debug(`UpdateQ initialized...`),this.readDB.info().then(e=>{logger.debug(`db info: ${JSON.stringify(e)}`)})}async applyUpdates(e){if(logger.debug(`Applying updates on doc: ${e}`),this.inprogressUpdates[e]){for(;this.inprogressUpdates[e];)await new Promise(e=>setTimeout(e,Math.random()*50));return this.applyUpdates(e)}else if(this.pendingUpdates[e]&&this.pendingUpdates[e].length>0){this.inprogressUpdates[e]=!0;let t=5;for(let t=0;t<5;t++)try{let t={...await this.readDB.get(e)},n=[...this.pendingUpdates[e]];for(let e of n)t=typeof e==`function`?{...t,...e(t)}:{...t,...e};if(await this.writeDB.put(t),this.pendingUpdates[e].splice(0,n.length),this.pendingUpdates[e].length===0)this.inprogressUpdates[e]=!1,delete this.inprogressUpdates[e];else return this.applyUpdates(e);return t}catch(n){if(n.name===`conflict`&&t<4)logger.warn(`Conflict on update for doc ${e}, retry #${t+1}`),await new Promise(e=>setTimeout(e,50*Math.random()));else if(n.name===`not_found`&&t===0)throw logger.warn(`Update failed for ${e} - does not exist. Throwing to caller.`),delete this.inprogressUpdates[e],n;else throw delete this.inprogressUpdates[e],this.pendingUpdates[e]&&delete this.pendingUpdates[e],logger.error(`Error on attemped update (retry ${t}): ${JSON.stringify(n)}`),n}throw Error(`UpdateQueue failed for doc ${e} after 5 retries.`)}else throw Error(`Empty Updates Queue Triggered`)}}}}),UsrCrsData,init_user_course_relDB=__esm({"src/impl/couch/user-course-relDB.ts"(){init_logger(),UsrCrsData=class{constructor(e,t){_defineProperty(this,`user`,void 0),_defineProperty(this,`_courseId`,void 0),this.user=e,this._courseId=t}async getReviewsForcast(e){let t=hooks.utc().add(e,`days`);return this.getReviewstoDate(t)}async getPendingReviews(){let e=hooks.utc();return this.getReviewstoDate(e)}async getScheduledReviewCount(){return(await this.getPendingReviews()).length}async getCourseSettings(){let e=(await this.user.getCourseRegistrationsDoc()).courses.find(e=>e.courseID===this._courseId);return e&&e.settings?e.settings:(logger.warn(`no settings found during lookup on course ${this._courseId}`),{})}updateCourseSettings(e){`updateCourseSettings`in this.user&&this.user.updateCourseSettings(this._courseId,e)}async getStrategyState(e){return this.user.getStrategyState(this._courseId,e)}async putStrategyState(e,t){return this.user.putStrategyState(this._courseId,e,t)}async deleteStrategyState(e){return this.user.deleteStrategyState(this._courseId,e)}async getReviewstoDate(e){let t=await this.user.getPendingReviews(this._courseId);return logger.debug(`Fetching ${this.user.getUsername()}'s scheduled reviews for course ${this._courseId}.`),t.filter(t=>{let n=hooks.utc(t.reviewTime);return e.isAfter(n)})}}}});async function GET_CACHED(e,t){return CLIENT_CACHE[e]?CLIENT_CACHE[e]:(CLIENT_CACHE[e]=t?await t(e):await GET_ITEM(e),GET_CACHED(e))}async function GET_ITEM(e){throw Error(`No implementation found for GET_CACHED(${e})`)}var CLIENT_CACHE,init_clientCache=__esm({"src/impl/couch/clientCache.ts"(){CLIENT_CACHE={}}});async function addNote55(e,t,n,r,a,o,s,c=blankCourseElo()){let l=getCourseDB(e),u=prepareNote55(e,t,n,r,a,o,s),d=`${DocTypePrefixes.DISPLAYABLE_DATA}-${v4()}`,p=await l.put({...u,_id:d}),m=NameSpacer.getDataShapeString({course:t,dataShape:n.name});if(p.ok)try{await createCards(e,m,p.id,o,c,a)}catch(e){let t=`Unknown error`;t=e instanceof Error?e.message:e&&typeof e==`object`&&`reason`in e?e.reason:e&&typeof e==`object`&&`message`in e?e.message:String(e),logger.error(`[addNote55] Failed to create cards for note ${p.id}: ${t}`),p.cardCreationFailed=!0,p.cardCreationError=t}else logger.error(`[addNote55] Error adding note. Result: ${JSON.stringify(p)}`);return p}async function createCards(e,t,n,r,a=blankCourseElo(),o){let s=await getCredentialledCourseConfig(e),c=NameSpacer.getDataShapeDescriptor(t),l=[];for(let e of s.dataShapes)e.name===t&&(l=e.questionTypes);if(l.length===0){let e=`No questionViewTypes found for datashapeID: ${t} in course config. Cards cannot be created.`;throw logger.error(e),Error(e)}for(let t of l)await createCard(t,e,c,n,r,a,o)}async function createCard(e,t,n,r,a,o=blankCourseElo(),s){let c=NameSpacer.getQuestionDescriptor(e),l=await getCredentialledCourseConfig(t);for(let u of l.questionTypes)if(u.name===e)for(let e of u.viewList)await addCard(t,n.course,[r],NameSpacer.getViewString({course:c.course,questionType:c.questionType,view:e}),o,a,s)}async function addCard(e,t,n,r,a,o,s){let c=getCourseDB(e),l=`${DocTypePrefixes.CARD}-${v4()}`,u=await c.put({_id:l,course:t,id_displayable_data:n,id_view:r,docType:`CARD`,elo:a||toCourseElo(990+Math.round(20*Math.random())),author:s});for(let t of o)logger.info(`adding tag: ${t} to card ${u.id}`),await addTagToCard(e,u.id,t,s,!1);return u}async function getCredentialledCourseConfig(e){try{let t=await getCourseDB(e).get(`CourseConfig`);return t.courseID=e,logger.info(`Returning course config: ${JSON.stringify(t)}`),t}catch(t){throw logger.error(`Error fetching config for ${e}:`,t),t}}async function addTagToCard(e,t,n,r,a=!0){let o=getTagID(n),s=getCourseDB(e),c=new CourseDB(e,async()=>BaseUser.Dummy({setupRemoteDB:()=>null,startSync:()=>{},canCreateAccount:()=>!1,canAuthenticate:()=>!1,getCurrentUsername:async()=>`DummyUser`}));try{logger.info(`Applying tag ${n} to card ${e+`-`+t}...`);let r=await s.get(o);if(r.taggedCards.includes(t))throw new AlreadyTaggedErr(`Card ${t} is already tagged with ${n}`);if(r.taggedCards.push(t),a)try{let r=(await c.getCardEloData([t]))[0];r.tags[n]={count:0,score:r.global.score},await updateCardElo(e,t,r)}catch(e){logger.error(`Failed to update ELO data for card:`,t,e)}return s.put(r)}catch(o){if(o instanceof AlreadyTaggedErr)throw o;return await createTag(e,n,r),addTagToCard(e,t,n,r,a)}}async function updateCardElo(e,t,n){if(n){let r=getCourseDB(e),a=await r.get(t);return logger.debug(`Replacing ${JSON.stringify(a.elo)} with ${JSON.stringify(n)}`),a.elo=n,r.put(a)}}function getTagID(e){let t=`TAG`.valueOf()+`-`;return e.indexOf(t)===0?e:t+e}function getCourseDB(e){let t=`coursedb-${e}`;return new pouchdb_setup_default(ENV.COUCHDB_SERVER_PROTOCOL+`://`+ENV.COUCHDB_SERVER_URL+t,createPouchDBConfig())}var AlreadyTaggedErr,init_courseAPI=__esm({"src/impl/couch/courseAPI.ts"(){init_pouchdb_setup(),init_couch(),init_factory(),init_courseDB(),init_types_legacy(),init_common(),init_logger(),AlreadyTaggedErr=class extends Error{constructor(e){super(e),this.name=`AlreadyTaggedErr`}}}}),courseLookupDBTitle,CourseLookup,init_courseLookupDB=__esm({"src/impl/couch/courseLookupDB.ts"(){var e;init_pouchdb_setup(),init_factory(),init_logger(),courseLookupDBTitle=`coursedb-lookup`,logger.debug(`COURSELOOKUP FILE RUNNING`),CourseLookup=(e=class _CourseLookup{static get _db(){if(this._dbInstance)return this._dbInstance;if(ENV.COUCHDB_SERVER_URL===`NOT_SET`||!ENV.COUCHDB_SERVER_URL)throw Error(`CourseLookup.db: COUCHDB_SERVER_URL is not set. Ensure initializeDataLayer has been called with valid configuration.`);if(ENV.COUCHDB_SERVER_PROTOCOL===`NOT_SET`||!ENV.COUCHDB_SERVER_PROTOCOL)throw Error(`CourseLookup.db: COUCHDB_SERVER_PROTOCOL is not set. Ensure initializeDataLayer has been called with valid configuration.`);let e=`${ENV.COUCHDB_SERVER_PROTOCOL}://${ENV.COUCHDB_SERVER_URL}/${courseLookupDBTitle}`,t={};ENV.COUCHDB_USERNAME&&ENV.COUCHDB_PASSWORD?(t.auth={username:ENV.COUCHDB_USERNAME,password:ENV.COUCHDB_PASSWORD},logger.info(`CourseLookup: Connecting to ${e} with authentication.`)):logger.info(`CourseLookup: Connecting to ${e} without authentication.`);try{return this._dbInstance=new pouchdb_setup_default(e,t),logger.info(`CourseLookup: Database instance created for ${courseLookupDBTitle}.`),this._dbInstance}catch(t){throw logger.error(`CourseLookup: Failed to create PouchDB instance for ${e}`,t),this._dbInstance=null,Error(`CourseLookup: Failed to initialize database connection: ${t instanceof Error?t.message:String(t)}`)}}static async add(e){return(await _CourseLookup._db.post({name:e})).id}static async addWithId(e,t,n){let r={_id:e,name:t};n&&(r.disambiguator=n),await _CourseLookup._db.put(r)}static async delete(e){let t=await _CourseLookup._db.get(e);return await _CourseLookup._db.remove(t)}static async allCourseWare(){return(await _CourseLookup._db.allDocs({include_docs:!0})).rows.map(e=>e.doc)}static async updateDisambiguator(e,t){let n=await _CourseLookup._db.get(e);return n.disambiguator=t,await _CourseLookup._db.put(n)}static async isCourse(e){try{return await _CourseLookup._db.get(e),!0}catch(e){return logger.info(`Courselookup failed:`,e),!1}}},_defineProperty(e,`_dbInstance`,null),e)}}),diversityRerank_exports={};__export(diversityRerank_exports,{DIVERSITY_FLOOR:()=>DIVERSITY_FLOOR,DIVERSITY_STRENGTH:()=>DIVERSITY_STRENGTH,diversityRerank:()=>diversityRerank});function diversityRerank(e,t={}){let n=t.strength??DIVERSITY_STRENGTH,r=t.floor??DIVERSITY_FLOOR,a=e.length;if(a<=1)return e;let o=new Map;for(let t of e)for(let e of t.tags??[])o.set(e,(o.get(e)??0)+1);let s=new Map;for(let[e,t]of o)s.set(e,Math.log(a/t));let c=[...e],l=new Map,u=[],repetitionLoad=e=>{let t=0;for(let n of e.tags??[]){let e=l.get(n);e&&(t+=(s.get(n)??0)*e)}return t};for(;c.length>0;){let e=0,t=-1/0,a=1,o=0;for(let s=0;s<c.length;s++){let l=c[s],u=repetitionLoad(l),d=u>0?Math.max(r,1/(1+n*u)):1,p=l.score*d;p>t&&(t=p,e=s,a=d,o=u)}let[s]=c.splice(e,1);if(Number.isFinite(s.score)&&a<1){let e=s.score*a;u.push({...s,score:e,provenance:[...s.provenance,{strategy:STRATEGY,strategyId:STRATEGY_ID,strategyName:STRATEGY_NAME,action:`penalized`,score:e,reason:`repeated tags (load ${o.toFixed(2)}) \u2192 \xD7${a.toFixed(2)}`}]})}else u.push(s);for(let e of s.tags??[])l.set(e,(l.get(e)??0)+1)}return u}var DIVERSITY_STRENGTH,DIVERSITY_FLOOR,STRATEGY,STRATEGY_ID,STRATEGY_NAME,init_diversityRerank=__esm({"src/core/navigators/diversityRerank.ts"(){DIVERSITY_STRENGTH=.6,DIVERSITY_FLOOR=.3,STRATEGY=`diversityRerank`,STRATEGY_ID=`DIVERSITY_RERANK`,STRATEGY_NAME=`Diversity Re-rank`}}),PipelineDebugger_exports={};__export(PipelineDebugger_exports,{buildRunReport:()=>buildRunReport,captureRun:()=>captureRun,clearRunHistory:()=>clearRunHistory,mountPipelineDebugger:()=>mountPipelineDebugger,pipelineDebugAPI:()=>pipelineDebugAPI,registerPipelineForDebug:()=>registerPipelineForDebug});function registerPipelineForDebug(e){_activePipeline=e}function clearRunHistory(){runHistory.length=0}function getOrigin(e){let t=e.provenance[0];if(!t)return`unknown`;let n=t.reason?.toLowerCase()||``,r=t.strategy?.toLowerCase()||``;return n.includes(`new card`)||r.includes(`elo`)?`new`:n.includes(`review`)||r.includes(`srs`)?`review`:`unknown`}function captureRun(e){let t={...e,runId:`run-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,timestamp:new Date};runHistory.unshift(t),runHistory.length>MAX_RUNS&&runHistory.pop()}function parseCardElo(e){let t=e.find(e=>e.strategy===`elo`);if(!t?.reason)return;let n=t.reason.match(/card:\s*(\d+)/);return n?parseInt(n[1],10):void 0}function buildRunReport(e,t,n,r,a,o,s,c,l,u){let d=new Set(c.map(e=>e.cardId)),toReport=e=>({cardId:e.cardId,courseId:e.courseId,origin:getOrigin(e),generator:e.provenance[0]?.strategyName||e.provenance[0]?.strategy,finalScore:e.score,cardElo:parseCardElo(e.provenance),provenance:e.provenance,tags:e.tags,selected:d.has(e.cardId)}),p=[],m=[],g=[],_=0;for(let e of s)d.has(e.cardId)?p.push(toReport(e)):_<DISCARDED_KEEP_TOP?(m.push(toReport(e)),_++):g.push(e);let v=[...p,...m],y;if(g.length>0){let e=1/0,t=-1/0,n=1/0,r=-1/0,a=!1;for(let o of g){o.score<e&&(e=o.score),o.score>t&&(t=o.score);let s=parseCardElo(o.provenance);s!==void 0&&(a=!0,s<n&&(n=s),s>r&&(r=s))}let o=a?`, ELO ${n}\u2013${r}`:``;y={count:g.length,scoreRange:[e,t],eloRange:a?[n,r]:void 0,note:`${g.length} additional candidate(s) scored below the top ${DISCARDED_KEEP_TOP} near-misses and were not retained (score ${e.toExponential(2)}\u2013${t.toExponential(2)}${o}). Likely ELO-window pull remnants filtered out by hierarchy/lesson/priority gates.`}}let b=c.filter(e=>getOrigin(e)===`review`).length,x=c.filter(e=>getOrigin(e)===`new`).length;return{courseId:e,courseName:t,userElo:l,generatorName:n,generators:r,generatedCount:a,filters:o,hints:u,finalCount:c.length,reviewsSelected:b,newSelected:x,cards:v,discardedTail:y}}function formatProvenance(e){return e.map(e=>` ${e.action===`generated`?`🎲`:e.action===`boosted`?`⬆️`:e.action===`penalized`?`⬇️`:`➡️`} ${e.strategyName}: ${e.score.toFixed(3)} - ${e.reason}`).join(`
14
+ ${JSON.stringify(e)}`)})}var REVIEW_TIME_FORMAT,log2,init_userDBHelpers=__esm({"src/impl/common/userDBHelpers.ts"(){init_core(),init_logger(),init_pouchdb_setup(),init_dataDirectory(),REVIEW_TIME_FORMAT=`YYYY-MM-DD--kk:mm:ss-SSS`,log2=e=>{logger.info(e)}}}),Loggable,init_Loggable=__esm({"src/util/Loggable.ts"(){Loggable=class{log(...e){console.log(`LOG-${this._className}@${new Date}:`,...e)}error(...e){console.error(`ERROR-${this._className}@${new Date}:`,...e)}}}}),UpdateQueue,init_updateQueue=__esm({"src/impl/couch/updateQueue.ts"(){init_Loggable(),init_logger(),UpdateQueue=class extends Loggable{update(e,t){return logger.debug(`Update requested on doc: ${e}`),this.pendingUpdates[e]?this.pendingUpdates[e].push(t):this.pendingUpdates[e]=[t],this.applyUpdates(e)}constructor(e,t){super(),_defineProperty(this,`_className`,`UpdateQueue`),_defineProperty(this,`pendingUpdates`,{}),_defineProperty(this,`inprogressUpdates`,{}),_defineProperty(this,`readDB`,void 0),_defineProperty(this,`writeDB`,void 0),this.readDB=e,this.writeDB=t||e,logger.debug(`UpdateQ initialized...`),this.readDB.info().then(e=>{logger.debug(`db info: ${JSON.stringify(e)}`)})}async applyUpdates(e){if(logger.debug(`Applying updates on doc: ${e}`),this.inprogressUpdates[e]){for(;this.inprogressUpdates[e];)await new Promise(e=>setTimeout(e,Math.random()*50));return this.applyUpdates(e)}else if(this.pendingUpdates[e]&&this.pendingUpdates[e].length>0){this.inprogressUpdates[e]=!0;let t=5;for(let t=0;t<5;t++)try{let t={...await this.readDB.get(e)},n=[...this.pendingUpdates[e]];for(let e of n)t=typeof e==`function`?{...t,...e(t)}:{...t,...e};if(await this.writeDB.put(t),this.pendingUpdates[e].splice(0,n.length),this.pendingUpdates[e].length===0)this.inprogressUpdates[e]=!1,delete this.inprogressUpdates[e];else return this.applyUpdates(e);return t}catch(n){if(n.name===`conflict`&&t<4)logger.warn(`Conflict on update for doc ${e}, retry #${t+1}`),await new Promise(e=>setTimeout(e,50*Math.random()));else if(n.name===`not_found`&&t===0)throw logger.warn(`Update failed for ${e} - does not exist. Throwing to caller.`),delete this.inprogressUpdates[e],n;else throw delete this.inprogressUpdates[e],this.pendingUpdates[e]&&delete this.pendingUpdates[e],logger.error(`Error on attemped update (retry ${t}): ${JSON.stringify(n)}`),n}throw Error(`UpdateQueue failed for doc ${e} after 5 retries.`)}else throw Error(`Empty Updates Queue Triggered`)}}}}),UsrCrsData,init_user_course_relDB=__esm({"src/impl/couch/user-course-relDB.ts"(){init_logger(),UsrCrsData=class{constructor(e,t){_defineProperty(this,`user`,void 0),_defineProperty(this,`_courseId`,void 0),this.user=e,this._courseId=t}async getReviewsForcast(e){let t=hooks.utc().add(e,`days`);return this.getReviewstoDate(t)}async getPendingReviews(){let e=hooks.utc();return this.getReviewstoDate(e)}async getScheduledReviewCount(){return(await this.getPendingReviews()).length}async getCourseSettings(){let e=(await this.user.getCourseRegistrationsDoc()).courses.find(e=>e.courseID===this._courseId);return e&&e.settings?e.settings:(logger.warn(`no settings found during lookup on course ${this._courseId}`),{})}updateCourseSettings(e){`updateCourseSettings`in this.user&&this.user.updateCourseSettings(this._courseId,e)}async getStrategyState(e){return this.user.getStrategyState(this._courseId,e)}async putStrategyState(e,t){return this.user.putStrategyState(this._courseId,e,t)}async deleteStrategyState(e){return this.user.deleteStrategyState(this._courseId,e)}async getReviewstoDate(e){let t=await this.user.getPendingReviews(this._courseId);return logger.debug(`Fetching ${this.user.getUsername()}'s scheduled reviews for course ${this._courseId}.`),t.filter(t=>{let n=hooks.utc(t.reviewTime);return e.isAfter(n)})}}}});async function GET_CACHED(e,t){return CLIENT_CACHE[e]?CLIENT_CACHE[e]:(CLIENT_CACHE[e]=t?await t(e):await GET_ITEM(e),GET_CACHED(e))}async function GET_ITEM(e){throw Error(`No implementation found for GET_CACHED(${e})`)}var CLIENT_CACHE,init_clientCache=__esm({"src/impl/couch/clientCache.ts"(){CLIENT_CACHE={}}});async function addNote55(e,t,n,r,a,o,s,c=blankCourseElo()){let l=getCourseDB(e),u=prepareNote55(e,t,n,r,a,o,s),d=`${DocTypePrefixes.DISPLAYABLE_DATA}-${v4()}`,p=await l.put({...u,_id:d}),m=NameSpacer.getDataShapeString({course:t,dataShape:n.name});if(p.ok)try{await createCards(e,m,p.id,o,c,a)}catch(e){let t=`Unknown error`;t=e instanceof Error?e.message:e&&typeof e==`object`&&`reason`in e?e.reason:e&&typeof e==`object`&&`message`in e?e.message:String(e),logger.error(`[addNote55] Failed to create cards for note ${p.id}: ${t}`),p.cardCreationFailed=!0,p.cardCreationError=t}else logger.error(`[addNote55] Error adding note. Result: ${JSON.stringify(p)}`);return p}async function createCards(e,t,n,r,a=blankCourseElo(),o){let s=await getCredentialledCourseConfig(e),c=NameSpacer.getDataShapeDescriptor(t),l=[];for(let e of s.dataShapes)e.name===t&&(l=e.questionTypes);if(l.length===0){let e=`No questionViewTypes found for datashapeID: ${t} in course config. Cards cannot be created.`;throw logger.error(e),Error(e)}for(let t of l)await createCard(t,e,c,n,r,a,o)}async function createCard(e,t,n,r,a,o=blankCourseElo(),s){let c=NameSpacer.getQuestionDescriptor(e),l=await getCredentialledCourseConfig(t);for(let u of l.questionTypes)if(u.name===e)for(let e of u.viewList)await addCard(t,n.course,[r],NameSpacer.getViewString({course:c.course,questionType:c.questionType,view:e}),o,a,s)}async function addCard(e,t,n,r,a,o,s){let c=getCourseDB(e),l=`${DocTypePrefixes.CARD}-${v4()}`,u=await c.put({_id:l,course:t,id_displayable_data:n,id_view:r,docType:`CARD`,elo:a||toCourseElo(990+Math.round(20*Math.random())),author:s});for(let t of o)logger.info(`adding tag: ${t} to card ${u.id}`),await addTagToCard(e,u.id,t,s,!1);return u}async function getCredentialledCourseConfig(e){try{let t=await getCourseDB(e).get(`CourseConfig`);return t.courseID=e,logger.info(`Returning course config: ${JSON.stringify(t)}`),t}catch(t){throw logger.error(`Error fetching config for ${e}:`,t),t}}async function addTagToCard(e,t,n,r,a=!0){let o=getTagID(n),s=getCourseDB(e),c=new CourseDB(e,async()=>BaseUser.Dummy({setupRemoteDB:()=>null,startSync:()=>{},canCreateAccount:()=>!1,canAuthenticate:()=>!1,getCurrentUsername:async()=>`DummyUser`}));try{logger.info(`Applying tag ${n} to card ${e+`-`+t}...`);let r=await s.get(o);if(r.taggedCards.includes(t))throw new AlreadyTaggedErr(`Card ${t} is already tagged with ${n}`);if(r.taggedCards.push(t),a)try{let r=(await c.getCardEloData([t]))[0];r.tags[n]={count:0,score:r.global.score},await updateCardElo(e,t,r)}catch(e){logger.error(`Failed to update ELO data for card:`,t,e)}return s.put(r)}catch(o){if(o instanceof AlreadyTaggedErr)throw o;return await createTag(e,n,r),addTagToCard(e,t,n,r,a)}}async function updateCardElo(e,t,n){if(n){let r=getCourseDB(e),a=await r.get(t);return logger.debug(`Replacing ${JSON.stringify(a.elo)} with ${JSON.stringify(n)}`),a.elo=n,r.put(a)}}function getTagID(e){let t=`TAG`.valueOf()+`-`;return e.indexOf(t)===0?e:t+e}function getCourseDB(e){let t=`coursedb-${e}`;return new pouchdb_setup_default(ENV.COUCHDB_SERVER_PROTOCOL+`://`+ENV.COUCHDB_SERVER_URL+t,createPouchDBConfig())}var AlreadyTaggedErr,init_courseAPI=__esm({"src/impl/couch/courseAPI.ts"(){init_pouchdb_setup(),init_couch(),init_factory(),init_courseDB(),init_types_legacy(),init_common(),init_logger(),AlreadyTaggedErr=class extends Error{constructor(e){super(e),this.name=`AlreadyTaggedErr`}}}}),courseLookupDBTitle,CourseLookup,init_courseLookupDB=__esm({"src/impl/couch/courseLookupDB.ts"(){var e;init_pouchdb_setup(),init_factory(),init_logger(),courseLookupDBTitle=`coursedb-lookup`,logger.debug(`COURSELOOKUP FILE RUNNING`),CourseLookup=(e=class _CourseLookup{static get _db(){if(this._dbInstance)return this._dbInstance;if(ENV.COUCHDB_SERVER_URL===`NOT_SET`||!ENV.COUCHDB_SERVER_URL)throw Error(`CourseLookup.db: COUCHDB_SERVER_URL is not set. Ensure initializeDataLayer has been called with valid configuration.`);if(ENV.COUCHDB_SERVER_PROTOCOL===`NOT_SET`||!ENV.COUCHDB_SERVER_PROTOCOL)throw Error(`CourseLookup.db: COUCHDB_SERVER_PROTOCOL is not set. Ensure initializeDataLayer has been called with valid configuration.`);let e=`${ENV.COUCHDB_SERVER_PROTOCOL}://${ENV.COUCHDB_SERVER_URL}/${courseLookupDBTitle}`,t={};ENV.COUCHDB_USERNAME&&ENV.COUCHDB_PASSWORD?(t.auth={username:ENV.COUCHDB_USERNAME,password:ENV.COUCHDB_PASSWORD},logger.info(`CourseLookup: Connecting to ${e} with authentication.`)):logger.info(`CourseLookup: Connecting to ${e} without authentication.`);try{return this._dbInstance=new pouchdb_setup_default(e,t),logger.info(`CourseLookup: Database instance created for ${courseLookupDBTitle}.`),this._dbInstance}catch(t){throw logger.error(`CourseLookup: Failed to create PouchDB instance for ${e}`,t),this._dbInstance=null,Error(`CourseLookup: Failed to initialize database connection: ${t instanceof Error?t.message:String(t)}`)}}static async add(e){return(await _CourseLookup._db.post({name:e})).id}static async addWithId(e,t,n){let r={_id:e,name:t};n&&(r.disambiguator=n),await _CourseLookup._db.put(r)}static async delete(e){let t=await _CourseLookup._db.get(e);return await _CourseLookup._db.remove(t)}static async allCourseWare(){return(await _CourseLookup._db.allDocs({include_docs:!0})).rows.map(e=>e.doc)}static async updateDisambiguator(e,t){let n=await _CourseLookup._db.get(e);return n.disambiguator=t,await _CourseLookup._db.put(n)}static async isCourse(e){try{return await _CourseLookup._db.get(e),!0}catch(e){return logger.info(`Courselookup failed:`,e),!1}}},_defineProperty(e,`_dbInstance`,null),e)}}),diversityRerank_exports={};__export(diversityRerank_exports,{DIVERSITY_FLOOR:()=>DIVERSITY_FLOOR,DIVERSITY_STRENGTH:()=>DIVERSITY_STRENGTH,diversityRerank:()=>diversityRerank});function diversityRerank(e,t={}){let n=t.strength??DIVERSITY_STRENGTH,r=t.floor??DIVERSITY_FLOOR,a=e.length;if(a<=1)return e;let o=new Map;for(let t of e)for(let e of t.tags??[])o.set(e,(o.get(e)??0)+1);let s=new Map;for(let[e,t]of o)s.set(e,Math.log(a/t));let c=[...e],l=new Map,u=[],repetitionLoad=e=>{let t=0;for(let n of e.tags??[]){let e=l.get(n);e&&(t+=(s.get(n)??0)*e)}return t};for(;c.length>0;){let e=0,t=-1/0,a=1,o=0;for(let s=0;s<c.length;s++){let l=c[s],u=repetitionLoad(l),d=u>0?Math.max(r,1/(1+n*u)):1,p=l.score*d;p>t&&(t=p,e=s,a=d,o=u)}let[s]=c.splice(e,1);if(Number.isFinite(s.score)&&a<1){let e=s.score*a;u.push({...s,score:e,provenance:[...s.provenance,{strategy:STRATEGY,strategyId:STRATEGY_ID,strategyName:STRATEGY_NAME,action:`penalized`,score:e,reason:`repeated tags (load ${o.toFixed(2)}) \u2192 \xD7${a.toFixed(2)}`}]})}else u.push(s);for(let e of s.tags??[])l.set(e,(l.get(e)??0)+1)}return u}var DIVERSITY_STRENGTH,DIVERSITY_FLOOR,STRATEGY,STRATEGY_ID,STRATEGY_NAME,init_diversityRerank=__esm({"src/core/navigators/diversityRerank.ts"(){DIVERSITY_STRENGTH=.6,DIVERSITY_FLOOR=.3,STRATEGY=`diversityRerank`,STRATEGY_ID=`DIVERSITY_RERANK`,STRATEGY_NAME=`Diversity Re-rank`}}),PipelineDebugger_exports={};__export(PipelineDebugger_exports,{buildRunReport:()=>buildRunReport,captureRun:()=>captureRun,clearRunHistory:()=>clearRunHistory,getActivePipeline:()=>getActivePipeline,mountPipelineDebugger:()=>mountPipelineDebugger,pipelineDebugAPI:()=>pipelineDebugAPI,registerPipelineForDebug:()=>registerPipelineForDebug});function registerPipelineForDebug(e){_activePipeline=e}function getActivePipeline(){return _activePipeline}function clearRunHistory(){runHistory.length=0}function getOrigin(e){let t=e.provenance[0];if(!t)return`unknown`;let n=t.reason?.toLowerCase()||``,r=t.strategy?.toLowerCase()||``;return n.includes(`new card`)||r.includes(`elo`)?`new`:n.includes(`review`)||r.includes(`srs`)?`review`:`unknown`}function captureRun(e){let t={...e,runId:`run-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,timestamp:new Date};runHistory.unshift(t),runHistory.length>MAX_RUNS&&runHistory.pop()}function parseCardElo(e){let t=e.find(e=>e.strategy===`elo`);if(!t?.reason)return;let n=t.reason.match(/card:\s*(\d+)/);return n?parseInt(n[1],10):void 0}function buildRunReport(e,t,n,r,a,o,s,c,l,u){let d=new Set(c.map(e=>e.cardId)),toReport=e=>({cardId:e.cardId,courseId:e.courseId,origin:getOrigin(e),generator:e.provenance[0]?.strategyName||e.provenance[0]?.strategy,finalScore:e.score,cardElo:parseCardElo(e.provenance),provenance:e.provenance,tags:e.tags,selected:d.has(e.cardId)}),p=[],m=[],g=[],_=0;for(let e of s)d.has(e.cardId)?p.push(toReport(e)):_<DISCARDED_KEEP_TOP?(m.push(toReport(e)),_++):g.push(e);let v=[...p,...m],y;if(g.length>0){let e=1/0,t=-1/0,n=1/0,r=-1/0,a=!1;for(let o of g){o.score<e&&(e=o.score),o.score>t&&(t=o.score);let s=parseCardElo(o.provenance);s!==void 0&&(a=!0,s<n&&(n=s),s>r&&(r=s))}let o=a?`, ELO ${n}\u2013${r}`:``;y={count:g.length,scoreRange:[e,t],eloRange:a?[n,r]:void 0,note:`${g.length} additional candidate(s) scored below the top ${DISCARDED_KEEP_TOP} near-misses and were not retained (score ${e.toExponential(2)}\u2013${t.toExponential(2)}${o}). Likely ELO-window pull remnants filtered out by hierarchy/lesson/priority gates.`}}let b=c.filter(e=>getOrigin(e)===`review`).length,x=c.filter(e=>getOrigin(e)===`new`).length;return{courseId:e,courseName:t,userElo:l,generatorName:n,generators:r,generatedCount:a,filters:o,hints:u,finalCount:c.length,reviewsSelected:b,newSelected:x,cards:v,discardedTail:y}}function formatProvenance(e){return e.map(e=>` ${e.action===`generated`?`🎲`:e.action===`boosted`?`⬆️`:e.action===`penalized`?`⬇️`:`➡️`} ${e.strategyName}: ${e.score.toFixed(3)} - ${e.reason}`).join(`
15
15
  `)}function printRunSummary(e){if(console.group(`\u{1F50D} Pipeline Run: ${e.courseId} (${e.courseName||`unnamed`})`),logger.info(`Run ID: ${e.runId}`),logger.info(`Time: ${e.timestamp.toISOString()}`),logger.info(`User ELO: ${e.userElo??`unknown`}`),logger.info(`Generator: ${e.generatorName} \u2192 ${e.generatedCount} candidates`),e.generators&&e.generators.length>0){console.group(`Generator breakdown:`);for(let t of e.generators)logger.info(` ${t.name}: ${t.cardCount} cards (${t.newCount} new, ${t.reviewCount} reviews, top: ${t.topScore.toFixed(2)})`);console.groupEnd()}if(e.filters.length>0){console.group(`Filter impact:`);for(let t of e.filters)logger.info(` ${t.name}: \u2191${t.boosted} \u2193${t.penalized} =${t.passed} \u2715${t.removed}`);console.groupEnd()}logger.info(`Result: ${e.finalCount} cards selected (${e.newSelected} new, ${e.reviewsSelected} reviews)`),console.groupEnd()}function escapeHtml(e){return e.replace(/&/g,`&amp;`).replace(/</g,`&lt;`).replace(/>/g,`&gt;`)}function escapeAttr(e){return escapeHtml(e).replace(/"/g,`&quot;`)}function copyTextToClipboard(e,t){let done=()=>{if(!t)return;let e=t.textContent??`Copy`;t.textContent=`Copied!`,t.classList.add(`copied`),setTimeout(()=>{t.textContent=e,t.classList.remove(`copied`)},1200)},fallback=()=>{let t=document.createElement(`textarea`);t.value=e,t.style.position=`fixed`,t.style.opacity=`0`,document.body.appendChild(t),t.select();try{document.execCommand(`copy`)}catch(e){logger.warn(`[Pipeline Debug] Copy failed: ${e}`)}document.body.removeChild(t),done()};navigator.clipboard?.writeText?navigator.clipboard.writeText(e).then(done).catch(fallback):fallback()}function renderUI(){if(!_uiContainer)return;let e=runHistory,t=_selectedRunIndex===null?null:e[_selectedRunIndex],n=`
16
16
  #sk-pipeline-debugger {
17
17
  position: fixed;
@@ -252,7 +252,7 @@ Example:
252
252
  Generator: ${e.name}
253
253
  Filters:${n}`)}function logTagHydration(e,t){let n=Array.from(t.values()).reduce((e,t)=>e+t.length,0),r=Array.from(t.values()).filter(e=>e.length>0).length;logger.debug(`[Pipeline] Tag hydration: ${e.length} cards, ${r} have tags (${n} total tags) - single batch query`)}function logExecutionSummary(e,t,n,r,a,o){let s=a.length>0?a.map(e=>e.toFixed(2)).join(`, `):`none`,c=``;o.length>0&&(c=`
254
254
  Filter impact: ${o.map(e=>{let t=[];return e.boosted>0&&t.push(`+${e.boosted}`),e.penalized>0&&t.push(`-${e.penalized}`),e.passed>0&&t.push(`=${e.passed}`),`${e.name}: ${t.join(`/`)}`}).join(`, `)}`),logger.info(`[Pipeline] Execution: ${e} produced ${t} \u2192 ${n} filters \u2192 ${r} results (top scores: ${s})`+c+`
255
- 💡 Inspect: window.skuilder.pipeline`)}function logResultCards(e){if(!(!VERBOSE_RESULTS||e.length===0)){logger.info(`[Pipeline] Results (${e.length} cards):`);for(let t=0;t<e.length;t++){let n=e[t],r=n.tags?.slice(0,3).join(`, `)||``,a=n.provenance.filter(e=>e.strategy===`hierarchyDefinition`||e.strategy===`priorityDefinition`||e.strategy===`interferenceFilter`||e.strategy===`letterGating`||e.strategy===`ephemeralHint`||e.strategy===`diversityRerank`).map(e=>{let t=e.action===`boosted`?`↑`:e.action===`penalized`?`↓`:`=`;return`${e.strategyName}${t}${e.score.toFixed(2)}`}).join(` | `);logger.info(`[Pipeline] ${String(t+1).padStart(2)}. ${n.score.toFixed(4)} ${n.cardId} [${r}]${a?` {${a}}`:``}`)}}}function logCardProvenance(e,t=3){let n=e.slice(0,t);logger.debug(`[Pipeline] Provenance for top ${n.length} cards:`);for(let e of n){logger.debug(`[Pipeline] ${e.cardId} (final score: ${e.score.toFixed(3)}):`);for(let t of e.provenance){let e=t.score.toFixed(3),n=t.action.padEnd(9);logger.debug(`[Pipeline] ${n} ${e} - ${t.strategyName}: ${t.reason}`)}}}var VERBOSE_RESULTS,Pipeline,init_Pipeline=__esm({"src/core/navigators/Pipeline.ts"(){init_navigators(),init_logger(),init_orchestration(),init_PipelineDebugger(),init_diversityRerank(),VERBOSE_RESULTS=!0,Pipeline=class extends ContentNavigator{constructor(e,t,n,r){super(),_defineProperty(this,`generator`,void 0),_defineProperty(this,`filters`,void 0),_defineProperty(this,`_cachedOrchestration`,null),_defineProperty(this,`_tagCache`,new Map),_defineProperty(this,`_ephemeralHints`,null),this.generator=e,this.filters=t,this.user=n,this.course=r,r.getCourseConfig().then(e=>{logger.debug(`[pipeline] Crated pipeline for ${e.name}`)}).catch(e=>{logger.error(`[pipeline] Failed to lookup courseCfg: ${e}`)}),logPipelineConfig(e,t),registerPipelineForDebug(this)}setEphemeralHints(e){this._ephemeralHints=e,logger.info(`[Pipeline] Ephemeral hints set: ${JSON.stringify(e)}`)}async getWeightedCards(e){let t=performance.now(),n=await this.buildContext(),r=performance.now(),a=500;logger.debug(`[Pipeline] Fetching 500 candidates from generator '${this.generator.name}'`);let o=await this.generator.getWeightedCards(500,n),s=o.cards,c=performance.now(),l=s.length;this._ephemeralHints=mergeHints2([this._ephemeralHints,o.hints])??null;let u;if(this.generator.generators){let e=new Map;for(let t of s){let n=t.provenance[0];if(n){let r=n.strategyName;e.has(r)||e.set(r,{cards:[]}),e.get(r).cards.push(t)}}u=Array.from(e.entries()).map(([e,t])=>{let n=t.cards.filter(e=>e.provenance[0]?.reason?.includes(`new card`)),r=t.cards.filter(e=>e.provenance[0]?.reason?.includes(`review`));return{name:e,cardCount:t.cards.length,newCount:n.length,reviewCount:r.length,topScore:Math.max(...t.cards.map(e=>e.score),0)}})}logger.debug(`[Pipeline] Generator returned ${l} candidates`),s=await this.hydrateTags(s);let d=performance.now(),p=[...s],m=this._ephemeralHints;if(m?.requireCards?.length){let e=new Set(p.map(e=>e.cardId)),t=m.requireCards.filter(t=>!t.includes(`*`)&&!e.has(t));if(t.length>0){let e=await this.course.getAppliedTagsBatch(t),n=this.course.getCourseID();for(let r of t)p.push({cardId:r,courseId:n,score:1,tags:e.get(r)??[],provenance:[]});logger.info(`[Pipeline] Pre-fetched ${t.length} required card(s) into pool: ${t.join(`, `)}`)}}let g=new Set(s.filter(e=>e.provenance.some(e=>e.strategy===`prescribed`)).map(e=>e.cardId)),_=[];for(let e of this.filters){let t=s.length,r=new Map(s.map(e=>[e.cardId,e.score]));s=await e.transform(s,n);let a=0,o=0,c=0,l=t-s.length;for(let e of s){let t=r.get(e.cardId)??0;e.score>t?a++:e.score<t?o++:c++}if(_.push({name:e.name,boosted:a,penalized:o,passed:c,removed:l}),g.size>0){let t=new Set(s.map(e=>e.cardId)),n=[...g].filter(e=>!t.has(e)),r=s.filter(e=>g.has(e.cardId)&&e.score===0).map(e=>e.cardId);(n.length>0||r.length>0)&&(logger.info(`[Pipeline] Filter '${e.name}' impact on prescribed cards: `+(n.length>0?`removed=[${n.join(`, `)}] `:``)+(r.length>0?`zeroed=[${r.join(`, `)}]`:``)),n.forEach(e=>g.delete(e)))}logger.debug(`[Pipeline] Filter '${e.name}': ${r.size} \u2192 ${s.length} cards (\u2191${a} \u2193${o} =${c})`)}s=s.filter(e=>e.score>0);let v=this._ephemeralHints;v&&(this._ephemeralHints=null,s=this.applyHints(s,v,p)),s=diversityRerank(s),s.sort((e,t)=>t.score-e.score);let y=performance.now(),b=s.slice(0,e);logger.info(`[Pipeline:timing] total=${(y-t).toFixed(0)}ms (context=${(r-t).toFixed(0)} generate=${(c-r).toFixed(0)} hydrate=${(d-c).toFixed(0)} filter=${(y-d).toFixed(0)})`);let x=b.slice(0,3).map(e=>e.score);logExecutionSummary(this.generator.name,l,this.filters.length,b.length,x,_),logResultCards(b),logCardProvenance(b,3);try{let e=await this.course?.getCourseConfig().then(e=>e.name).catch(()=>void 0);captureRun(buildRunReport(this.course?.getCourseID()||`unknown`,e,this.generator.name,u,l,_,s,b,n.userElo,v??void 0))}catch(e){logger.debug(`[Pipeline] Failed to capture debug run: ${e}`)}return{cards:b}}async hydrateTags(e){if(e.length===0)return e;let t=[];for(let n of e)this._tagCache.has(n.cardId)||t.push(n.cardId);if(t.length>0){let e=await this.course.getAppliedTagsBatch(t);for(let[t,n]of e)this._tagCache.set(t,n)}let n=new Map;for(let t of e)n.set(t.cardId,this._tagCache.get(t.cardId)??[]);return logTagHydration(e,n),e.map(e=>({...e,tags:this._tagCache.get(e.cardId)??[]}))}applyHints(e,t,n){let r=e.length;if(t.excludeCards?.length&&(e=e.filter(e=>!t.excludeCards.some(t=>globMatch(e.cardId,t)))),t.excludeTags?.length&&(e=e.filter(e=>!t.excludeTags.some(t=>cardMatchesTagPattern(e,t)))),t.boostTags)for(let[n,r]of Object.entries(t.boostTags))for(let a of e)cardMatchesTagPattern(a,n)&&(a.score*=r,a.provenance.push({strategy:`ephemeralHint`,strategyId:`ephemeral-hint`,strategyName:t._label?`Replan Hint (${t._label})`:`Replan Hint`,action:`boosted`,score:a.score,reason:`boostTag ${n} \xD7${r}`}));if(t.boostCards)for(let[n,r]of Object.entries(t.boostCards))for(let a of e)globMatch(a.cardId,n)&&(a.score*=r,a.provenance.push({strategy:`ephemeralHint`,strategyId:`ephemeral-hint`,strategyName:t._label?`Replan Hint (${t._label})`:`Replan Hint`,action:`boosted`,score:a.score,reason:`boostCard ${n} \xD7${r}`}));let a=new Set(e.map(e=>e.cardId)),o=new Map(e.map(e=>[e.cardId,e])),s=t._label?`Replan Hint (${t._label})`:`Replan Hint`,applyRequirement=(t,n)=>{let r=1/0,c=o.get(t.cardId);c?c.score<r&&(c.score=r,c.provenance.push({strategy:`ephemeralHint`,strategyId:`ephemeral-hint`,strategyName:s,action:`boosted`,score:r,reason:`${n} (upgrade to mandatory score)`})):(e.push({...t,score:r,provenance:[...t.provenance,{strategy:`ephemeralHint`,strategyId:`ephemeral-hint`,strategyName:s,action:`boosted`,score:r,reason:n}]}),a.add(t.cardId),o.set(t.cardId,e[e.length-1]))};if(t.requireCards?.length)for(let e of t.requireCards){for(let t of a)globMatch(t,e)&&applyRequirement(o.get(t),`requireCard ${e}`);for(let t of n)globMatch(t.cardId,e)&&applyRequirement(t,`requireCard ${e}`)}if(t.requireTags?.length)for(let e of t.requireTags){for(let t of a){let n=o.get(t);cardMatchesTagPattern(n,e)&&applyRequirement(n,`requireTag ${e}`)}for(let t of n)cardMatchesTagPattern(t,e)&&applyRequirement(t,`requireTag ${e}`)}return logger.info(`[Pipeline] Hints applied: ${r} \u2192 ${e.length} cards`),e}async buildContext(){let e=1e3;try{e=toCourseElo((await this.user.getCourseRegDoc(this.course.getCourseID())).elo).global.score}catch(e){logger.debug(`[Pipeline] Could not get user ELO, using default: ${e}`)}this._cachedOrchestration||(this._cachedOrchestration=await createOrchestrationContext(this.user,this.course));let t=this._cachedOrchestration;return{user:this.user,course:this.course,userElo:e,orchestration:t}}getCourseID(){return this.course.getCourseID()}async getOrchestrationContext(){return createOrchestrationContext(this.user,this.course)}getStrategyIds(){let e=[],extractId=e=>e.strategyId?e.strategyId:null,t=extractId(this.generator);t&&e.push(t),this.generator.generators&&Array.isArray(this.generator.generators)&&this.generator.generators.forEach(t=>{let n=extractId(t);n&&e.push(n)});for(let t of this.filters){let n=extractId(t);n&&e.push(n)}return[...new Set(e)]}async getTagEloStatus(e){let t=toCourseElo((await this.user.getCourseRegDoc(this.course.getCourseID())).elo),n={};if(e){let r=Array.isArray(e)?e:[e];for(let e of r){let r=globToRegex(e);for(let[e,a]of Object.entries(t.tags))r.test(e)&&(n[e]={score:a.score,count:a.count})}}else for(let[e,r]of Object.entries(t.tags))n[e]={score:r.score,count:r.count};return n}async diagnoseCardSpace(e){let t=e?.threshold??.1,n=performance.now(),r=await this.course.getAllCardIds(),a=r.map(e=>({cardId:e,courseId:this.course.getCourseID(),score:1,provenance:[]}));a=await this.hydrateTags(a);let o=await this.buildContext(),s=[];for(let e of this.filters){a=await e.transform(a,o);let n=a.filter(e=>e.score>=t).length;s.push({name:e.name,wellIndicated:n})}let c=a.filter(e=>e.score>=t),l=new Set(c.map(e=>e.cardId)),u;try{let e=this.course.getCourseID(),t=await this.user.getSeenCards(e);u=new Set(t)}catch{u=new Set}let d=c.filter(e=>!u.has(e.cardId)),p=new Map;for(let e of a){let n=e.cardId.split(`-`)[1]||`unknown`;p.has(n)||p.set(n,{total:0,wellIndicated:0,new:0});let r=p.get(n);r.total++,e.score>=t&&(r.wellIndicated++,u.has(e.cardId)||r.new++)}let m=performance.now()-n,g={totalCards:r.length,threshold:t,wellIndicated:l.size,encountered:u.size,wellIndicatedNew:d.length,byType:Object.fromEntries(p),filterBreakdown:s,elapsedMs:Math.round(m)};logger.info(`[Pipeline:diagnose] Card space scan (${g.elapsedMs}ms):`),logger.info(`[Pipeline:diagnose] Total cards: ${g.totalCards}`),logger.info(`[Pipeline:diagnose] Well-indicated (score >= ${t}): ${g.wellIndicated}`),logger.info(`[Pipeline:diagnose] Encountered: ${g.encountered}`),logger.info(`[Pipeline:diagnose] Well-indicated & new: ${g.wellIndicatedNew}`),logger.info(`[Pipeline:diagnose] By type:`);for(let[e,t]of p)logger.info(`[Pipeline:diagnose] ${e}: ${t.wellIndicated}/${t.total} well-indicated, ${t.new} new`);logger.info(`[Pipeline:diagnose] After each filter:`);for(let e of s)logger.info(`[Pipeline:diagnose] ${e.name}: ${e.wellIndicated} well-indicated`);return g}}}}),defaults_exports={};__export(defaults_exports,{createDefaultEloStrategy:()=>createDefaultEloStrategy,createDefaultPipeline:()=>createDefaultPipeline,createDefaultSrsStrategy:()=>createDefaultSrsStrategy});function createDefaultEloStrategy(e){return{_id:`NAVIGATION_STRATEGY-ELO-default`,docType:`NAVIGATION_STRATEGY`,name:`ELO (default)`,description:`Default ELO-based navigation strategy for new cards`,implementingClass:`elo`,course:e,serializedData:``}}function createDefaultSrsStrategy(e){return{_id:`NAVIGATION_STRATEGY-SRS-default`,docType:`NAVIGATION_STRATEGY`,name:`SRS (default)`,description:`Default SRS-based navigation strategy for reviews`,implementingClass:`srs`,course:e,serializedData:``}}function createDefaultPipeline(e,t){let n=t.getCourseID(),r=new ELONavigator(e,t,createDefaultEloStrategy(n)),a=new SRSNavigator(e,t,createDefaultSrsStrategy(n)),o=new CompositeGenerator([r,a]),s=createEloDistanceFilter();return new Pipeline(o,[s],e,t)}var init_defaults=__esm({"src/core/navigators/defaults.ts"(){init_navigators(),init_Pipeline(),init_CompositeGenerator(),init_elo(),init_srs(),init_eloDistance(),init_types_legacy()}}),PipelineAssembler_exports={};__export(PipelineAssembler_exports,{PipelineAssembler:()=>PipelineAssembler});var PipelineAssembler,init_PipelineAssembler=__esm({"src/core/navigators/PipelineAssembler.ts"(){init_navigators(),init_WeightedFilter(),init_Pipeline(),init_logger(),init_CompositeGenerator(),init_defaults(),PipelineAssembler=class{async assemble(e){let{strategies:t,user:n,course:r}=e,a=[];if(t.length===0)return{pipeline:null,generatorStrategies:[],filterStrategies:[],warnings:a};let o=[],s=[];for(let e of t)isGenerator(e.implementingClass)?o.push(e):isFilter(e.implementingClass)?s.push(e):a.push(`Unknown strategy type '${e.implementingClass}', skipping: ${e.name}`);let c=r.getCourseID(),l=o.some(e=>e.implementingClass===`elo`),u=o.some(e=>e.implementingClass===`srs`);if(l||(logger.debug(`[PipelineAssembler] No ELO generator configured, adding default`),o.push(createDefaultEloStrategy(c))),u||(logger.debug(`[PipelineAssembler] No SRS generator configured, adding default`),o.push(createDefaultSrsStrategy(c))),o.length===0)return a.push(`No generator strategy found`),{pipeline:null,generatorStrategies:[],filterStrategies:[],warnings:a};let d;o.length===1?(d=await ContentNavigator.create(n,r,o[0]),logger.debug(`[PipelineAssembler] Using single generator: ${o[0].name}`)):(logger.debug(`[PipelineAssembler] Using CompositeGenerator for ${o.length} generators: ${o.map(e=>e.name).join(`, `)}`),d=await CompositeGenerator.fromStrategies(n,r,o));let p=[],m=[...s].sort((e,t)=>e.name.localeCompare(t.name));for(let e of m)try{let t=await ContentNavigator.create(n,r,e);if(`transform`in t&&typeof t.transform==`function`){let n=t;e.learnable&&(n=new WeightedFilter(n,e.learnable,e.staticWeight,e._id)),p.push(n),logger.debug(`[PipelineAssembler] Added filter: ${e.name}`)}else a.push(`Filter '${e.name}' does not implement CardFilter.transform(), skipping`)}catch(t){a.push(`Failed to instantiate filter '${e.name}': ${t}`)}let g=new Pipeline(d,p,n,r);return logger.debug(`[PipelineAssembler] Assembled pipeline with ${o.length} generator(s) and ${p.length} filter(s)`),{pipeline:g,generatorStrategies:o,filterStrategies:m,warnings:a}}}}}),globImport,init_3=__esm({'import("./**/*") in src/core/navigators/index.ts'(){globImport=__glob({"./Pipeline.ts":()=>Promise.resolve().then(()=>(init_Pipeline(),Pipeline_exports)),"./PipelineAssembler.ts":()=>Promise.resolve().then(()=>(init_PipelineAssembler(),PipelineAssembler_exports)),"./PipelineDebugger.ts":()=>Promise.resolve().then(()=>(init_PipelineDebugger(),PipelineDebugger_exports)),"./defaults.ts":()=>Promise.resolve().then(()=>(init_defaults(),defaults_exports)),"./diversityRerank.ts":()=>Promise.resolve().then(()=>(init_diversityRerank(),diversityRerank_exports)),"./filters/WeightedFilter.ts":()=>Promise.resolve().then(()=>(init_WeightedFilter(),WeightedFilter_exports)),"./filters/eloDistance.ts":()=>Promise.resolve().then(()=>(init_eloDistance(),eloDistance_exports)),"./filters/hierarchyDefinition.ts":()=>Promise.resolve().then(()=>(init_hierarchyDefinition(),hierarchyDefinition_exports)),"./filters/index.ts":()=>Promise.resolve().then(()=>(init_filters(),filters_exports)),"./filters/inferredPreferenceStub.ts":()=>Promise.resolve().then(()=>(init_inferredPreferenceStub(),inferredPreferenceStub_exports)),"./filters/interferenceMitigator.ts":()=>Promise.resolve().then(()=>(init_interferenceMitigator(),interferenceMitigator_exports)),"./filters/relativePriority.ts":()=>Promise.resolve().then(()=>(init_relativePriority(),relativePriority_exports)),"./filters/types.ts":()=>Promise.resolve().then(()=>(init_types2(),types_exports2)),"./filters/userGoalStub.ts":()=>Promise.resolve().then(()=>(init_userGoalStub(),userGoalStub_exports)),"./filters/userTagPreference.ts":()=>Promise.resolve().then(()=>(init_userTagPreference(),userTagPreference_exports)),"./generators/CompositeGenerator.ts":()=>Promise.resolve().then(()=>(init_CompositeGenerator(),CompositeGenerator_exports)),"./generators/elo.ts":()=>Promise.resolve().then(()=>(init_elo(),elo_exports)),"./generators/index.ts":()=>Promise.resolve().then(()=>(init_generators(),generators_exports)),"./generators/prescribed.ts":()=>Promise.resolve().then(()=>(init_prescribed(),prescribed_exports)),"./generators/srs.ts":()=>Promise.resolve().then(()=>(init_srs(),srs_exports)),"./generators/types.ts":()=>Promise.resolve().then(()=>(init_types(),types_exports)),"./index.ts":()=>Promise.resolve().then(()=>(init_navigators(),navigators_exports))})}}),navigators_exports={};__export(navigators_exports,{ContentNavigator:()=>ContentNavigator,DIVERSITY_FLOOR:()=>DIVERSITY_FLOOR,DIVERSITY_STRENGTH:()=>DIVERSITY_STRENGTH,NavigatorRole:()=>NavigatorRole,NavigatorRoles:()=>NavigatorRoles,Navigators:()=>Navigators,diversityRerank:()=>diversityRerank,getCardOrigin:()=>getCardOrigin,getRegisteredNavigator:()=>getRegisteredNavigator,getRegisteredNavigatorNames:()=>getRegisteredNavigatorNames,getRegisteredNavigatorRole:()=>getRegisteredNavigatorRole,hasRegisteredNavigator:()=>hasRegisteredNavigator,initializeNavigatorRegistry:()=>initializeNavigatorRegistry,isFilter:()=>isFilter,isGenerator:()=>isGenerator,mountPipelineDebugger:()=>mountPipelineDebugger,pipelineDebugAPI:()=>pipelineDebugAPI,registerNavigator:()=>registerNavigator});function registerNavigator(e,t,n){navigatorRegistry.set(e,{constructor:t,role:n}),logger.debug(`[NavigatorRegistry] Registered: ${e}${n?` (${n})`:``}`)}function getRegisteredNavigator(e){return navigatorRegistry.get(e)?.constructor}function hasRegisteredNavigator(e){return navigatorRegistry.has(e)}function getRegisteredNavigatorRole(e){return navigatorRegistry.get(e)?.role}function getRegisteredNavigatorNames(){return Array.from(navigatorRegistry.keys())}async function initializeNavigatorRegistry(){logger.debug(`[NavigatorRegistry] Initializing built-in navigators...`);let[e,t]=await Promise.all([Promise.resolve().then(()=>(init_elo(),elo_exports)),Promise.resolve().then(()=>(init_srs(),srs_exports))]),n=await Promise.resolve().then(()=>(init_prescribed(),prescribed_exports));registerNavigator(`elo`,e.default),registerNavigator(`srs`,t.default),registerNavigator(`prescribed`,n.default);let[r,a,o,s]=await Promise.all([Promise.resolve().then(()=>(init_hierarchyDefinition(),hierarchyDefinition_exports)),Promise.resolve().then(()=>(init_interferenceMitigator(),interferenceMitigator_exports)),Promise.resolve().then(()=>(init_relativePriority(),relativePriority_exports)),Promise.resolve().then(()=>(init_userTagPreference(),userTagPreference_exports))]);registerNavigator(`hierarchyDefinition`,r.default),registerNavigator(`interferenceMitigator`,a.default),registerNavigator(`relativePriority`,o.default),registerNavigator(`userTagPreference`,s.default),logger.debug(`[NavigatorRegistry] Initialized ${navigatorRegistry.size} navigators: ${getRegisteredNavigatorNames().join(`, `)}`)}function getCardOrigin(e){if(e.provenance.length===0)throw Error(`Card has no provenance - cannot determine origin`);let t=e.provenance[0].reason.toLowerCase();return t.includes(`failed`)?`failed`:t.includes(`review`)?`review`:`new`}function isGenerator(e){return NavigatorRoles[e]===`generator`?!0:getRegisteredNavigatorRole(e)===`generator`}function isFilter(e){return NavigatorRoles[e]===`filter`?!0:getRegisteredNavigatorRole(e)===`filter`}var navigatorRegistry,Navigators,NavigatorRole,NavigatorRoles,ContentNavigator,init_navigators=__esm({"src/core/navigators/index.ts"(){init_diversityRerank(),init_PipelineDebugger(),init_logger(),init_(),init_2(),init_3(),navigatorRegistry=new Map,Navigators=(e=>(e.ELO=`elo`,e.SRS=`srs`,e.PRESCRIBED=`prescribed`,e.HIERARCHY=`hierarchyDefinition`,e.INTERFERENCE=`interferenceMitigator`,e.RELATIVE_PRIORITY=`relativePriority`,e.USER_TAG_PREFERENCE=`userTagPreference`,e))(Navigators||{}),NavigatorRole=(e=>(e.GENERATOR=`generator`,e.FILTER=`filter`,e))(NavigatorRole||{}),NavigatorRoles={elo:`generator`,srs:`generator`,prescribed:`generator`,hierarchyDefinition:`filter`,interferenceMitigator:`filter`,relativePriority:`filter`,userTagPreference:`filter`},ContentNavigator=class{constructor(e,t,n){_defineProperty(this,`user`,void 0),_defineProperty(this,`course`,void 0),_defineProperty(this,`strategyName`,void 0),_defineProperty(this,`strategyId`,void 0),_defineProperty(this,`learnable`,void 0),_defineProperty(this,`staticWeight`,void 0),this.user=e,this.course=t,n&&(this.strategyName=n.name,this.strategyId=n._id,this.learnable=n.learnable,this.staticWeight=n.staticWeight)}get strategyKey(){return this.constructor.name}async getStrategyState(){if(!this.user||!this.course)throw Error(`Cannot get strategy state: navigator not properly initialized. Ensure user and course are provided to constructor.`);return this.user.getStrategyState(this.course.getCourseID(),this.strategyKey)}async putStrategyState(e){if(!this.user||!this.course)throw Error(`Cannot put strategy state: navigator not properly initialized. Ensure user and course are provided to constructor.`);return this.user.putStrategyState(this.course.getCourseID(),this.strategyKey,e)}static async create(e,t,n){let r=n.implementingClass,a=getRegisteredNavigator(r);if(a)return logger.debug(`[ContentNavigator.create] Using registered navigator: ${r}`),new a(e,t,n);logger.debug(`[ContentNavigator.create] Navigator not in registry, attempting dynamic import: ${r}`);let o;for(let e of[`.ts`,`.js`,``]){try{if(o=(await globImport_generators(`./generators/${r}${e}`)).default,o)break}catch(t){logger.debug(`Failed to load generator ${r}${e}:`,t)}try{if(o=(await globImport_filters(`./filters/${r}${e}`)).default,o)break}catch(t){logger.debug(`Failed to load filter ${r}${e}:`,t)}try{if(o=(await globImport(`./${r}${e}`)).default,o)break}catch(t){logger.debug(`Failed to load legacy ${r}${e}:`,t)}if(o)break}if(!o)throw Error(`Could not load navigator implementation for: ${r}`);return new o(e,t,n)}async getWeightedCards(e){throw Error(`${this.constructor.name} must implement getWeightedCards(). `)}setEphemeralHints(e){}}}});function randIntWeightedTowardZero(e){return Math.floor(Math.random()*Math.random()*Math.random()*e)}async function getCourseTagStubs(e){logger.debug(`Getting tag stubs for course: ${e}`);let t=await filterAllDocsByPrefix2(getCourseDB2(e),`TAG`.valueOf()+`-`);return t.rows.forEach(e=>{logger.debug(` Tag stub for doc: ${e.id}`)}),t}async function createTag(e,t,n){logger.debug(`Creating tag: ${t}...`);let r=getTagID(t);return await getCourseDB2(e).put({course:e,docType:`TAG`,name:t,snippet:``,taggedCards:[],wiki:``,author:n,_id:r})}async function updateTag(e){let t=await getTag(e.course,e.name);return await getCourseDB2(e.course).put({...e,_rev:t._rev})}async function getTag(e,t){let n=getTagID(t);return getCourseDB2(e).get(n)}async function removeTagFromCard(e,t,n){n=getTagID(n);let r=getCourseDB2(e),a=await r.get(n);return a.taggedCards=a.taggedCards.filter(e=>t!==e),r.put(a)}async function getAppliedTags(e,t){return await getCourseDB2(e).query(`getTags`,{startkey:t,endkey:t})}async function updateCredentialledCourseConfig(e,t){logger.debug(`Updating course config:
255
+ 💡 Inspect: window.skuilder.pipeline`)}function logResultCards(e){if(!(!VERBOSE_RESULTS||e.length===0)){logger.info(`[Pipeline] Results (${e.length} cards):`);for(let t=0;t<e.length;t++){let n=e[t],r=n.tags?.slice(0,3).join(`, `)||``,a=n.provenance.filter(e=>e.strategy===`hierarchyDefinition`||e.strategy===`priorityDefinition`||e.strategy===`interferenceFilter`||e.strategy===`letterGating`||e.strategy===`ephemeralHint`||e.strategy===`diversityRerank`).map(e=>{let t=e.action===`boosted`?`↑`:e.action===`penalized`?`↓`:`=`;return`${e.strategyName}${t}${e.score.toFixed(2)}`}).join(` | `);logger.info(`[Pipeline] ${String(t+1).padStart(2)}. ${n.score.toFixed(4)} ${n.cardId} [${r}]${a?` {${a}}`:``}`)}}}function logCardProvenance(e,t=3){let n=e.slice(0,t);logger.debug(`[Pipeline] Provenance for top ${n.length} cards:`);for(let e of n){logger.debug(`[Pipeline] ${e.cardId} (final score: ${e.score.toFixed(3)}):`);for(let t of e.provenance){let e=t.score.toFixed(3),n=t.action.padEnd(9);logger.debug(`[Pipeline] ${n} ${e} - ${t.strategyName}: ${t.reason}`)}}}var VERBOSE_RESULTS,Pipeline,init_Pipeline=__esm({"src/core/navigators/Pipeline.ts"(){init_navigators(),init_logger(),init_orchestration(),init_PipelineDebugger(),init_diversityRerank(),VERBOSE_RESULTS=!0,Pipeline=class extends ContentNavigator{constructor(e,t,n,r){super(),_defineProperty(this,`generator`,void 0),_defineProperty(this,`filters`,void 0),_defineProperty(this,`_cachedOrchestration`,null),_defineProperty(this,`_tagCache`,new Map),_defineProperty(this,`_ephemeralHints`,null),this.generator=e,this.filters=t,this.user=n,this.course=r,r.getCourseConfig().then(e=>{logger.debug(`[pipeline] Crated pipeline for ${e.name}`)}).catch(e=>{logger.error(`[pipeline] Failed to lookup courseCfg: ${e}`)}),logPipelineConfig(e,t),registerPipelineForDebug(this)}setEphemeralHints(e){this._ephemeralHints=e,logger.info(`[Pipeline] Ephemeral hints set: ${JSON.stringify(e)}`)}async getWeightedCards(e){let t=performance.now(),n=await this.buildContext(),r=performance.now(),a=500;logger.debug(`[Pipeline] Fetching 500 candidates from generator '${this.generator.name}'`);let o=await this.generator.getWeightedCards(500,n),s=o.cards,c=performance.now(),l=s.length;this._ephemeralHints=mergeHints2([this._ephemeralHints,o.hints])??null;let u;if(this.generator.generators){let e=new Map;for(let t of s){let n=t.provenance[0];if(n){let r=n.strategyName;e.has(r)||e.set(r,{cards:[]}),e.get(r).cards.push(t)}}u=Array.from(e.entries()).map(([e,t])=>{let n=t.cards.filter(e=>e.provenance[0]?.reason?.includes(`new card`)),r=t.cards.filter(e=>e.provenance[0]?.reason?.includes(`review`));return{name:e,cardCount:t.cards.length,newCount:n.length,reviewCount:r.length,topScore:Math.max(...t.cards.map(e=>e.score),0)}})}logger.debug(`[Pipeline] Generator returned ${l} candidates`),s=await this.hydrateTags(s);let d=performance.now(),p=[...s],m=this._ephemeralHints;if(m?.requireCards?.length){let e=new Set(p.map(e=>e.cardId)),t=m.requireCards.filter(t=>!t.includes(`*`)&&!e.has(t));if(t.length>0){let e=await this.course.getAppliedTagsBatch(t),n=this.course.getCourseID();for(let r of t)p.push({cardId:r,courseId:n,score:1,tags:e.get(r)??[],provenance:[]});logger.info(`[Pipeline] Pre-fetched ${t.length} required card(s) into pool: ${t.join(`, `)}`)}}let g=new Set(s.filter(e=>e.provenance.some(e=>e.strategy===`prescribed`)).map(e=>e.cardId)),_=[];for(let e of this.filters){let t=s.length,r=new Map(s.map(e=>[e.cardId,e.score]));s=await e.transform(s,n);let a=0,o=0,c=0,l=t-s.length;for(let e of s){let t=r.get(e.cardId)??0;e.score>t?a++:e.score<t?o++:c++}if(_.push({name:e.name,boosted:a,penalized:o,passed:c,removed:l}),g.size>0){let t=new Set(s.map(e=>e.cardId)),n=[...g].filter(e=>!t.has(e)),r=s.filter(e=>g.has(e.cardId)&&e.score===0).map(e=>e.cardId);(n.length>0||r.length>0)&&(logger.info(`[Pipeline] Filter '${e.name}' impact on prescribed cards: `+(n.length>0?`removed=[${n.join(`, `)}] `:``)+(r.length>0?`zeroed=[${r.join(`, `)}]`:``)),n.forEach(e=>g.delete(e)))}logger.debug(`[Pipeline] Filter '${e.name}': ${r.size} \u2192 ${s.length} cards (\u2191${a} \u2193${o} =${c})`)}s=s.filter(e=>e.score>0);let v=this._ephemeralHints;v&&(this._ephemeralHints=null,s=this.applyHints(s,v,p)),s=diversityRerank(s),s.sort((e,t)=>t.score-e.score);let y=performance.now(),b=s.slice(0,e);logger.info(`[Pipeline:timing] total=${(y-t).toFixed(0)}ms (context=${(r-t).toFixed(0)} generate=${(c-r).toFixed(0)} hydrate=${(d-c).toFixed(0)} filter=${(y-d).toFixed(0)})`);let x=b.slice(0,3).map(e=>e.score);logExecutionSummary(this.generator.name,l,this.filters.length,b.length,x,_),logResultCards(b),logCardProvenance(b,3);try{let e=await this.course?.getCourseConfig().then(e=>e.name).catch(()=>void 0);captureRun(buildRunReport(this.course?.getCourseID()||`unknown`,e,this.generator.name,u,l,_,s,b,n.userElo,v??void 0))}catch(e){logger.debug(`[Pipeline] Failed to capture debug run: ${e}`)}return{cards:b}}async hydrateTags(e){if(e.length===0)return e;let t=[];for(let n of e)this._tagCache.has(n.cardId)||t.push(n.cardId);if(t.length>0){let e=await this.course.getAppliedTagsBatch(t);for(let[t,n]of e)this._tagCache.set(t,n)}let n=new Map;for(let t of e)n.set(t.cardId,this._tagCache.get(t.cardId)??[]);return logTagHydration(e,n),e.map(e=>({...e,tags:this._tagCache.get(e.cardId)??[]}))}applyHints(e,t,n){let r=e.length;if(t.excludeCards?.length&&(e=e.filter(e=>!t.excludeCards.some(t=>globMatch(e.cardId,t)))),t.excludeTags?.length&&(e=e.filter(e=>!t.excludeTags.some(t=>cardMatchesTagPattern(e,t)))),t.boostTags)for(let[n,r]of Object.entries(t.boostTags))for(let a of e)cardMatchesTagPattern(a,n)&&(a.score*=r,a.provenance.push({strategy:`ephemeralHint`,strategyId:`ephemeral-hint`,strategyName:t._label?`Replan Hint (${t._label})`:`Replan Hint`,action:`boosted`,score:a.score,reason:`boostTag ${n} \xD7${r}`}));if(t.boostCards)for(let[n,r]of Object.entries(t.boostCards))for(let a of e)globMatch(a.cardId,n)&&(a.score*=r,a.provenance.push({strategy:`ephemeralHint`,strategyId:`ephemeral-hint`,strategyName:t._label?`Replan Hint (${t._label})`:`Replan Hint`,action:`boosted`,score:a.score,reason:`boostCard ${n} \xD7${r}`}));let a=new Set(e.map(e=>e.cardId)),o=new Map(e.map(e=>[e.cardId,e])),s=t._label?`Replan Hint (${t._label})`:`Replan Hint`,applyRequirement=(t,n)=>{let r=1/0,c=o.get(t.cardId);c?c.score<r&&(c.score=r,c.provenance.push({strategy:`ephemeralHint`,strategyId:`ephemeral-hint`,strategyName:s,action:`boosted`,score:r,reason:`${n} (upgrade to mandatory score)`})):(e.push({...t,score:r,provenance:[...t.provenance,{strategy:`ephemeralHint`,strategyId:`ephemeral-hint`,strategyName:s,action:`boosted`,score:r,reason:n}]}),a.add(t.cardId),o.set(t.cardId,e[e.length-1]))};if(t.requireCards?.length)for(let e of t.requireCards){for(let t of a)globMatch(t,e)&&applyRequirement(o.get(t),`requireCard ${e}`);for(let t of n)globMatch(t.cardId,e)&&applyRequirement(t,`requireCard ${e}`)}if(t.requireTags?.length)for(let e of t.requireTags){for(let t of a){let n=o.get(t);cardMatchesTagPattern(n,e)&&applyRequirement(n,`requireTag ${e}`)}for(let t of n)cardMatchesTagPattern(t,e)&&applyRequirement(t,`requireTag ${e}`)}return logger.info(`[Pipeline] Hints applied: ${r} \u2192 ${e.length} cards`),e}async buildContext(){let e=1e3;try{e=toCourseElo((await this.user.getCourseRegDoc(this.course.getCourseID())).elo).global.score}catch(e){logger.debug(`[Pipeline] Could not get user ELO, using default: ${e}`)}this._cachedOrchestration||(this._cachedOrchestration=await createOrchestrationContext(this.user,this.course));let t=this._cachedOrchestration;return{user:this.user,course:this.course,userElo:e,orchestration:t}}getCourseID(){return this.course.getCourseID()}async getOrchestrationContext(){return createOrchestrationContext(this.user,this.course)}getStrategyIds(){let e=[],extractId=e=>e.strategyId?e.strategyId:null,t=extractId(this.generator);t&&e.push(t),this.generator.generators&&Array.isArray(this.generator.generators)&&this.generator.generators.forEach(t=>{let n=extractId(t);n&&e.push(n)});for(let t of this.filters){let n=extractId(t);n&&e.push(n)}return[...new Set(e)]}async getTagEloStatus(e){let t=toCourseElo((await this.user.getCourseRegDoc(this.course.getCourseID())).elo),n={};if(e){let r=Array.isArray(e)?e:[e];for(let e of r){let r=globToRegex(e);for(let[e,a]of Object.entries(t.tags))r.test(e)&&(n[e]={score:a.score,count:a.count})}}else for(let[e,r]of Object.entries(t.tags))n[e]={score:r.score,count:r.count};return n}async forecast(e){let t=e?.threshold??.1,n=e?.unseenOnly??!0,r=this.course.getCourseID(),a=(await this.course.getAllCardIds()).map(e=>({cardId:e,courseId:r,score:1,provenance:[]}));a=await this.hydrateTags(a);let o=a.slice(),s=await this.buildContext();for(let e of this.filters)a=await e.transform(a,s);if(e?.hints&&(a=this.applyHints(a,e.hints,o)),a=a.filter(e=>e.score>=t),n){let e;try{e=new Set(await this.user.getSeenCards(r))}catch{e=new Set}a=a.filter(t=>!e.has(t.cardId))}return a.sort((e,t)=>t.score-e.score),e?.limit?a.slice(0,e.limit):a}async diagnoseCardSpace(e){let t=e?.threshold??.1,n=performance.now(),r=await this.course.getAllCardIds(),a=r.map(e=>({cardId:e,courseId:this.course.getCourseID(),score:1,provenance:[]}));a=await this.hydrateTags(a);let o=await this.buildContext(),s=[];for(let e of this.filters){a=await e.transform(a,o);let n=a.filter(e=>e.score>=t).length;s.push({name:e.name,wellIndicated:n})}let c=a.filter(e=>e.score>=t),l=new Set(c.map(e=>e.cardId)),u;try{let e=this.course.getCourseID(),t=await this.user.getSeenCards(e);u=new Set(t)}catch{u=new Set}let d=c.filter(e=>!u.has(e.cardId)),p=new Map;for(let e of a){let n=e.cardId.split(`-`)[1]||`unknown`;p.has(n)||p.set(n,{total:0,wellIndicated:0,new:0});let r=p.get(n);r.total++,e.score>=t&&(r.wellIndicated++,u.has(e.cardId)||r.new++)}let m=performance.now()-n,g={totalCards:r.length,threshold:t,wellIndicated:l.size,encountered:u.size,wellIndicatedNew:d.length,byType:Object.fromEntries(p),filterBreakdown:s,elapsedMs:Math.round(m)};logger.info(`[Pipeline:diagnose] Card space scan (${g.elapsedMs}ms):`),logger.info(`[Pipeline:diagnose] Total cards: ${g.totalCards}`),logger.info(`[Pipeline:diagnose] Well-indicated (score >= ${t}): ${g.wellIndicated}`),logger.info(`[Pipeline:diagnose] Encountered: ${g.encountered}`),logger.info(`[Pipeline:diagnose] Well-indicated & new: ${g.wellIndicatedNew}`),logger.info(`[Pipeline:diagnose] By type:`);for(let[e,t]of p)logger.info(`[Pipeline:diagnose] ${e}: ${t.wellIndicated}/${t.total} well-indicated, ${t.new} new`);logger.info(`[Pipeline:diagnose] After each filter:`);for(let e of s)logger.info(`[Pipeline:diagnose] ${e.name}: ${e.wellIndicated} well-indicated`);return g}}}}),defaults_exports={};__export(defaults_exports,{createDefaultEloStrategy:()=>createDefaultEloStrategy,createDefaultPipeline:()=>createDefaultPipeline,createDefaultSrsStrategy:()=>createDefaultSrsStrategy});function createDefaultEloStrategy(e){return{_id:`NAVIGATION_STRATEGY-ELO-default`,docType:`NAVIGATION_STRATEGY`,name:`ELO (default)`,description:`Default ELO-based navigation strategy for new cards`,implementingClass:`elo`,course:e,serializedData:``}}function createDefaultSrsStrategy(e){return{_id:`NAVIGATION_STRATEGY-SRS-default`,docType:`NAVIGATION_STRATEGY`,name:`SRS (default)`,description:`Default SRS-based navigation strategy for reviews`,implementingClass:`srs`,course:e,serializedData:``}}function createDefaultPipeline(e,t){let n=t.getCourseID(),r=new ELONavigator(e,t,createDefaultEloStrategy(n)),a=new SRSNavigator(e,t,createDefaultSrsStrategy(n)),o=new CompositeGenerator([r,a]),s=createEloDistanceFilter();return new Pipeline(o,[s],e,t)}var init_defaults=__esm({"src/core/navigators/defaults.ts"(){init_navigators(),init_Pipeline(),init_CompositeGenerator(),init_elo(),init_srs(),init_eloDistance(),init_types_legacy()}}),PipelineAssembler_exports={};__export(PipelineAssembler_exports,{PipelineAssembler:()=>PipelineAssembler});var PipelineAssembler,init_PipelineAssembler=__esm({"src/core/navigators/PipelineAssembler.ts"(){init_navigators(),init_WeightedFilter(),init_Pipeline(),init_logger(),init_CompositeGenerator(),init_defaults(),PipelineAssembler=class{async assemble(e){let{strategies:t,user:n,course:r}=e,a=[];if(t.length===0)return{pipeline:null,generatorStrategies:[],filterStrategies:[],warnings:a};let o=[],s=[];for(let e of t)isGenerator(e.implementingClass)?o.push(e):isFilter(e.implementingClass)?s.push(e):a.push(`Unknown strategy type '${e.implementingClass}', skipping: ${e.name}`);let c=r.getCourseID(),l=o.some(e=>e.implementingClass===`elo`),u=o.some(e=>e.implementingClass===`srs`);if(l||(logger.debug(`[PipelineAssembler] No ELO generator configured, adding default`),o.push(createDefaultEloStrategy(c))),u||(logger.debug(`[PipelineAssembler] No SRS generator configured, adding default`),o.push(createDefaultSrsStrategy(c))),o.length===0)return a.push(`No generator strategy found`),{pipeline:null,generatorStrategies:[],filterStrategies:[],warnings:a};let d;o.length===1?(d=await ContentNavigator.create(n,r,o[0]),logger.debug(`[PipelineAssembler] Using single generator: ${o[0].name}`)):(logger.debug(`[PipelineAssembler] Using CompositeGenerator for ${o.length} generators: ${o.map(e=>e.name).join(`, `)}`),d=await CompositeGenerator.fromStrategies(n,r,o));let p=[],m=[...s].sort((e,t)=>e.name.localeCompare(t.name));for(let e of m)try{let t=await ContentNavigator.create(n,r,e);if(`transform`in t&&typeof t.transform==`function`){let n=t;e.learnable&&(n=new WeightedFilter(n,e.learnable,e.staticWeight,e._id)),p.push(n),logger.debug(`[PipelineAssembler] Added filter: ${e.name}`)}else a.push(`Filter '${e.name}' does not implement CardFilter.transform(), skipping`)}catch(t){a.push(`Failed to instantiate filter '${e.name}': ${t}`)}let g=new Pipeline(d,p,n,r);return logger.debug(`[PipelineAssembler] Assembled pipeline with ${o.length} generator(s) and ${p.length} filter(s)`),{pipeline:g,generatorStrategies:o,filterStrategies:m,warnings:a}}}}}),globImport,init_3=__esm({'import("./**/*") in src/core/navigators/index.ts'(){globImport=__glob({"./Pipeline.ts":()=>Promise.resolve().then(()=>(init_Pipeline(),Pipeline_exports)),"./PipelineAssembler.ts":()=>Promise.resolve().then(()=>(init_PipelineAssembler(),PipelineAssembler_exports)),"./PipelineDebugger.ts":()=>Promise.resolve().then(()=>(init_PipelineDebugger(),PipelineDebugger_exports)),"./defaults.ts":()=>Promise.resolve().then(()=>(init_defaults(),defaults_exports)),"./diversityRerank.ts":()=>Promise.resolve().then(()=>(init_diversityRerank(),diversityRerank_exports)),"./filters/WeightedFilter.ts":()=>Promise.resolve().then(()=>(init_WeightedFilter(),WeightedFilter_exports)),"./filters/eloDistance.ts":()=>Promise.resolve().then(()=>(init_eloDistance(),eloDistance_exports)),"./filters/hierarchyDefinition.ts":()=>Promise.resolve().then(()=>(init_hierarchyDefinition(),hierarchyDefinition_exports)),"./filters/index.ts":()=>Promise.resolve().then(()=>(init_filters(),filters_exports)),"./filters/inferredPreferenceStub.ts":()=>Promise.resolve().then(()=>(init_inferredPreferenceStub(),inferredPreferenceStub_exports)),"./filters/interferenceMitigator.ts":()=>Promise.resolve().then(()=>(init_interferenceMitigator(),interferenceMitigator_exports)),"./filters/relativePriority.ts":()=>Promise.resolve().then(()=>(init_relativePriority(),relativePriority_exports)),"./filters/types.ts":()=>Promise.resolve().then(()=>(init_types2(),types_exports2)),"./filters/userGoalStub.ts":()=>Promise.resolve().then(()=>(init_userGoalStub(),userGoalStub_exports)),"./filters/userTagPreference.ts":()=>Promise.resolve().then(()=>(init_userTagPreference(),userTagPreference_exports)),"./generators/CompositeGenerator.ts":()=>Promise.resolve().then(()=>(init_CompositeGenerator(),CompositeGenerator_exports)),"./generators/elo.ts":()=>Promise.resolve().then(()=>(init_elo(),elo_exports)),"./generators/index.ts":()=>Promise.resolve().then(()=>(init_generators(),generators_exports)),"./generators/prescribed.ts":()=>Promise.resolve().then(()=>(init_prescribed(),prescribed_exports)),"./generators/srs.ts":()=>Promise.resolve().then(()=>(init_srs(),srs_exports)),"./generators/types.ts":()=>Promise.resolve().then(()=>(init_types(),types_exports)),"./index.ts":()=>Promise.resolve().then(()=>(init_navigators(),navigators_exports))})}}),navigators_exports={};__export(navigators_exports,{ContentNavigator:()=>ContentNavigator,DIVERSITY_FLOOR:()=>DIVERSITY_FLOOR,DIVERSITY_STRENGTH:()=>DIVERSITY_STRENGTH,NavigatorRole:()=>NavigatorRole,NavigatorRoles:()=>NavigatorRoles,Navigators:()=>Navigators,diversityRerank:()=>diversityRerank,getActivePipeline:()=>getActivePipeline,getCardOrigin:()=>getCardOrigin,getRegisteredNavigator:()=>getRegisteredNavigator,getRegisteredNavigatorNames:()=>getRegisteredNavigatorNames,getRegisteredNavigatorRole:()=>getRegisteredNavigatorRole,hasRegisteredNavigator:()=>hasRegisteredNavigator,initializeNavigatorRegistry:()=>initializeNavigatorRegistry,isFilter:()=>isFilter,isGenerator:()=>isGenerator,mountPipelineDebugger:()=>mountPipelineDebugger,pipelineDebugAPI:()=>pipelineDebugAPI,registerNavigator:()=>registerNavigator});function registerNavigator(e,t,n){navigatorRegistry.set(e,{constructor:t,role:n}),logger.debug(`[NavigatorRegistry] Registered: ${e}${n?` (${n})`:``}`)}function getRegisteredNavigator(e){return navigatorRegistry.get(e)?.constructor}function hasRegisteredNavigator(e){return navigatorRegistry.has(e)}function getRegisteredNavigatorRole(e){return navigatorRegistry.get(e)?.role}function getRegisteredNavigatorNames(){return Array.from(navigatorRegistry.keys())}async function initializeNavigatorRegistry(){logger.debug(`[NavigatorRegistry] Initializing built-in navigators...`);let[e,t]=await Promise.all([Promise.resolve().then(()=>(init_elo(),elo_exports)),Promise.resolve().then(()=>(init_srs(),srs_exports))]),n=await Promise.resolve().then(()=>(init_prescribed(),prescribed_exports));registerNavigator(`elo`,e.default),registerNavigator(`srs`,t.default),registerNavigator(`prescribed`,n.default);let[r,a,o,s]=await Promise.all([Promise.resolve().then(()=>(init_hierarchyDefinition(),hierarchyDefinition_exports)),Promise.resolve().then(()=>(init_interferenceMitigator(),interferenceMitigator_exports)),Promise.resolve().then(()=>(init_relativePriority(),relativePriority_exports)),Promise.resolve().then(()=>(init_userTagPreference(),userTagPreference_exports))]);registerNavigator(`hierarchyDefinition`,r.default),registerNavigator(`interferenceMitigator`,a.default),registerNavigator(`relativePriority`,o.default),registerNavigator(`userTagPreference`,s.default),logger.debug(`[NavigatorRegistry] Initialized ${navigatorRegistry.size} navigators: ${getRegisteredNavigatorNames().join(`, `)}`)}function getCardOrigin(e){if(e.provenance.length===0)throw Error(`Card has no provenance - cannot determine origin`);let t=e.provenance[0].reason.toLowerCase();return t.includes(`failed`)?`failed`:t.includes(`review`)?`review`:`new`}function isGenerator(e){return NavigatorRoles[e]===`generator`?!0:getRegisteredNavigatorRole(e)===`generator`}function isFilter(e){return NavigatorRoles[e]===`filter`?!0:getRegisteredNavigatorRole(e)===`filter`}var navigatorRegistry,Navigators,NavigatorRole,NavigatorRoles,ContentNavigator,init_navigators=__esm({"src/core/navigators/index.ts"(){init_diversityRerank(),init_PipelineDebugger(),init_logger(),init_(),init_2(),init_3(),navigatorRegistry=new Map,Navigators=(e=>(e.ELO=`elo`,e.SRS=`srs`,e.PRESCRIBED=`prescribed`,e.HIERARCHY=`hierarchyDefinition`,e.INTERFERENCE=`interferenceMitigator`,e.RELATIVE_PRIORITY=`relativePriority`,e.USER_TAG_PREFERENCE=`userTagPreference`,e))(Navigators||{}),NavigatorRole=(e=>(e.GENERATOR=`generator`,e.FILTER=`filter`,e))(NavigatorRole||{}),NavigatorRoles={elo:`generator`,srs:`generator`,prescribed:`generator`,hierarchyDefinition:`filter`,interferenceMitigator:`filter`,relativePriority:`filter`,userTagPreference:`filter`},ContentNavigator=class{constructor(e,t,n){_defineProperty(this,`user`,void 0),_defineProperty(this,`course`,void 0),_defineProperty(this,`strategyName`,void 0),_defineProperty(this,`strategyId`,void 0),_defineProperty(this,`learnable`,void 0),_defineProperty(this,`staticWeight`,void 0),this.user=e,this.course=t,n&&(this.strategyName=n.name,this.strategyId=n._id,this.learnable=n.learnable,this.staticWeight=n.staticWeight)}get strategyKey(){return this.constructor.name}async getStrategyState(){if(!this.user||!this.course)throw Error(`Cannot get strategy state: navigator not properly initialized. Ensure user and course are provided to constructor.`);return this.user.getStrategyState(this.course.getCourseID(),this.strategyKey)}async putStrategyState(e){if(!this.user||!this.course)throw Error(`Cannot put strategy state: navigator not properly initialized. Ensure user and course are provided to constructor.`);return this.user.putStrategyState(this.course.getCourseID(),this.strategyKey,e)}static async create(e,t,n){let r=n.implementingClass,a=getRegisteredNavigator(r);if(a)return logger.debug(`[ContentNavigator.create] Using registered navigator: ${r}`),new a(e,t,n);logger.debug(`[ContentNavigator.create] Navigator not in registry, attempting dynamic import: ${r}`);let o;for(let e of[`.ts`,`.js`,``]){try{if(o=(await globImport_generators(`./generators/${r}${e}`)).default,o)break}catch(t){logger.debug(`Failed to load generator ${r}${e}:`,t)}try{if(o=(await globImport_filters(`./filters/${r}${e}`)).default,o)break}catch(t){logger.debug(`Failed to load filter ${r}${e}:`,t)}try{if(o=(await globImport(`./${r}${e}`)).default,o)break}catch(t){logger.debug(`Failed to load legacy ${r}${e}:`,t)}if(o)break}if(!o)throw Error(`Could not load navigator implementation for: ${r}`);return new o(e,t,n)}async getWeightedCards(e){throw Error(`${this.constructor.name} must implement getWeightedCards(). `)}setEphemeralHints(e){}}}});function randIntWeightedTowardZero(e){return Math.floor(Math.random()*Math.random()*Math.random()*e)}async function getCourseTagStubs(e){logger.debug(`Getting tag stubs for course: ${e}`);let t=await filterAllDocsByPrefix2(getCourseDB2(e),`TAG`.valueOf()+`-`);return t.rows.forEach(e=>{logger.debug(` Tag stub for doc: ${e.id}`)}),t}async function createTag(e,t,n){logger.debug(`Creating tag: ${t}...`);let r=getTagID(t);return await getCourseDB2(e).put({course:e,docType:`TAG`,name:t,snippet:``,taggedCards:[],wiki:``,author:n,_id:r})}async function updateTag(e){let t=await getTag(e.course,e.name);return await getCourseDB2(e.course).put({...e,_rev:t._rev})}async function getTag(e,t){let n=getTagID(t);return getCourseDB2(e).get(n)}async function removeTagFromCard(e,t,n){n=getTagID(n);let r=getCourseDB2(e),a=await r.get(n);return a.taggedCards=a.taggedCards.filter(e=>t!==e),r.put(a)}async function getAppliedTags(e,t){return await getCourseDB2(e).query(`getTags`,{startkey:t,endkey:t})}async function updateCredentialledCourseConfig(e,t){logger.debug(`Updating course config:
256
256
 
257
257
  ${JSON.stringify(t)}
258
258
  `);let n=getCourseDB2(e),r=await getCredentialledCourseConfig(e);return await n.put({...t,_rev:r._rev})}function isSuccessRow(e){return`doc`in e&&e.doc!==null&&e.doc!==void 0}var CoursesDB,CourseDB,init_courseDB=__esm({"src/impl/couch/courseDB.ts"(){init_couch(),init_updateQueue(),init_types_legacy(),init_logger(),init_clientCache(),init_courseAPI(),init_courseLookupDB(),init_navigators(),init_PipelineAssembler(),init_defaults(),CoursesDB=class{constructor(e){_defineProperty(this,`_courseIDs`,void 0),e&&e.length>0?this._courseIDs=e:this._courseIDs=void 0}async getCourseList(){let e=await CourseLookup.allCourseWare();return logger.debug(`AllCourses: ${e.map(e=>e.name+`, `+e._id+`
@@ -376,5 +376,5 @@ Example:
376
376
  `;if(!t)for(let e of d){let t={cardID:e.cardId,courseID:e.courseId,contentSourceType:`course`,contentSourceID:e.courseId,reviewID:e.reviewID,status:`review`};this.reviewQ.add(t,t.cardID),y+=`Review: ${e.courseId}::${e.cardId} (score: ${e.score.toFixed(2)})
377
377
  `}let b=_.filter(e=>e.score>=_SessionController.WELL_INDICATED_SCORE).length,x=[];for(let e of _){let t={cardID:e.cardId,courseID:e.courseId,contentSourceType:`course`,contentSourceID:e.courseId,status:`new`};x.push(t),y+=`New: ${e.courseId}::${e.cardId} (score: ${e.score.toFixed(2)})
378
378
  `}if(n){let e=this.newQ.mergeToFront(x,e=>e.cardID,v);y+=`Additive merge: ${e} new cards added to front of newQ
379
- `}else if(t)this.newQ.replaceAll(x,e=>e.cardID);else for(let e of x)this.newQ.add(e,e.cardID);return this.log(y),b}_getItemsToHydrate(){let e=[],t=2;for(let t=0;t<Math.min(2,this.reviewQ.length);t++)e.push(this.reviewQ.peek(t));for(let t=0;t<Math.min(2,this.newQ.length);t++)e.push(this.newQ.peek(t));for(let t=0;t<Math.min(2,this.failedQ.length);t++)e.push(this.failedQ.peek(t));return e}_selectNextItemToHydrate(){let e=Math.random(),t=.1,n=.75;if(this.reviewQ.length===0&&this.failedQ.length===0&&this.newQ.length===0||this._secondsRemaining<2&&this.failedQ.length===0&&this._minCardsGuarantee<=0)return null;if(this._secondsRemaining<=0&&this._minCardsGuarantee<=0)return this.failedQ.length>0?this.failedQ.peek(0):null;if(this.newQ.dequeueCount<this.sources.length&&this.newQ.length)return this.newQ.peek(0);let r=this.estimateCleanupTime(),a=this.estimateReviewTime();return this._secondsRemaining-(r+a)>20?(t=.5,n=.9):this._secondsRemaining-r>20?(t=.05,n=.9):(t=.01,n=.1),this.failedQ.length===0&&(n=1),this.reviewQ.length===0&&(t=n),e<t&&this.newQ.length?this.newQ.peek(0):e<n&&this.reviewQ.length?this.reviewQ.peek(0):this.failedQ.length?this.failedQ.peek(0):(this.log(`No more cards available for the session!`),null)}async nextCard(e=`dismiss-success`){if(this.dismissCurrentCard(e),this._minCardsGuarantee>0&&(this._minCardsGuarantee--,this.log(`[CardGuarantee] ${this._minCardsGuarantee} guaranteed cards remaining`)),this._replanPromise&&this.newQ.length===0&&this.reviewQ.length===0&&this.failedQ.length===0&&(this.log(`nextCard: queues empty, awaiting in-flight replan before drawing`),await this._replanPromise),this.newQ.length<=_SessionController.DEPLETION_PREFETCH_THRESHOLD&&this._secondsRemaining>0&&!this._replanPromise){this._suppressQualityReplan=!1;let e=this.reviewQ.length+this.failedQ.length;this.log(`[AutoReplan:depletion] newQ has ${this.newQ.length} card(s) (${e} in other queues) with ${this._secondsRemaining}s remaining. Triggering background replan.`),this.requestReplan({label:`auto:depletion`,mode:`merge`})}if(!this._suppressQualityReplan&&this._wellIndicatedRemaining<=3&&this.newQ.length>0&&!this._replanPromise&&(this.log(`[AutoReplan:quality] ${this._wellIndicatedRemaining} well-indicated cards remaining (newQ: ${this.newQ.length}). Triggering background replan.`),this.requestReplan({label:`auto:quality`})),this._secondsRemaining<=0&&this.failedQ.length===0&&this._minCardsGuarantee<=0)return this._currentCard=null,endSessionTracking(),null;let t=3,n=250,r=0;for(;this._secondsRemaining>0&&this.newQ.length===0&&this.reviewQ.length===0&&this.failedQ.length===0;)if(this.log(`[WedgeBreaker] All queues empty with ${this._secondsRemaining}s remaining. Running pipeline (attempt ${r+1}/3).`),await this._replanUncoalesced({label:`wedge-breaker`}),this.newQ.length===0&&this.reviewQ.length===0&&this.failedQ.length===0){if(r++,r>=3){this.log(`[WedgeBreaker] Pipeline returned no content 3 consecutive times. Giving up; session will end.`);break}await new Promise(e=>setTimeout(e,250))}else r=0;let a=20;for(let e=0;e<20;e++){let e=this._selectNextItemToHydrate();if(!e)return this._currentCard=null,endSessionTracking(),null;let t=this.hydrationService.getHydratedCard(e.cardID);if(t||(t=await this.hydrationService.waitForCard(e.cardID)),this.removeItemFromQueue(e),t){await this.hydrationService.ensureHydratedCards(),this._currentCard=t;let n=e.status===`review`||e.status===`failed-review`?`review`:e.status===`new`||e.status===`failed-new`?`new`:`failed`,r=e.status.startsWith(`failed`)?`failedQ`:e.status===`review`?`reviewQ`:`newQ`;return recordCardPresentation(e.cardID,e.courseID,this.courseNameCache.get(e.courseID),n,r),snapshotQueues(this.reviewQ.length,this.newQ.length,this.failedQ.length),t}this.log(`Skipping card ${e.cardID}: hydration failed, trying next`),isReview(e)&&this.srsService.removeReview(e.reviewID)}return this.log(`Exhausted 20 skip attempts finding a hydratable card`),this._currentCard=null,endSessionTracking(),null}async submitResponse(e,t,n,r,a,o,s,c,l){let u={...r.item},d=await this.services.response.processResponse(e,t,u,n,r,a,o,s,c,l);return await this._notifyOutcomeObservers(e,r,d),d}dismissCurrentCard(e=`dismiss-success`){if(this._currentCard)if(e===`dismiss-success`)this.hydrationService.removeCard(this._currentCard.item.cardID);else if(e===`marked-failed`){let e;e=isReview(this._currentCard.item)?{cardID:this._currentCard.item.cardID,courseID:this._currentCard.item.courseID,contentSourceID:this._currentCard.item.contentSourceID,contentSourceType:this._currentCard.item.contentSourceType,status:`failed-review`,reviewID:this._currentCard.item.reviewID}:{cardID:this._currentCard.item.cardID,courseID:this._currentCard.item.courseID,contentSourceID:this._currentCard.item.contentSourceID,contentSourceType:this._currentCard.item.contentSourceType,status:`failed-new`},this.failedQ.add(e,e.cardID)}else (e===`dismiss-error`||e===`dismiss-failed`)&&this.hydrationService.removeCard(this._currentCard.item.cardID)}removeItemFromQueue(e){this._clearDurableRequirement(e.cardID),this._servedCardIds.add(e.cardID),this.reviewQ.peek(0)?.cardID===e.cardID?this.reviewQ.dequeue(e=>e.cardID):this.newQ.peek(0)?.cardID===e.cardID?(this.newQ.dequeue(e=>e.cardID),this._wellIndicatedRemaining>0&&this._wellIndicatedRemaining--):this.failedQ.peek(0)?.cardID===e.cardID&&this.failedQ.dequeue(e=>e.cardID)}_clearDurableRequirement(e){let t=this._sessionHints?.requireCards;if(!t||t.length===0)return;let n=t.filter(t=>t!==e);n.length!==t.length&&(this._sessionHints={...this._sessionHints,requireCards:n.length>0?n:void 0},this.log(`[Replan] Durable requirement satisfied & cleared on draw: ${e}`))}async endSession(){if(!this._sessionRecord||this._sessionRecord.length===0)return;let e=this._sessionRecord.flatMap(e=>e.records).filter(e=>e.userAnswer!==void 0);if(e.length===0)return;let t=null,n=[];for(let e of this.sources)if(e.getOrchestrationContext){try{t=await e.getOrchestrationContext(),e.getStrategyIds&&n.push(...e.getStrategyIds())}catch(e){logger.warn(`[SessionController] Failed to get orchestration context: ${e}`)}if(t)break}if(!t){logger.debug(`[SessionController] No orchestration context available, skipping outcome recording`);return}let r=new Date().toISOString(),a=new Date(this.startTime).toISOString();await recordUserOutcome(t,a,r,e,n)}},_defineProperty(_SessionController2,`MIN_WELL_INDICATED`,5),_defineProperty(_SessionController2,`WELL_INDICATED_SCORE`,.1),_defineProperty(_SessionController2,`DEPLETION_PREFETCH_THRESHOLD`,3),_SessionController2);init_TagFilteredContentSource(),init_factory();export{mountUserDBDebugger as $,getCardHistoryID as A,initializeDataDirectory as B,computeOutcomeSignal as C,endSessionTracking as D,diversityRerank as E,getRegisteredNavigatorNames as F,isQuestionRecord as G,initializeNavigatorRegistry as H,getRegisteredNavigatorRole as I,log as J,isQuestionTypeRegistered as K,getStudySource as L,getDataLayer as M,getDbPath as N,ensureAppDataDirectory as O,getRegisteredNavigator as P,mountSessionDebugger as Q,hasRegisteredNavigator as R,computeEffectiveWeight as S,createOrchestrationContext as T,isFilter as U,initializeDataLayer as V,isGenerator as W,mountMixerDebugger as X,mixerDebugAPI as Y,mountPipelineDebugger as Z,TagFilteredContentSource as _,userDBDebugAPI as _t,DocType as a,registerDataShape as at,captureMixerRun as b,validateStaticCourse as bt,FileSystemError as c,registerSeedData as ct,NOT_SET as d,scoreAccuracyInZone as dt,newInterval as et,NavigatorRole as f,sessionDebugAPI as ft,SessionController as g,updateStrategyWeight as gt,QuotaRoundRobinMixer as h,updateLearningState as ht,DIVERSITY_STRENGTH as i,recordUserOutcome as it,getCardOrigin as j,getAppDataDirectory as k,GuestUsername as l,removeDataShape as lt,Navigators as m,startSessionTracking as mt,CourseLookup as n,processCustomQuestionsData as nt,DocTypePrefixes as o,registerNavigator as ot,NavigatorRoles as p,snapshotQueues as pt,isReview as q,DIVERSITY_FLOOR as r,recordCardPresentation as rt,ENV as s,registerQuestionType as st,ContentNavigator as t,pipelineDebugAPI as tt,Loggable as u,removeQuestionType as ut,areQuestionRecords as v,validateMigration as vt,computeSpread as w,computeDeviation as x,buildStrategyStateId as y,validateProcessorConfig as yt,importParsedCards as z};
380
- //# sourceMappingURL=dist-DvMZVJ8K.js.map
379
+ `}else if(t)this.newQ.replaceAll(x,e=>e.cardID);else for(let e of x)this.newQ.add(e,e.cardID);return this.log(y),b}_getItemsToHydrate(){let e=[],t=2;for(let t=0;t<Math.min(2,this.reviewQ.length);t++)e.push(this.reviewQ.peek(t));for(let t=0;t<Math.min(2,this.newQ.length);t++)e.push(this.newQ.peek(t));for(let t=0;t<Math.min(2,this.failedQ.length);t++)e.push(this.failedQ.peek(t));return e}_selectNextItemToHydrate(){let e=Math.random(),t=.1,n=.75;if(this.reviewQ.length===0&&this.failedQ.length===0&&this.newQ.length===0||this._secondsRemaining<2&&this.failedQ.length===0&&this._minCardsGuarantee<=0)return null;if(this._secondsRemaining<=0&&this._minCardsGuarantee<=0)return this.failedQ.length>0?this.failedQ.peek(0):null;if(this.newQ.dequeueCount<this.sources.length&&this.newQ.length)return this.newQ.peek(0);let r=this.estimateCleanupTime(),a=this.estimateReviewTime();return this._secondsRemaining-(r+a)>20?(t=.5,n=.9):this._secondsRemaining-r>20?(t=.05,n=.9):(t=.01,n=.1),this.failedQ.length===0&&(n=1),this.reviewQ.length===0&&(t=n),e<t&&this.newQ.length?this.newQ.peek(0):e<n&&this.reviewQ.length?this.reviewQ.peek(0):this.failedQ.length?this.failedQ.peek(0):(this.log(`No more cards available for the session!`),null)}async nextCard(e=`dismiss-success`){if(this.dismissCurrentCard(e),this._minCardsGuarantee>0&&(this._minCardsGuarantee--,this.log(`[CardGuarantee] ${this._minCardsGuarantee} guaranteed cards remaining`)),this._replanPromise&&this.newQ.length===0&&this.reviewQ.length===0&&this.failedQ.length===0&&(this.log(`nextCard: queues empty, awaiting in-flight replan before drawing`),await this._replanPromise),this.newQ.length<=_SessionController.DEPLETION_PREFETCH_THRESHOLD&&this._secondsRemaining>0&&!this._replanPromise){this._suppressQualityReplan=!1;let e=this.reviewQ.length+this.failedQ.length;this.log(`[AutoReplan:depletion] newQ has ${this.newQ.length} card(s) (${e} in other queues) with ${this._secondsRemaining}s remaining. Triggering background replan.`),this.requestReplan({label:`auto:depletion`,mode:`merge`})}if(!this._suppressQualityReplan&&this._wellIndicatedRemaining<=3&&this.newQ.length>0&&!this._replanPromise&&(this.log(`[AutoReplan:quality] ${this._wellIndicatedRemaining} well-indicated cards remaining (newQ: ${this.newQ.length}). Triggering background replan.`),this.requestReplan({label:`auto:quality`})),this._secondsRemaining<=0&&this.failedQ.length===0&&this._minCardsGuarantee<=0)return this._currentCard=null,endSessionTracking(),null;let t=3,n=250,r=0;for(;this._secondsRemaining>0&&this.newQ.length===0&&this.reviewQ.length===0&&this.failedQ.length===0;)if(this.log(`[WedgeBreaker] All queues empty with ${this._secondsRemaining}s remaining. Running pipeline (attempt ${r+1}/3).`),await this._replanUncoalesced({label:`wedge-breaker`}),this.newQ.length===0&&this.reviewQ.length===0&&this.failedQ.length===0){if(r++,r>=3){this.log(`[WedgeBreaker] Pipeline returned no content 3 consecutive times. Giving up; session will end.`);break}await new Promise(e=>setTimeout(e,250))}else r=0;let a=20;for(let e=0;e<20;e++){let e=this._selectNextItemToHydrate();if(!e)return this._currentCard=null,endSessionTracking(),null;let t=this.hydrationService.getHydratedCard(e.cardID);if(t||(t=await this.hydrationService.waitForCard(e.cardID)),this.removeItemFromQueue(e),t){await this.hydrationService.ensureHydratedCards(),this._currentCard=t;let n=e.status===`review`||e.status===`failed-review`?`review`:e.status===`new`||e.status===`failed-new`?`new`:`failed`,r=e.status.startsWith(`failed`)?`failedQ`:e.status===`review`?`reviewQ`:`newQ`;return recordCardPresentation(e.cardID,e.courseID,this.courseNameCache.get(e.courseID),n,r),snapshotQueues(this.reviewQ.length,this.newQ.length,this.failedQ.length),t}this.log(`Skipping card ${e.cardID}: hydration failed, trying next`),isReview(e)&&this.srsService.removeReview(e.reviewID)}return this.log(`Exhausted 20 skip attempts finding a hydratable card`),this._currentCard=null,endSessionTracking(),null}async submitResponse(e,t,n,r,a,o,s,c,l){let u={...r.item},d=await this.services.response.processResponse(e,t,u,n,r,a,o,s,c,l);return await this._notifyOutcomeObservers(e,r,d),d}dismissCurrentCard(e=`dismiss-success`){if(this._currentCard)if(e===`dismiss-success`)this.hydrationService.removeCard(this._currentCard.item.cardID);else if(e===`marked-failed`){let e;e=isReview(this._currentCard.item)?{cardID:this._currentCard.item.cardID,courseID:this._currentCard.item.courseID,contentSourceID:this._currentCard.item.contentSourceID,contentSourceType:this._currentCard.item.contentSourceType,status:`failed-review`,reviewID:this._currentCard.item.reviewID}:{cardID:this._currentCard.item.cardID,courseID:this._currentCard.item.courseID,contentSourceID:this._currentCard.item.contentSourceID,contentSourceType:this._currentCard.item.contentSourceType,status:`failed-new`},this.failedQ.add(e,e.cardID)}else (e===`dismiss-error`||e===`dismiss-failed`)&&this.hydrationService.removeCard(this._currentCard.item.cardID)}removeItemFromQueue(e){this._clearDurableRequirement(e.cardID),this._servedCardIds.add(e.cardID),this.reviewQ.peek(0)?.cardID===e.cardID?this.reviewQ.dequeue(e=>e.cardID):this.newQ.peek(0)?.cardID===e.cardID?(this.newQ.dequeue(e=>e.cardID),this._wellIndicatedRemaining>0&&this._wellIndicatedRemaining--):this.failedQ.peek(0)?.cardID===e.cardID&&this.failedQ.dequeue(e=>e.cardID)}_clearDurableRequirement(e){let t=this._sessionHints?.requireCards;if(!t||t.length===0)return;let n=t.filter(t=>t!==e);n.length!==t.length&&(this._sessionHints={...this._sessionHints,requireCards:n.length>0?n:void 0},this.log(`[Replan] Durable requirement satisfied & cleared on draw: ${e}`))}async endSession(){if(!this._sessionRecord||this._sessionRecord.length===0)return;let e=this._sessionRecord.flatMap(e=>e.records).filter(e=>e.userAnswer!==void 0);if(e.length===0)return;let t=null,n=[];for(let e of this.sources)if(e.getOrchestrationContext){try{t=await e.getOrchestrationContext(),e.getStrategyIds&&n.push(...e.getStrategyIds())}catch(e){logger.warn(`[SessionController] Failed to get orchestration context: ${e}`)}if(t)break}if(!t){logger.debug(`[SessionController] No orchestration context available, skipping outcome recording`);return}let r=new Date().toISOString(),a=new Date(this.startTime).toISOString();await recordUserOutcome(t,a,r,e,n)}},_defineProperty(_SessionController2,`MIN_WELL_INDICATED`,5),_defineProperty(_SessionController2,`WELL_INDICATED_SCORE`,.1),_defineProperty(_SessionController2,`DEPLETION_PREFETCH_THRESHOLD`,3),_SessionController2);init_TagFilteredContentSource(),init_factory();export{mountSessionDebugger as $,getAppDataDirectory as A,importParsedCards as B,computeOutcomeSignal as C,endSessionTracking as D,diversityRerank as E,getRegisteredNavigator as F,isGenerator as G,initializeDataLayer as H,getRegisteredNavigatorNames as I,isReview as J,isQuestionRecord as K,getRegisteredNavigatorRole as L,getCardOrigin as M,getDataLayer as N,ensureAppDataDirectory as O,getDbPath as P,mountPipelineDebugger as Q,getStudySource as R,computeEffectiveWeight as S,createOrchestrationContext as T,initializeNavigatorRegistry as U,initializeDataDirectory as V,isFilter as W,mixerDebugAPI as X,log as Y,mountMixerDebugger as Z,TagFilteredContentSource as _,updateStrategyWeight as _t,DocType as a,recordUserOutcome as at,captureMixerRun as b,validateProcessorConfig as bt,FileSystemError as c,registerQuestionType as ct,NOT_SET as d,removeQuestionType as dt,mountUserDBDebugger as et,NavigatorRole as f,scoreAccuracyInZone as ft,SessionController as g,updateLearningState as gt,QuotaRoundRobinMixer as h,startSessionTracking as ht,DIVERSITY_STRENGTH as i,recordCardPresentation as it,getCardHistoryID as j,getActivePipeline as k,GuestUsername as l,registerSeedData as lt,Navigators as m,snapshotQueues as mt,CourseLookup as n,pipelineDebugAPI as nt,DocTypePrefixes as o,registerDataShape as ot,NavigatorRoles as p,sessionDebugAPI as pt,isQuestionTypeRegistered as q,DIVERSITY_FLOOR as r,processCustomQuestionsData as rt,ENV as s,registerNavigator as st,ContentNavigator as t,newInterval as tt,Loggable as u,removeDataShape as ut,areQuestionRecords as v,userDBDebugAPI as vt,computeSpread as w,computeDeviation as x,validateStaticCourse as xt,buildStrategyStateId as y,validateMigration as yt,hasRegisteredNavigator as z};
380
+ //# sourceMappingURL=dist-DHNWv7su.js.map