diff --git a/.env.example b/.env.example index 9caadf92..78ded632 100644 --- a/.env.example +++ b/.env.example @@ -246,6 +246,19 @@ EMAIL_REQUIRE_SENDER_AUTH_HEADERS=true MIGADU_API_USER=your_migadu_api_user MIGADU_API_KEY=your_migadu_api_key MIGADU_MAILBOX_DOMAIN=508.dev +# Newsletter sync settings are dashboard-managed in normal deployments. +# Set non-empty env values only when you intentionally want env to lock dashboard edits. +# BREVO_API_KEY= +# BREVO_API_BASE_URL=https://api.brevo.com/v3 +# BREVO_API_TIMEOUT_SECONDS=20.0 +# BREVO_508_MEMBERS_NEWSLETTER_LIST_ID=4 +# BREVO_508_MEMBERS_NEWSLETTER_LIST_NAME="508 members" +# KEILA_API_KEY= +# KEILA_API_BASE_URL=https://app.keila.io +# KEILA_API_TIMEOUT_SECONDS=20.0 +# NEWSLETTER_SYNC_ENABLED=true +# NEWSLETTER_SYNC_INTERVAL_SECONDS=604800 +# NEWSLETTER_SYNC_EXCLUDED_MAILBOXES=system,service-account # EspoCRM (required for worker integration) ESPO_API_KEY=your_key_here diff --git a/ENVIRONMENT.md b/ENVIRONMENT.md index 85d954a8..e9f03d3a 100644 --- a/ENVIRONMENT.md +++ b/ENVIRONMENT.md @@ -173,6 +173,20 @@ current precedence rules. - `Required for /create-mailbox and /create-user-accounts`: `MIGADU_API_USER`, `MIGADU_API_KEY` - `Optional`: `MIGADU_MAILBOX_DOMAIN` (default: `508.dev`) +- Newsletter sync settings are normally set from the admin dashboard configuration page. A non-empty env or `.env` value locks the matching dashboard field. +- `Optional for Brevo newsletter sync`: `BREVO_API_KEY` +- `Optional`: `BREVO_API_BASE_URL` (default: `https://api.brevo.com/v3`) +- `Optional`: `BREVO_API_TIMEOUT_SECONDS` (default: `20.0`) +- `Optional for Brevo newsletter sync`: `BREVO_508_MEMBERS_NEWSLETTER_LIST_ID` (explicit Brevo list ID override; use `4` for the 508 members list when setting it directly) +- `Optional`: `BREVO_508_MEMBERS_NEWSLETTER_LIST_NAME` (default: `508 members`; used to look up the list ID when the explicit ID is unset) +- `Optional for Keila contact sync`: `KEILA_API_KEY` +- `Optional`: `KEILA_API_BASE_URL` (default: `https://app.keila.io`) +- `Optional`: `KEILA_API_TIMEOUT_SECONDS` (default: `20.0`) +- `Optional`: `NEWSLETTER_SYNC_ENABLED` (default: `true`) +- `Optional`: `NEWSLETTER_SYNC_INTERVAL_SECONDS` (default: `604800`, one week) +- `Optional`: `NEWSLETTER_SYNC_EXCLUDED_MAILBOXES` (comma-separated mailbox local-parts or full addresses to skip during Migadu resync) +- Note: mailbox and backup email subscription to configured newsletter tools is best effort. Failures are reported as warnings and do not block mailbox or account creation. +- Note: the periodic sync uses Migadu mailboxes and password recovery emails as the source of truth for `@508.dev`. When CRM is configured, it only syncs mailboxes that match a CRM contact; it also skips configured excluded mailboxes and does not re-add provider-suppressed contacts. ## Authentik SSO Provisioning diff --git a/apps/admin_dashboard/src/main.tsx b/apps/admin_dashboard/src/main.tsx index 7059a010..59babaa6 100644 --- a/apps/admin_dashboard/src/main.tsx +++ b/apps/admin_dashboard/src/main.tsx @@ -116,6 +116,11 @@ const configurationGroups: ConfigurationGroupMetadata[] = [ description: "Editable onboarding integrations such as DocuSeal, Outline, and onboarding email SMTP.", }, + { + category: "Newsletter", + label: "Newsletter", + description: "Brevo, Keila, and recurring 508 members audience sync settings.", + }, { category: "AI", label: "AI Providers", @@ -1939,6 +1944,32 @@ function App() { } } + async function syncNewsletters() { + setBusy("syncNewsletters", true) + showToast("Queueing newsletter sync") + try { + const payload = await requestJson<{ + job_id?: string + dry_run?: boolean + would_enqueue?: { job_type?: string } + }>("/dashboard/api/sync/newsletters", { + method: "POST", + }) + if (payload.dry_run) { + showToast( + `Dry run only: would queue ${payload.would_enqueue?.job_type || "newsletter sync"}`, + "warning", + ) + } else { + showToast(`Queued newsletter sync ${payload.job_id}`, "ok") + } + } catch (error) { + showError(error, "Unable to queue newsletter sync") + } finally { + setBusy("syncNewsletters", false) + } + } + async function assignOnboarder(contactId: string | undefined, onboarder: string) { const normalizedContactId = String(contactId || "").trim() const normalizedOnboarder = onboarder.trim() @@ -2454,6 +2485,7 @@ function App() { people={sortedPeople} sort={sort.people} canSync={canUse("people:sync")} + canSyncNewsletters={canUse("people:sync")} loading={loading} peopleQuery={peopleQuery} peopleMember={peopleMember} @@ -2463,6 +2495,7 @@ function App() { peopleFilterKeys={peopleFilterKeys} onSearch={loadPeople} onSync={syncPeople} + onSyncNewsletters={syncNewsletters} onSort={(key) => handleSort("people", key)} setPeopleQuery={setPeopleQuery} setPeopleMember={setPeopleMember} @@ -5492,6 +5525,7 @@ function PeopleView(props: { people: Person[] sort: { key: string; direction: SortDirection } canSync: boolean + canSyncNewsletters: boolean loading: Record peopleQuery: string peopleMember: string @@ -5501,6 +5535,7 @@ function PeopleView(props: { peopleFilterKeys: PeopleFilterKey[] onSearch: () => void onSync: () => void + onSyncNewsletters: () => void onSort: (key: string) => void setPeopleQuery: (value: string) => void setPeopleMember: (value: string) => void @@ -5529,6 +5564,19 @@ function PeopleView(props: { Sync people ) : null} + {props.canSyncNewsletters ? ( + + ) : null} {props.crmBaseUrl ? ( AgentOrchestrator: _AGENT_ORCHESTRATOR = AgentOrchestrator( registry=ToolRegistry( _AGENT_TASK_STORE, - runtime_config=ToolRuntimeConfig.from_settings(settings), + runtime_config_factory=lambda: ToolRuntimeConfig.from_settings( + settings + ), ), model_config=AgentModelConfig.from_settings(settings), intent_normalizer=OpenAICompatibleIntentNormalizer.from_settings( @@ -580,6 +582,12 @@ def _crm_sync_idempotency_key(*, now: datetime) -> str: return f"crm-sync:{bucket}" +def _newsletter_sync_idempotency_key(*, now: datetime) -> str: + interval_seconds = max(1, settings.newsletter_sync_interval_seconds) + bucket = int(now.timestamp()) // interval_seconds + return f"newsletter-sync:508-members:{bucket}" + + def _normalize_google_forms_input(value: str | None) -> str | None: if not isinstance(value, str): return None @@ -663,6 +671,37 @@ async def _enqueue_erpnext_project_sync_job( return job +async def _enqueue_newsletter_sync_job( + queue: QueueClient, + *, + reason: str, +) -> EnqueuedJob: + now = datetime.now(tz=timezone.utc) + idempotency_key = ( + _newsletter_sync_idempotency_key(now=now) + if reason == "scheduler" + else ( + f"newsletter-sync:508-members:{reason}:" + f"{now.strftime('%Y%m%d%H%M%S%f')}:{uuid4().hex}" + ) + ) + job: EnqueuedJob = await asyncio.to_thread( + enqueue_job, + queue=queue, + fn=JOB_FUNCTIONS["sync_508_members_newsletters_job"], + args=(), + settings=settings, + idempotency_key=idempotency_key, + ) + logger.info( + "Enqueued 508 members newsletter sync job id=%s created=%s reason=%s", + job.id, + job.created, + reason, + ) + return job + + async def _crm_sync_scheduler(app: FastAPI) -> None: queue = app.state.queue interval_seconds = max(1, settings.crm_sync_interval_seconds) @@ -674,6 +713,17 @@ async def _crm_sync_scheduler(app: FastAPI) -> None: await asyncio.sleep(interval_seconds) +async def _newsletter_sync_scheduler(app: FastAPI) -> None: + queue = app.state.queue + interval_seconds = max(1, settings.newsletter_sync_interval_seconds) + while True: + try: + await _enqueue_newsletter_sync_job(queue, reason="scheduler") + except Exception: + logger.exception("Failed scheduling 508 members newsletter sync job") + await asyncio.sleep(interval_seconds) + + async def _email_resume_scheduler() -> None: """Run periodic mailbox polling for resume ingestion.""" poller = ResumeMailboxProcessor(settings) @@ -6611,6 +6661,64 @@ async def dashboard_sync_people_handler(request: Request) -> JSONResponse: ) +async def dashboard_sync_newsletters_handler(request: Request) -> JSONResponse: + """Queue a 508 members newsletter sync from the authenticated dashboard.""" + session, error_response, dry_run = await _dashboard_write_session_or_dry_run( + request, + required_permission=DASHBOARD_PERMISSION_PEOPLE_SYNC, + dry_run_permission=DASHBOARD_PERMISSION_PEOPLE_SYNC_DRY_RUN, + ) + if error_response is not None: + return error_response + assert session is not None + + csrf_error = _dashboard_same_origin_post_or_error(request) + if csrf_error is not None: + return csrf_error + + if dry_run: + return JSONResponse( + { + "status": "dry_run", + "dry_run": True, + "source": "dashboard", + "would_enqueue": { + "queue": settings.redis_queue_name, + "job_type": "sync_508_members_newsletters_job", + "reason": "dashboard", + "idempotency_key_pattern": "newsletter-sync:508-members:dashboard::", + }, + } + ) + + job = await _enqueue_newsletter_sync_job( + request.app.state.queue, reason="dashboard" + ) + actor_provider, actor_subject = _session_audit_actor(session) + await _write_auth_audit_event( + action="newsletter.508_members_sync", + result=AuditResult.SUCCESS, + actor_subject=actor_subject, + actor_display_name=session.display_name, + actor_provider=actor_provider, + resource_type="newsletter_sync", + resource_id=job.id, + metadata={ + "source": "dashboard", + "queue": settings.redis_queue_name, + }, + ) + return JSONResponse( + { + "status": "queued", + "source": "dashboard", + "job_id": job.id, + "created": job.created, + }, + status_code=202, + ) + + async def espocrm_people_sync_webhook_handler(request: Request) -> JSONResponse: """Queue per-contact people cache sync jobs from CRM webhook events.""" if not _is_webhook_authorized(request): @@ -8085,6 +8193,13 @@ async def _lifespan(app: FastAPI) -> Any: else: logger.info("CRM sync scheduler disabled by config") + if settings.newsletter_sync_enabled: + app.state.newsletter_sync_task = asyncio.create_task( + _newsletter_sync_scheduler(app) + ) + else: + logger.info("508 members newsletter sync scheduler disabled by config") + if settings.email_resume_intake_enabled: app.state.email_resume_task = asyncio.create_task(_email_resume_scheduler()) else: @@ -8105,6 +8220,12 @@ async def _lifespan(app: FastAPI) -> Any: with contextlib.suppress(asyncio.CancelledError): await task + if hasattr(app.state, "newsletter_sync_task"): + task = app.state.newsletter_sync_task + task.cancel() + with contextlib.suppress(asyncio.CancelledError): + await task + if hasattr(app.state, "http_client"): await app.state.http_client.aclose() @@ -8330,6 +8451,11 @@ def create_app(*, run_lifespan: bool = True) -> FastAPI: dashboard_sync_people_handler, methods=["POST"], ) + app.add_api_route( + "/dashboard/api/sync/newsletters", + dashboard_sync_newsletters_handler, + methods=["POST"], + ) app.add_api_route( "/dashboard/gigs/{item_id}", dashboard_handler, diff --git a/apps/api/src/five08/backend/static/dashboard/.vite/manifest.json b/apps/api/src/five08/backend/static/dashboard/.vite/manifest.json index e905a9b4..fe56994a 100644 --- a/apps/api/src/five08/backend/static/dashboard/.vite/manifest.json +++ b/apps/api/src/five08/backend/static/dashboard/.vite/manifest.json @@ -1,6 +1,6 @@ { "index.html": { - "file": "assets/index-DUbmN0NW.js", + "file": "assets/index-DLefnMNV.js", "name": "index", "src": "index.html", "isEntry": true, diff --git a/apps/api/src/five08/backend/static/dashboard/assets/index-DUbmN0NW.js b/apps/api/src/five08/backend/static/dashboard/assets/index-DLefnMNV.js similarity index 64% rename from apps/api/src/five08/backend/static/dashboard/assets/index-DUbmN0NW.js rename to apps/api/src/five08/backend/static/dashboard/assets/index-DLefnMNV.js index 38e1a73c..8b534697 100644 --- a/apps/api/src/five08/backend/static/dashboard/assets/index-DUbmN0NW.js +++ b/apps/api/src/five08/backend/static/dashboard/assets/index-DLefnMNV.js @@ -6,4 +6,4 @@ var e=(e,t)=>()=>(t||(e((t={exports:{}}).exports,t),e=null),t.exports);(function `+c[r].replace(` at new `,` at `);return e.displayName&&u.includes(``)&&(u=u.replace(``,e.displayName)),u}while(1<=r&&0<=i);break}}}finally{we=!1,Error.prepareStackTrace=n}return(n=e?e.displayName||e.name:``)?Ce(n):``}function Ee(e,t){switch(e.tag){case 26:case 27:case 5:return Ce(e.type);case 16:return Ce(`Lazy`);case 13:return e.child!==t&&t!==null?Ce(`Suspense Fallback`):Ce(`Suspense`);case 19:return Ce(`SuspenseList`);case 0:case 15:return Te(e.type,!1);case 11:return Te(e.type.render,!1);case 1:return Te(e.type,!0);case 31:return Ce(`Activity`);default:return``}}function De(e){try{var t=``,n=null;do t+=Ee(e,n),n=e,e=e.return;while(e);return t}catch(e){return` Error generating stack: `+e.message+` `+e.stack}}var Oe=Object.prototype.hasOwnProperty,ke=t.unstable_scheduleCallback,Ae=t.unstable_cancelCallback,je=t.unstable_shouldYield,Me=t.unstable_requestPaint,Ne=t.unstable_now,Pe=t.unstable_getCurrentPriorityLevel,Fe=t.unstable_ImmediatePriority,Ie=t.unstable_UserBlockingPriority,Le=t.unstable_NormalPriority,Re=t.unstable_LowPriority,ze=t.unstable_IdlePriority,Be=t.log,Ve=t.unstable_setDisableYieldValue,He=null,Ue=null;function We(e){if(typeof Be==`function`&&Ve(e),Ue&&typeof Ue.setStrictMode==`function`)try{Ue.setStrictMode(He,e)}catch{}}var Ge=Math.clz32?Math.clz32:Je,Ke=Math.log,qe=Math.LN2;function Je(e){return e>>>=0,e===0?32:31-(Ke(e)/qe|0)|0}var Ye=256,Xe=262144,Ze=4194304;function Qe(e){var t=e&42;if(t!==0)return t;switch(e&-e){case 1:return 1;case 2:return 2;case 4:return 4;case 8:return 8;case 16:return 16;case 32:return 32;case 64:return 64;case 128:return 128;case 256:case 512:case 1024:case 2048:case 4096:case 8192:case 16384:case 32768:case 65536:case 131072:return e&261888;case 262144:case 524288:case 1048576:case 2097152:return e&3932160;case 4194304:case 8388608:case 16777216:case 33554432:return e&62914560;case 67108864:return 67108864;case 134217728:return 134217728;case 268435456:return 268435456;case 536870912:return 536870912;case 1073741824:return 0;default:return e}}function $e(e,t,n){var r=e.pendingLanes;if(r===0)return 0;var i=0,a=e.suspendedLanes,o=e.pingedLanes;e=e.warmLanes;var s=r&134217727;return s===0?(s=r&~a,s===0?o===0?n||(n=r&~e,n!==0&&(i=Qe(n))):i=Qe(o):i=Qe(s)):(r=s&~a,r===0?(o&=s,o===0?n||(n=s&~e,n!==0&&(i=Qe(n))):i=Qe(o)):i=Qe(r)),i===0?0:t!==0&&t!==i&&(t&a)===0&&(a=i&-i,n=t&-t,a>=n||a===32&&n&4194048)?t:i}function et(e,t){return(e.pendingLanes&~(e.suspendedLanes&~e.pingedLanes)&t)===0}function tt(e,t){switch(e){case 1:case 2:case 4:case 8:case 64:return t+250;case 16:case 32:case 128:case 256:case 512:case 1024:case 2048:case 4096:case 8192:case 16384:case 32768:case 65536:case 131072:case 262144:case 524288:case 1048576:case 2097152:return t+5e3;case 4194304:case 8388608:case 16777216:case 33554432:return-1;case 67108864:case 134217728:case 268435456:case 536870912:case 1073741824:return-1;default:return-1}}function nt(){var e=Ze;return Ze<<=1,!(Ze&62914560)&&(Ze=4194304),e}function j(e){for(var t=[],n=0;31>n;n++)t.push(e);return t}function rt(e,t){e.pendingLanes|=t,t!==268435456&&(e.suspendedLanes=0,e.pingedLanes=0,e.warmLanes=0)}function it(e,t,n,r,i,a){var o=e.pendingLanes;e.pendingLanes=n,e.suspendedLanes=0,e.pingedLanes=0,e.warmLanes=0,e.expiredLanes&=n,e.entangledLanes&=n,e.errorRecoveryDisabledLanes&=n,e.shellSuspendCounter=0;var s=e.entanglements,c=e.expirationTimes,l=e.hiddenUpdates;for(n=o&~n;0`u`||window.document===void 0||window.document.createElement===void 0),tn=!1;if(en)try{var nn={};Object.defineProperty(nn,`passive`,{get:function(){tn=!0}}),window.addEventListener(`test`,nn,nn),window.removeEventListener(`test`,nn,nn)}catch{tn=!1}var rn=null,an=null,on=null;function sn(){if(on)return on;var e,t=an,n=t.length,r,i=`value`in rn?rn.value:rn.textContent,a=i.length;for(e=0;e=Rn),Vn=` `,Hn=!1;function Un(e,t){switch(e){case`keyup`:return In.indexOf(t.keyCode)!==-1;case`keydown`:return t.keyCode!==229;case`keypress`:case`mousedown`:case`focusout`:return!0;default:return!1}}function Wn(e){return e=e.detail,typeof e==`object`&&`data`in e?e.data:null}var Gn=!1;function Kn(e,t){switch(e){case`compositionend`:return Wn(t);case`keypress`:return t.which===32?(Hn=!0,Vn):null;case`textInput`:return e=t.data,e===Vn&&Hn?null:e;default:return null}}function qn(e,t){if(Gn)return e===`compositionend`||!Ln&&Un(e,t)?(e=sn(),on=an=rn=null,Gn=!1,e):null;switch(e){case`paste`:return null;case`keypress`:if(!(t.ctrlKey||t.altKey||t.metaKey)||t.ctrlKey&&t.altKey){if(t.char&&1=t)return{node:n,offset:t-e};e=r}a:{for(;n;){if(n.nextSibling){n=n.nextSibling;break a}n=n.parentNode}n=void 0}n=hr(n)}}function _r(e,t){return e&&t?e===t?!0:e&&e.nodeType===3?!1:t&&t.nodeType===3?_r(e,t.parentNode):`contains`in e?e.contains(t):e.compareDocumentPosition?!!(e.compareDocumentPosition(t)&16):!1:!1}function vr(e){e=e!=null&&e.ownerDocument!=null&&e.ownerDocument.defaultView!=null?e.ownerDocument.defaultView:window;for(var t=R(e.document);t instanceof e.HTMLIFrameElement;){try{var n=typeof t.contentWindow.location.href==`string`}catch{n=!1}if(n)e=t.contentWindow;else break;t=R(e.document)}return t}function yr(e){var t=e&&e.nodeName&&e.nodeName.toLowerCase();return t&&(t===`input`&&(e.type===`text`||e.type===`search`||e.type===`tel`||e.type===`url`||e.type===`password`)||t===`textarea`||e.contentEditable===`true`)}var br=en&&`documentMode`in document&&11>=document.documentMode,xr=null,Sr=null,Cr=null,wr=!1;function Tr(e,t,n){var r=n.window===n?n.document:n.nodeType===9?n:n.ownerDocument;wr||xr==null||xr!==R(r)||(r=xr,`selectionStart`in r&&yr(r)?r={start:r.selectionStart,end:r.selectionEnd}:(r=(r.ownerDocument&&r.ownerDocument.defaultView||window).getSelection(),r={anchorNode:r.anchorNode,anchorOffset:r.anchorOffset,focusNode:r.focusNode,focusOffset:r.focusOffset}),Cr&&mr(Cr,r)||(Cr=r,r=Td(Sr,`onSelect`),0>=o,i-=o,_i=1<<32-Ge(t)+i|n<h?(g=d,d=null):g=d.sibling;var _=p(i,d,s[h],c);if(_===null){d===null&&(d=g);break}e&&d&&_.alternate===null&&t(i,d),a=o(_,a,h),u===null?l=_:u.sibling=_,u=_,d=g}if(h===s.length)return n(i,d),J&&yi(i,h),l;if(d===null){for(;hg?(_=h,h=null):_=h.sibling;var y=p(i,h,v.value,l);if(y===null){h===null&&(h=_);break}e&&h&&y.alternate===null&&t(i,h),s=o(y,s,g),d===null?u=y:d.sibling=y,d=y,h=_}if(v.done)return n(i,h),J&&yi(i,g),u;if(h===null){for(;!v.done;g++,v=c.next())v=f(i,v.value,l),v!==null&&(s=o(v,s,g),d===null?u=v:d.sibling=v,d=v);return J&&yi(i,g),u}for(h=r(h);!v.done;g++,v=c.next())v=m(h,i,g,v.value,l),v!==null&&(e&&v.alternate!==null&&h.delete(v.key===null?g:v.key),s=o(v,s,g),d===null?u=v:d.sibling=v,d=v);return e&&h.forEach(function(e){return t(i,e)}),J&&yi(i,g),u}function b(e,r,o,c){if(typeof o==`object`&&o&&o.type===_&&o.key===null&&(o=o.props.children),typeof o==`object`&&o){switch(o.$$typeof){case h:a:{for(var l=o.key;r!==null;){if(r.key===l){if(l=o.type,l===_){if(r.tag===7){n(e,r.sibling),c=i(r,o.props.children),c.return=e,e=c;break a}}else if(r.elementType===l||typeof l==`object`&&l&&l.$$typeof===ne&&ya(l)===r.type){n(e,r.sibling),c=i(r,o.props),Ea(c,o),c.return=e,e=c;break a}n(e,r);break}else t(e,r);r=r.sibling}o.type===_?(c=ii(o.props.children,e.mode,c,o.key),c.return=e,e=c):(c=ri(o.type,o.key,o.props,null,e.mode,c),Ea(c,o),c.return=e,e=c)}return s(e);case g:a:{for(l=o.key;r!==null;){if(r.key===l)if(r.tag===4&&r.stateNode.containerInfo===o.containerInfo&&r.stateNode.implementation===o.implementation){n(e,r.sibling),c=i(r,o.children||[]),c.return=e,e=c;break a}else{n(e,r);break}else t(e,r);r=r.sibling}c=si(o,e.mode,c),c.return=e,e=c}return s(e);case ne:return o=ya(o),b(e,r,o,c)}if(w(o))return v(e,r,o,c);if(oe(o)){if(l=oe(o),typeof l!=`function`)throw Error(a(150));return o=l.call(o),y(e,r,o,c)}if(typeof o.then==`function`)return b(e,r,Ta(o),c);if(o.$$typeof===x)return b(e,r,qi(e,o),c);Da(e,o)}return typeof o==`string`&&o!==``||typeof o==`number`||typeof o==`bigint`?(o=``+o,r!==null&&r.tag===6?(n(e,r.sibling),c=i(r,o),c.return=e,e=c):(n(e,r),c=ai(o,e.mode,c),c.return=e,e=c),s(e)):n(e,r)}return function(e,t,n,r){try{wa=0;var i=b(e,t,n,r);return Ca=null,i}catch(t){if(t===pa||t===ha)throw t;var a=$r(29,t,null,e.mode);return a.lanes=r,a.return=e,a}}}var ka=Oa(!0),Aa=Oa(!1),ja=!1;function Ma(e){e.updateQueue={baseState:e.memoizedState,firstBaseUpdate:null,lastBaseUpdate:null,shared:{pending:null,lanes:0,hiddenCallbacks:null},callbacks:null}}function Na(e,t){e=e.updateQueue,t.updateQueue===e&&(t.updateQueue={baseState:e.baseState,firstBaseUpdate:e.firstBaseUpdate,lastBaseUpdate:e.lastBaseUpdate,shared:e.shared,callbacks:null})}function Pa(e){return{lane:e,tag:0,payload:null,callback:null,next:null}}function Fa(e,t,n){var r=e.updateQueue;if(r===null)return null;if(r=r.shared,X&2){var i=r.pending;return i===null?t.next=t:(t.next=i.next,i.next=t),r.pending=t,t=Xr(e),Yr(e,null,n),t}return Kr(e,r,t,n),Xr(e)}function Ia(e,t,n){if(t=t.updateQueue,t!==null&&(t=t.shared,n&4194048)){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,ot(e,n)}}function La(e,t){var n=e.updateQueue,r=e.alternate;if(r!==null&&(r=r.updateQueue,n===r)){var i=null,a=null;if(n=n.firstBaseUpdate,n!==null){do{var o={lane:n.lane,tag:n.tag,payload:n.payload,callback:null,next:null};a===null?i=a=o:a=a.next=o,n=n.next}while(n!==null);a===null?i=a=t:a=a.next=t}else i=a=t;n={baseState:r.baseState,firstBaseUpdate:i,lastBaseUpdate:a,shared:r.shared,callbacks:r.callbacks},e.updateQueue=n;return}e=n.lastBaseUpdate,e===null?n.firstBaseUpdate=t:e.next=t,n.lastBaseUpdate=t}var Ra=!1;function za(){if(Ra){var e=ia;if(e!==null)throw e}}function Ba(e,t,n,r){Ra=!1;var i=e.updateQueue;ja=!1;var a=i.firstBaseUpdate,o=i.lastBaseUpdate,s=i.shared.pending;if(s!==null){i.shared.pending=null;var c=s,l=c.next;c.next=null,o===null?a=l:o.next=l,o=c;var u=e.alternate;u!==null&&(u=u.updateQueue,s=u.lastBaseUpdate,s!==o&&(s===null?u.firstBaseUpdate=l:s.next=l,u.lastBaseUpdate=c))}if(a!==null){var d=i.baseState;o=0,u=l=c=null,s=a;do{var f=s.lane&-536870913,m=f!==s.lane;if(m?(Q&f)===f:(r&f)===f){f!==0&&f===ra&&(Ra=!0),u!==null&&(u=u.next={lane:0,tag:s.tag,payload:s.payload,callback:null,next:null});a:{var h=e,g=s;f=t;var _=n;switch(g.tag){case 1:if(h=g.payload,typeof h==`function`){d=h.call(_,d,f);break a}d=h;break a;case 3:h.flags=h.flags&-65537|128;case 0:if(h=g.payload,f=typeof h==`function`?h.call(_,d,f):h,f==null)break a;d=p({},d,f);break a;case 2:ja=!0}}f=s.callback,f!==null&&(e.flags|=64,m&&(e.flags|=8192),m=i.callbacks,m===null?i.callbacks=[f]:m.push(f))}else m={lane:f,tag:s.tag,payload:s.payload,callback:s.callback,next:null},u===null?(l=u=m,c=d):u=u.next=m,o|=f;if(s=s.next,s===null){if(s=i.shared.pending,s===null)break;m=s,s=m.next,m.next=null,i.lastBaseUpdate=m,i.shared.pending=null}}while(1);u===null&&(c=d),i.baseState=c,i.firstBaseUpdate=l,i.lastBaseUpdate=u,a===null&&(i.shared.lanes=0),Ul|=o,e.lanes=o,e.memoizedState=d}}function Va(e,t){if(typeof e!=`function`)throw Error(a(191,e));e.call(t)}function Ha(e,t){var n=e.callbacks;if(n!==null)for(e.callbacks=null,e=0;ea?a:8;var o=T.T,s={};T.T=s,Os(e,!1,t,n);try{var c=i(),l=T.S;l!==null&&l(s,c),typeof c==`object`&&c&&typeof c.then==`function`?Ds(e,t,sa(c,r),du(e)):Ds(e,t,r,du(e))}catch(n){Ds(e,t,{then:function(){},status:`rejected`,reason:n},du())}finally{E.p=a,o!==null&&s.types!==null&&(o.types=s.types),T.T=o}}function _s(){}function vs(e,t,n,r){if(e.tag!==5)throw Error(a(476));var i=ys(e).queue;gs(e,i,t,de,n===null?_s:function(){return bs(e),n(r)})}function ys(e){var t=e.memoizedState;if(t!==null)return t;t={memoizedState:de,baseState:de,baseQueue:null,queue:{pending:null,lanes:0,dispatch:null,lastRenderedReducer:ko,lastRenderedState:de},next:null};var n={};return t.next={memoizedState:n,baseState:n,baseQueue:null,queue:{pending:null,lanes:0,dispatch:null,lastRenderedReducer:ko,lastRenderedState:n},next:null},e.memoizedState=t,e=e.alternate,e!==null&&(e.memoizedState=t),t}function bs(e){var t=ys(e);t.next===null&&(t=e.alternate.memoizedState),Ds(e,t.next.queue,{},du())}function xs(){return Ki(Qf)}function Ss(){return wo().memoizedState}function Cs(){return wo().memoizedState}function ws(e){for(var t=e.return;t!==null;){switch(t.tag){case 24:case 3:var n=du();e=Pa(n);var r=Fa(t,e,n);r!==null&&(pu(r,t,n),Ia(r,t,n)),t={cache:$i()},e.payload=t;return}t=t.return}}function Ts(e,t,n){var r=du();n={lane:r,revertLane:0,gesture:null,action:n,hasEagerState:!1,eagerState:null,next:null},ks(e)?As(t,n):(n=qr(e,t,n,r),n!==null&&(pu(n,e,r),js(n,t,r)))}function Es(e,t,n){Ds(e,t,n,du())}function Ds(e,t,n,r){var i={lane:r,revertLane:0,gesture:null,action:n,hasEagerState:!1,eagerState:null,next:null};if(ks(e))As(t,i);else{var a=e.alternate;if(e.lanes===0&&(a===null||a.lanes===0)&&(a=t.lastRenderedReducer,a!==null))try{var o=t.lastRenderedState,s=a(o,n);if(i.hasEagerState=!0,i.eagerState=s,pr(s,o))return Kr(e,t,i,0),Fl===null&&Gr(),!1}catch{}if(n=qr(e,t,i,r),n!==null)return pu(n,e,r),js(n,t,r),!0}return!1}function Os(e,t,n,r){if(r={lane:2,revertLane:ud(),gesture:null,action:r,hasEagerState:!1,eagerState:null,next:null},ks(e)){if(t)throw Error(a(479))}else t=qr(e,n,r,2),t!==null&&pu(t,e,2)}function ks(e){var t=e.alternate;return e===Y||t!==null&&t===Y}function As(e,t){so=oo=!0;var n=e.pending;n===null?t.next=t:(t.next=n.next,n.next=t),e.pending=t}function js(e,t,n){if(n&4194048){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,ot(e,n)}}var Ms={readContext:Ki,use:Do,useCallback:mo,useContext:mo,useEffect:mo,useImperativeHandle:mo,useLayoutEffect:mo,useInsertionEffect:mo,useMemo:mo,useReducer:mo,useRef:mo,useState:mo,useDebugValue:mo,useDeferredValue:mo,useTransition:mo,useSyncExternalStore:mo,useId:mo,useHostTransitionStatus:mo,useFormState:mo,useActionState:mo,useOptimistic:mo,useMemoCache:mo,useCacheRefresh:mo};Ms.useEffectEvent=mo;var Ns={readContext:Ki,use:Do,useCallback:function(e,t){return Co().memoizedState=[e,t===void 0?null:t],e},useContext:Ki,useEffect:rs,useImperativeHandle:function(e,t,n){n=n==null?null:n.concat([e]),ts(4194308,4,ls.bind(null,t,e),n)},useLayoutEffect:function(e,t){return ts(4194308,4,e,t)},useInsertionEffect:function(e,t){ts(4,2,e,t)},useMemo:function(e,t){var n=Co();t=t===void 0?null:t;var r=e();if(co){We(!0);try{e()}finally{We(!1)}}return n.memoizedState=[r,t],r},useReducer:function(e,t,n){var r=Co();if(n!==void 0){var i=n(t);if(co){We(!0);try{n(t)}finally{We(!1)}}}else i=t;return r.memoizedState=r.baseState=i,e={pending:null,lanes:0,dispatch:null,lastRenderedReducer:e,lastRenderedState:i},r.queue=e,e=e.dispatch=Ts.bind(null,Y,e),[r.memoizedState,e]},useRef:function(e){var t=Co();return e={current:e},t.memoizedState=e},useState:function(e){e=zo(e);var t=e.queue,n=Es.bind(null,Y,t);return t.dispatch=n,[e.memoizedState,n]},useDebugValue:ds,useDeferredValue:function(e,t){return ms(Co(),e,t)},useTransition:function(){var e=zo(!1);return e=gs.bind(null,Y,e.queue,!0,!1),Co().memoizedState=e,[!1,e]},useSyncExternalStore:function(e,t,n){var r=Y,i=Co();if(J){if(n===void 0)throw Error(a(407));n=n()}else{if(n=t(),Fl===null)throw Error(a(349));Q&127||Po(r,t,n)}i.memoizedState=n;var o={value:n,getSnapshot:t};return i.queue=o,rs(Io.bind(null,r,o,e),[e]),r.flags|=2048,$o(9,{destroy:void 0},Fo.bind(null,r,o,n,t),null),n},useId:function(){var e=Co(),t=Fl.identifierPrefix;if(J){var n=vi,r=_i;n=(r&~(1<<32-Ge(r)-1)).toString(32)+n,t=`_`+t+`R_`+n,n=lo++,0<\/script>`,o=o.removeChild(o.firstChild);break;case`select`:o=typeof r.is==`string`?s.createElement(`select`,{is:r.is}):s.createElement(`select`),r.multiple?o.multiple=!0:r.size&&(o.size=r.size);break;default:o=typeof r.is==`string`?s.createElement(i,{is:r.is}):s.createElement(i)}}o[pt]=t,o[M]=r;a:for(s=t.child;s!==null;){if(s.tag===5||s.tag===6)o.appendChild(s.stateNode);else if(s.tag!==4&&s.tag!==27&&s.child!==null){s.child.return=s,s=s.child;continue}if(s===t)break a;for(;s.sibling===null;){if(s.return===null||s.return===t)break a;s=s.return}s.sibling.return=s.return,s=s.sibling}t.stateNode=o;a:switch(Pd(o,i,r),i){case`button`:case`input`:case`select`:case`textarea`:r=!!r.autoFocus;break a;case`img`:r=!0;break a;default:r=!1}r&&Dc(t)}}return Mc(t),Oc(t,t.type,e===null?null:e.memoizedProps,t.pendingProps,n),null;case 6:if(e&&t.stateNode!=null)e.memoizedProps!==r&&Dc(t);else{if(typeof r!=`string`&&t.stateNode===null)throw Error(a(166));if(e=he.current,Mi(t)){if(e=t.stateNode,n=t.memoizedProps,r=null,i=wi,i!==null)switch(i.tag){case 27:case 5:r=i.memoizedProps}e[pt]=t,e=!!(e.nodeValue===n||r!==null&&!0===r.suppressHydrationWarning||jd(e.nodeValue,n)),e||ki(t,!0)}else e=Bd(e).createTextNode(r),e[pt]=t,t.stateNode=e}return Mc(t),null;case 31:if(n=t.memoizedState,e===null||e.memoizedState!==null){if(r=Mi(t),n!==null){if(e===null){if(!r)throw Error(a(318));if(e=t.memoizedState,e=e===null?null:e.dehydrated,!e)throw Error(a(557));e[pt]=t}else Ni(),!(t.flags&128)&&(t.memoizedState=null),t.flags|=4;Mc(t),e=!1}else n=Pi(),e!==null&&e.memoizedState!==null&&(e.memoizedState.hydrationErrors=n),e=!0;if(!e)return t.flags&256?(eo(t),t):(eo(t),null);if(t.flags&128)throw Error(a(558))}return Mc(t),null;case 13:if(r=t.memoizedState,e===null||e.memoizedState!==null&&e.memoizedState.dehydrated!==null){if(i=Mi(t),r!==null&&r.dehydrated!==null){if(e===null){if(!i)throw Error(a(318));if(i=t.memoizedState,i=i===null?null:i.dehydrated,!i)throw Error(a(317));i[pt]=t}else Ni(),!(t.flags&128)&&(t.memoizedState=null),t.flags|=4;Mc(t),i=!1}else i=Pi(),e!==null&&e.memoizedState!==null&&(e.memoizedState.hydrationErrors=i),i=!0;if(!i)return t.flags&256?(eo(t),t):(eo(t),null)}return eo(t),t.flags&128?(t.lanes=n,t):(n=r!==null,e=e!==null&&e.memoizedState!==null,n&&(r=t.child,i=null,r.alternate!==null&&r.alternate.memoizedState!==null&&r.alternate.memoizedState.cachePool!==null&&(i=r.alternate.memoizedState.cachePool.pool),o=null,r.memoizedState!==null&&r.memoizedState.cachePool!==null&&(o=r.memoizedState.cachePool.pool),o!==i&&(r.flags|=2048)),n!==e&&n&&(t.child.flags|=8192),Ac(t,t.updateQueue),Mc(t),null);case 4:return ve(),e===null&&xd(t.stateNode.containerInfo),Mc(t),null;case 10:return Bi(t.type),Mc(t),null;case 19:if(O(to),r=t.memoizedState,r===null)return Mc(t),null;if(i=(t.flags&128)!=0,o=r.rendering,o===null)if(i)jc(r,!1);else{if(Hl!==0||e!==null&&e.flags&128)for(e=t.child;e!==null;){if(o=no(e),o!==null){for(t.flags|=128,jc(r,!1),e=o.updateQueue,t.updateQueue=e,Ac(t,e),t.subtreeFlags=0,e=n,n=t.child;n!==null;)ni(n,e),n=n.sibling;return k(to,to.current&1|2),J&&yi(t,r.treeForkCount),t.child}e=e.sibling}r.tail!==null&&Ne()>$l&&(t.flags|=128,i=!0,jc(r,!1),t.lanes=4194304)}else{if(!i)if(e=no(o),e!==null){if(t.flags|=128,i=!0,e=e.updateQueue,t.updateQueue=e,Ac(t,e),jc(r,!0),r.tail===null&&r.tailMode===`hidden`&&!o.alternate&&!J)return Mc(t),null}else 2*Ne()-r.renderingStartTime>$l&&n!==536870912&&(t.flags|=128,i=!0,jc(r,!1),t.lanes=4194304);r.isBackwards?(o.sibling=t.child,t.child=o):(e=r.last,e===null?t.child=o:e.sibling=o,r.last=o)}return r.tail===null?(Mc(t),null):(e=r.tail,r.rendering=e,r.tail=e.sibling,r.renderingStartTime=Ne(),e.sibling=null,n=to.current,k(to,i?n&1|2:n&1),J&&yi(t,r.treeForkCount),e);case 22:case 23:return eo(t),qa(),r=t.memoizedState!==null,e===null?r&&(t.flags|=8192):e.memoizedState!==null!==r&&(t.flags|=8192),r?n&536870912&&!(t.flags&128)&&(Mc(t),t.subtreeFlags&6&&(t.flags|=8192)):Mc(t),n=t.updateQueue,n!==null&&Ac(t,n.retryQueue),n=null,e!==null&&e.memoizedState!==null&&e.memoizedState.cachePool!==null&&(n=e.memoizedState.cachePool.pool),r=null,t.memoizedState!==null&&t.memoizedState.cachePool!==null&&(r=t.memoizedState.cachePool.pool),r!==n&&(t.flags|=2048),e!==null&&O(la),null;case 24:return n=null,e!==null&&(n=e.memoizedState.cache),t.memoizedState.cache!==n&&(t.flags|=2048),Bi(Qi),Mc(t),null;case 25:return null;case 30:return null}throw Error(a(156,t.tag))}function Pc(e,t){switch(Si(t),t.tag){case 1:return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 3:return Bi(Qi),ve(),e=t.flags,e&65536&&!(e&128)?(t.flags=e&-65537|128,t):null;case 26:case 27:case 5:return be(t),null;case 31:if(t.memoizedState!==null){if(eo(t),t.alternate===null)throw Error(a(340));Ni()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 13:if(eo(t),e=t.memoizedState,e!==null&&e.dehydrated!==null){if(t.alternate===null)throw Error(a(340));Ni()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 19:return O(to),null;case 4:return ve(),null;case 10:return Bi(t.type),null;case 22:case 23:return eo(t),qa(),e!==null&&O(la),e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 24:return Bi(Qi),null;case 25:return null;default:return null}}function Fc(e,t){switch(Si(t),t.tag){case 3:Bi(Qi),ve();break;case 26:case 27:case 5:be(t);break;case 4:ve();break;case 31:t.memoizedState!==null&&eo(t);break;case 13:eo(t);break;case 19:O(to);break;case 10:Bi(t.type);break;case 22:case 23:eo(t),qa(),e!==null&&O(la);break;case 24:Bi(Qi)}}function Ic(e,t){try{var n=t.updateQueue,r=n===null?null:n.lastEffect;if(r!==null){var i=r.next;n=i;do{if((n.tag&e)===e){r=void 0;var a=n.create,o=n.inst;r=a(),o.destroy=r}n=n.next}while(n!==i)}}catch(e){Uu(t,t.return,e)}}function Lc(e,t,n){try{var r=t.updateQueue,i=r===null?null:r.lastEffect;if(i!==null){var a=i.next;r=a;do{if((r.tag&e)===e){var o=r.inst,s=o.destroy;if(s!==void 0){o.destroy=void 0,i=t;var c=n,l=s;try{l()}catch(e){Uu(i,c,e)}}}r=r.next}while(r!==a)}}catch(e){Uu(t,t.return,e)}}function Rc(e){var t=e.updateQueue;if(t!==null){var n=e.stateNode;try{Ha(t,n)}catch(t){Uu(e,e.return,t)}}}function zc(e,t,n){n.props=Bs(e.type,e.memoizedProps),n.state=e.memoizedState;try{n.componentWillUnmount()}catch(n){Uu(e,t,n)}}function Bc(e,t){try{var n=e.ref;if(n!==null){switch(e.tag){case 26:case 27:case 5:var r=e.stateNode;break;case 30:r=e.stateNode;break;default:r=e.stateNode}typeof n==`function`?e.refCleanup=n(r):n.current=r}}catch(n){Uu(e,t,n)}}function Vc(e,t){var n=e.ref,r=e.refCleanup;if(n!==null)if(typeof r==`function`)try{r()}catch(n){Uu(e,t,n)}finally{e.refCleanup=null,e=e.alternate,e!=null&&(e.refCleanup=null)}else if(typeof n==`function`)try{n(null)}catch(n){Uu(e,t,n)}else n.current=null}function Hc(e){var t=e.type,n=e.memoizedProps,r=e.stateNode;try{a:switch(t){case`button`:case`input`:case`select`:case`textarea`:n.autoFocus&&r.focus();break a;case`img`:n.src?r.src=n.src:n.srcSet&&(r.srcset=n.srcSet)}}catch(t){Uu(e,e.return,t)}}function Uc(e,t,n){try{var r=e.stateNode;Fd(r,e.type,n,t),r[M]=t}catch(t){Uu(e,e.return,t)}}function Wc(e){return e.tag===5||e.tag===3||e.tag===26||e.tag===27&&Zd(e.type)||e.tag===4}function Gc(e){a:for(;;){for(;e.sibling===null;){if(e.return===null||Wc(e.return))return null;e=e.return}for(e.sibling.return=e.return,e=e.sibling;e.tag!==5&&e.tag!==6&&e.tag!==18;){if(e.tag===27&&Zd(e.type)||e.flags&2||e.child===null||e.tag===4)continue a;e.child.return=e,e=e.child}if(!(e.flags&2))return e.stateNode}}function Kc(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?(n.nodeType===9?n.body:n.nodeName===`HTML`?n.ownerDocument.body:n).insertBefore(e,t):(t=n.nodeType===9?n.body:n.nodeName===`HTML`?n.ownerDocument.body:n,t.appendChild(e),n=n._reactRootContainer,n!=null||t.onclick!==null||(t.onclick=K));else if(r!==4&&(r===27&&Zd(e.type)&&(n=e.stateNode,t=null),e=e.child,e!==null))for(Kc(e,t,n),e=e.sibling;e!==null;)Kc(e,t,n),e=e.sibling}function qc(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?n.insertBefore(e,t):n.appendChild(e);else if(r!==4&&(r===27&&Zd(e.type)&&(n=e.stateNode),e=e.child,e!==null))for(qc(e,t,n),e=e.sibling;e!==null;)qc(e,t,n),e=e.sibling}function Jc(e){var t=e.stateNode,n=e.memoizedProps;try{for(var r=e.type,i=t.attributes;i.length;)t.removeAttributeNode(i[0]);Pd(t,r,n),t[pt]=e,t[M]=n}catch(t){Uu(e,e.return,t)}}var Yc=!1,Xc=!1,Zc=!1,Qc=typeof WeakSet==`function`?WeakSet:Set,$c=null;function el(e,t){if(e=e.containerInfo,Rd=sp,e=vr(e),yr(e)){if(`selectionStart`in e)var n={start:e.selectionStart,end:e.selectionEnd};else a:{n=(n=e.ownerDocument)&&n.defaultView||window;var r=n.getSelection&&n.getSelection();if(r&&r.rangeCount!==0){n=r.anchorNode;var i=r.anchorOffset,o=r.focusNode;r=r.focusOffset;try{n.nodeType,o.nodeType}catch{n=null;break a}var s=0,c=-1,l=-1,u=0,d=0,f=e,p=null;b:for(;;){for(var m;f!==n||i!==0&&f.nodeType!==3||(c=s+i),f!==o||r!==0&&f.nodeType!==3||(l=s+r),f.nodeType===3&&(s+=f.nodeValue.length),(m=f.firstChild)!==null;)p=f,f=m;for(;;){if(f===e)break b;if(p===n&&++u===i&&(c=s),p===o&&++d===r&&(l=s),(m=f.nextSibling)!==null)break;f=p,p=f.parentNode}f=m}n=c===-1||l===-1?null:{start:c,end:l}}else n=null}n||={start:0,end:0}}else n=null;for(zd={focusedElem:e,selectionRange:n},sp=!1,$c=t;$c!==null;)if(t=$c,e=t.child,t.subtreeFlags&1028&&e!==null)e.return=t,$c=e;else for(;$c!==null;){switch(t=$c,o=t.alternate,e=t.flags,t.tag){case 0:if(e&4&&(e=t.updateQueue,e=e===null?null:e.events,e!==null))for(n=0;n title`))),Pd(o,r,n),o[pt]=e,P(o),r=o;break a;case`link`:var s=Vf(`link`,`href`,i).get(r+(n.href||``));if(s){for(var c=0;cg&&(o=g,g=h,h=o);var _=gr(s,h),v=gr(s,g);if(_&&v&&(p.rangeCount!==1||p.anchorNode!==_.node||p.anchorOffset!==_.offset||p.focusNode!==v.node||p.focusOffset!==v.offset)){var y=d.createRange();y.setStart(_.node,_.offset),p.removeAllRanges(),h>g?(p.addRange(y),p.extend(v.node,v.offset)):(y.setEnd(v.node,v.offset),p.addRange(y))}}}}for(d=[],p=s;p=p.parentNode;)p.nodeType===1&&d.push({element:p,left:p.scrollLeft,top:p.scrollTop});for(typeof s.focus==`function`&&s.focus(),s=0;sn?32:n,T.T=null,n=su,su=null;var o=ru,s=au;if(nu=0,iu=ru=null,au=0,X&6)throw Error(a(331));var c=X;if(X|=4,Al(o.current),Sl(o,o.current,s,n),X=c,rd(0,!1),Ue&&typeof Ue.onPostCommitFiberRoot==`function`)try{Ue.onPostCommitFiberRoot(He,o)}catch{}return!0}finally{E.p=i,T.T=r,zu(e,t)}}function Hu(e,t,n){t=li(n,t),t=Ks(e.stateNode,t,2),e=Fa(e,t,2),e!==null&&(rt(e,2),nd(e))}function Uu(e,t,n){if(e.tag===3)Hu(e,e,n);else for(;t!==null;){if(t.tag===3){Hu(t,e,n);break}else if(t.tag===1){var r=t.stateNode;if(typeof t.type.getDerivedStateFromError==`function`||typeof r.componentDidCatch==`function`&&(tu===null||!tu.has(r))){e=li(n,e),n=qs(2),r=Fa(t,n,2),r!==null&&(Js(n,r,t,e),rt(r,2),nd(r));break}}t=t.return}}function Wu(e,t,n){var r=e.pingCache;if(r===null){r=e.pingCache=new Pl;var i=new Set;r.set(t,i)}else i=r.get(t),i===void 0&&(i=new Set,r.set(t,i));i.has(n)||(Bl=!0,i.add(n),e=Gu.bind(null,e,t,n),t.then(e,e))}function Gu(e,t,n){var r=e.pingCache;r!==null&&r.delete(t),e.pingedLanes|=e.suspendedLanes&n,e.warmLanes&=~n,Fl===e&&(Q&n)===n&&(Hl===4||Hl===3&&(Q&62914560)===Q&&300>Ne()-Zl?!(X&2)&&bu(e,0):Gl|=n,ql===Q&&(ql=0)),nd(e)}function Ku(e,t){t===0&&(t=nt()),e=Jr(e,t),e!==null&&(rt(e,t),nd(e))}function qu(e){var t=e.memoizedState,n=0;t!==null&&(n=t.retryLane),Ku(e,n)}function Ju(e,t){var n=0;switch(e.tag){case 31:case 13:var r=e.stateNode,i=e.memoizedState;i!==null&&(n=i.retryLane);break;case 19:r=e.stateNode;break;case 22:r=e.stateNode._retryCache;break;default:throw Error(a(314))}r!==null&&r.delete(t),Ku(e,n)}function Yu(e,t){return ke(e,t)}var Xu=null,Zu=null,Qu=!1,$u=!1,ed=!1,td=0;function nd(e){e!==Zu&&e.next===null&&(Zu===null?Xu=Zu=e:Zu=Zu.next=e),$u=!0,Qu||(Qu=!0,ld())}function rd(e,t){if(!ed&&$u){ed=!0;do for(var n=!1,r=Xu;r!==null;){if(!t)if(e!==0){var i=r.pendingLanes;if(i===0)var a=0;else{var o=r.suspendedLanes,s=r.pingedLanes;a=(1<<31-Ge(42|e)+1)-1,a&=i&~(o&~s),a=a&201326741?a&201326741|1:a?a|2:0}a!==0&&(n=!0,cd(r,a))}else a=Q,a=$e(r,r===Fl?a:0,r.cancelPendingCommit!==null||r.timeoutHandle!==-1),!(a&3)||et(r,a)||(n=!0,cd(r,a));r=r.next}while(n);ed=!1}}function id(){ad()}function ad(){$u=Qu=!1;var e=0;td!==0&&Gd()&&(e=td);for(var t=Ne(),n=null,r=Xu;r!==null;){var i=r.next,a=od(r,t);a===0?(r.next=null,n===null?Xu=i:n.next=i,i===null&&(Zu=n)):(n=r,(e!==0||a&3)&&($u=!0)),r=i}nu!==0&&nu!==5||rd(e,!1),td!==0&&(td=0)}function od(e,t){for(var n=e.suspendedLanes,r=e.pingedLanes,i=e.expirationTimes,a=e.pendingLanes&-62914561;0s)break;var u=c.transferSize,d=c.initiatorType;u&&Id(d)&&(c=c.responseEnd,o+=u*(c`u`?null:document;function xf(e,t,n){var r=bf;if(r&&typeof t==`string`&&t){var i=z(t);i=`link[rel="`+e+`"][href="`+i+`"]`,typeof n==`string`&&(i+=`[crossorigin="`+n+`"]`),hf.has(i)||(hf.add(i),e={rel:e,crossOrigin:n,href:t},r.querySelector(i)===null&&(t=r.createElement(`link`),Pd(t,`link`,e),P(t),r.head.appendChild(t)))}}function Sf(e){_f.D(e),xf(`dns-prefetch`,e,null)}function Cf(e,t){_f.C(e,t),xf(`preconnect`,e,t)}function wf(e,t,n){_f.L(e,t,n);var r=bf;if(r&&e&&t){var i=`link[rel="preload"][as="`+z(t)+`"]`;t===`image`&&n&&n.imageSrcSet?(i+=`[imagesrcset="`+z(n.imageSrcSet)+`"]`,typeof n.imageSizes==`string`&&(i+=`[imagesizes="`+z(n.imageSizes)+`"]`)):i+=`[href="`+z(e)+`"]`;var a=i;switch(t){case`style`:a=Af(e);break;case`script`:a=Pf(e)}mf.has(a)||(e=p({rel:`preload`,href:t===`image`&&n&&n.imageSrcSet?void 0:e,as:t},n),mf.set(a,e),r.querySelector(i)!==null||t===`style`&&r.querySelector(jf(a))||t===`script`&&r.querySelector(Ff(a))||(t=r.createElement(`link`),Pd(t,`link`,e),P(t),r.head.appendChild(t)))}}function Tf(e,t){_f.m(e,t);var n=bf;if(n&&e){var r=t&&typeof t.as==`string`?t.as:`script`,i=`link[rel="modulepreload"][as="`+z(r)+`"][href="`+z(e)+`"]`,a=i;switch(r){case`audioworklet`:case`paintworklet`:case`serviceworker`:case`sharedworker`:case`worker`:case`script`:a=Pf(e)}if(!mf.has(a)&&(e=p({rel:`modulepreload`,href:e},t),mf.set(a,e),n.querySelector(i)===null)){switch(r){case`audioworklet`:case`paintworklet`:case`serviceworker`:case`sharedworker`:case`worker`:case`script`:if(n.querySelector(Ff(a)))return}r=n.createElement(`link`),Pd(r,`link`,e),P(r),n.head.appendChild(r)}}}function Ef(e,t,n){_f.S(e,t,n);var r=bf;if(r&&e){var i=Ct(r).hoistableStyles,a=Af(e);t||=`default`;var o=i.get(a);if(!o){var s={loading:0,preload:null};if(o=r.querySelector(jf(a)))s.loading=5;else{e=p({rel:`stylesheet`,href:e,"data-precedence":t},n),(n=mf.get(a))&&Rf(e,n);var c=o=r.createElement(`link`);P(c),Pd(c,`link`,e),c._p=new Promise(function(e,t){c.onload=e,c.onerror=t}),c.addEventListener(`load`,function(){s.loading|=1}),c.addEventListener(`error`,function(){s.loading|=2}),s.loading|=4,Lf(o,t,r)}o={type:`stylesheet`,instance:o,count:1,state:s},i.set(a,o)}}}function Df(e,t){_f.X(e,t);var n=bf;if(n&&e){var r=Ct(n).hoistableScripts,i=Pf(e),a=r.get(i);a||(a=n.querySelector(Ff(i)),a||(e=p({src:e,async:!0},t),(t=mf.get(i))&&zf(e,t),a=n.createElement(`script`),P(a),Pd(a,`link`,e),n.head.appendChild(a)),a={type:`script`,instance:a,count:1,state:null},r.set(i,a))}}function Of(e,t){_f.M(e,t);var n=bf;if(n&&e){var r=Ct(n).hoistableScripts,i=Pf(e),a=r.get(i);a||(a=n.querySelector(Ff(i)),a||(e=p({src:e,async:!0,type:`module`},t),(t=mf.get(i))&&zf(e,t),a=n.createElement(`script`),P(a),Pd(a,`link`,e),n.head.appendChild(a)),a={type:`script`,instance:a,count:1,state:null},r.set(i,a))}}function kf(e,t,n,r){var i=(i=he.current)?gf(i):null;if(!i)throw Error(a(446));switch(e){case`meta`:case`title`:return null;case`style`:return typeof n.precedence==`string`&&typeof n.href==`string`?(t=Af(n.href),n=Ct(i).hoistableStyles,r=n.get(t),r||(r={type:`style`,instance:null,count:0,state:null},n.set(t,r)),r):{type:`void`,instance:null,count:0,state:null};case`link`:if(n.rel===`stylesheet`&&typeof n.href==`string`&&typeof n.precedence==`string`){e=Af(n.href);var o=Ct(i).hoistableStyles,s=o.get(e);if(s||(i=i.ownerDocument||i,s={type:`stylesheet`,instance:null,count:0,state:{loading:0,preload:null}},o.set(e,s),(o=i.querySelector(jf(e)))&&!o._p&&(s.instance=o,s.state.loading=5),mf.has(e)||(n={rel:`preload`,as:`style`,href:n.href,crossOrigin:n.crossOrigin,integrity:n.integrity,media:n.media,hrefLang:n.hrefLang,referrerPolicy:n.referrerPolicy},mf.set(e,n),o||Nf(i,e,n,s.state))),t&&r===null)throw Error(a(528,``));return s}if(t&&r!==null)throw Error(a(529,``));return null;case`script`:return t=n.async,n=n.src,typeof n==`string`&&t&&typeof t!=`function`&&typeof t!=`symbol`?(t=Pf(n),n=Ct(i).hoistableScripts,r=n.get(t),r||(r={type:`script`,instance:null,count:0,state:null},n.set(t,r)),r):{type:`void`,instance:null,count:0,state:null};default:throw Error(a(444,e))}}function Af(e){return`href="`+z(e)+`"`}function jf(e){return`link[rel="stylesheet"][`+e+`]`}function Mf(e){return p({},e,{"data-precedence":e.precedence,precedence:null})}function Nf(e,t,n,r){e.querySelector(`link[rel="preload"][as="style"][`+t+`]`)?r.loading=1:(t=e.createElement(`link`),r.preload=t,t.addEventListener(`load`,function(){return r.loading|=1}),t.addEventListener(`error`,function(){return r.loading|=2}),Pd(t,`link`,n),P(t),e.head.appendChild(t))}function Pf(e){return`[src="`+z(e)+`"]`}function Ff(e){return`script[async]`+e}function If(e,t,n){if(t.count++,t.instance===null)switch(t.type){case`style`:var r=e.querySelector(`style[data-href~="`+z(n.href)+`"]`);if(r)return t.instance=r,P(r),r;var i=p({},n,{"data-href":n.href,"data-precedence":n.precedence,href:null,precedence:null});return r=(e.ownerDocument||e).createElement(`style`),P(r),Pd(r,`style`,i),Lf(r,n.precedence,e),t.instance=r;case`stylesheet`:i=Af(n.href);var o=e.querySelector(jf(i));if(o)return t.state.loading|=4,t.instance=o,P(o),o;r=Mf(n),(i=mf.get(i))&&Rf(r,i),o=(e.ownerDocument||e).createElement(`link`),P(o);var s=o;return s._p=new Promise(function(e,t){s.onload=e,s.onerror=t}),Pd(o,`link`,r),t.state.loading|=4,Lf(o,n.precedence,e),t.instance=o;case`script`:return o=Pf(n.src),(i=e.querySelector(Ff(o)))?(t.instance=i,P(i),i):(r=n,(i=mf.get(o))&&(r=p({},n),zf(r,i)),e=e.ownerDocument||e,i=e.createElement(`script`),P(i),Pd(i,`link`,r),e.head.appendChild(i),t.instance=i);case`void`:return null;default:throw Error(a(443,t.type))}else t.type===`stylesheet`&&!(t.state.loading&4)&&(r=t.instance,t.state.loading|=4,Lf(r,n.precedence,e));return t.instance}function Lf(e,t,n){for(var r=n.querySelectorAll(`link[rel="stylesheet"][data-precedence],style[data-precedence]`),i=r.length?r[r.length-1]:null,a=i,o=0;o title`):null)}function Uf(e,t,n){if(n===1||t.itemProp!=null)return!1;switch(e){case`meta`:case`title`:return!0;case`style`:if(typeof t.precedence!=`string`||typeof t.href!=`string`||t.href===``)break;return!0;case`link`:if(typeof t.rel!=`string`||typeof t.href!=`string`||t.href===``||t.onLoad||t.onError)break;switch(t.rel){case`stylesheet`:return e=t.disabled,typeof t.precedence==`string`&&e==null;default:return!0}case`script`:if(t.async&&typeof t.async!=`function`&&typeof t.async!=`symbol`&&!t.onLoad&&!t.onError&&t.src&&typeof t.src==`string`)return!0}return!1}function Wf(e){return!(e.type===`stylesheet`&&!(e.state.loading&3))}function Gf(e,t,n,r){if(n.type===`stylesheet`&&(typeof r.media!=`string`||!1!==matchMedia(r.media).matches)&&!(n.state.loading&4)){if(n.instance===null){var i=Af(r.href),a=t.querySelector(jf(i));if(a){t=a._p,typeof t==`object`&&t&&typeof t.then==`function`&&(e.count++,e=Jf.bind(e),t.then(e,e)),n.state.loading|=4,n.instance=a,P(a);return}a=t.ownerDocument||t,r=Mf(r),(i=mf.get(i))&&Rf(r,i),a=a.createElement(`link`),P(a);var o=a;o._p=new Promise(function(e,t){o.onload=e,o.onerror=t}),Pd(a,`link`,r),n.instance=a}e.stylesheets===null&&(e.stylesheets=new Map),e.stylesheets.set(n,t),(t=n.state.preload)&&!(n.state.loading&3)&&(e.count++,n=Jf.bind(e),t.addEventListener(`load`,n),t.addEventListener(`error`,n))}}var Kf=0;function qf(e,t){return e.stylesheets&&e.count===0&&Xf(e,e.stylesheets),0Kf?50:800)+t);return e.unsuspend=n,function(){e.unsuspend=null,clearTimeout(r),clearTimeout(i)}}:null}function Jf(){if(this.count--,this.count===0&&(this.imgCount===0||!this.waitingForImages)){if(this.stylesheets)Xf(this,this.stylesheets);else if(this.unsuspend){var e=this.unsuspend;this.unsuspend=null,e()}}}var Yf=null;function Xf(e,t){e.stylesheets=null,e.unsuspend!==null&&(e.count++,Yf=new Map,t.forEach(Zf,e),Yf=null,Jf.call(e))}function Zf(e,t){if(!(t.state.loading&4)){var n=Yf.get(e);if(n)var r=n.get(null);else{n=new Map,Yf.set(e,n);for(var i=e.querySelectorAll(`link[data-precedence],style[data-precedence]`),a=0;a{function n(){if(!(typeof __REACT_DEVTOOLS_GLOBAL_HOOK__>`u`||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!=`function`))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(n)}catch(e){console.error(e)}}n(),t.exports=de()}))();function pe(e){var t,n,r=``;if(typeof e==`string`||typeof e==`number`)r+=e;else if(typeof e==`object`)if(Array.isArray(e)){var i=e.length;for(t=0;ttypeof e==`boolean`?`${e}`:e===0?`0`:e,k=D,me=(e,t)=>n=>{if(t?.variants==null)return k(e,n?.class,n?.className);let{variants:r,defaultVariants:i}=t,a=Object.keys(r).map(e=>{let t=n?.[e],a=i?.[e];if(t===null)return null;let o=O(t)||O(a);return r[e][o]}),o=n&&Object.entries(n).reduce((e,t)=>{let[n,r]=t;return r===void 0||(e[n]=r),e},{});return k(e,a,t?.compoundVariants?.reduce((e,t)=>{let{class:n,className:r,...a}=t;return Object.entries(a).every(e=>{let[t,n]=e;return Array.isArray(n)?n.includes({...i,...o}[t]):{...i,...o}[t]===n})?[...e,n,r]:e},[]),n?.class,n?.className)},A=(e,t)=>{let n=Array(e.length+t.length);for(let t=0;t({classGroupId:e,validator:t}),ge=(e=new Map,t=null,n)=>({nextPart:e,validators:t,classGroupId:n}),_e=`-`,ve=[],ye=`arbitrary..`,be=e=>{let t=Ce(e),{conflictingClassGroups:n,conflictingClassGroupModifiers:r}=e;return{getClassGroupId:e=>{if(e.startsWith(`[`)&&e.endsWith(`]`))return Se(e);let n=e.split(_e);return xe(n,+(n[0]===``&&n.length>1),t)},getConflictingClassGroupIds:(e,t)=>{if(t){let t=r[e],i=n[e];return t?i?A(i,t):t:i||ve}return n[e]||ve}}},xe=(e,t,n)=>{if(e.length-t===0)return n.classGroupId;let r=e[t],i=n.nextPart.get(r);if(i){let n=xe(e,t+1,i);if(n)return n}let a=n.validators;if(a===null)return;let o=t===0?e.join(_e):e.slice(t).join(_e),s=a.length;for(let e=0;ee.slice(1,-1).indexOf(`:`)===-1?void 0:(()=>{let t=e.slice(1,-1),n=t.indexOf(`:`),r=t.slice(0,n);return r?ye+r:void 0})(),Ce=e=>{let{theme:t,classGroups:n}=e;return we(n,t)},we=(e,t)=>{let n=ge();for(let r in e){let i=e[r];Te(i,n,r,t)}return n},Te=(e,t,n,r)=>{let i=e.length;for(let a=0;a{if(typeof e==`string`){De(e,t,n);return}if(typeof e==`function`){Oe(e,t,n,r);return}ke(e,t,n,r)},De=(e,t,n)=>{let r=e===``?t:Ae(t,e);r.classGroupId=n},Oe=(e,t,n,r)=>{if(je(e)){Te(e(r),t,n,r);return}t.validators===null&&(t.validators=[]),t.validators.push(he(n,e))},ke=(e,t,n,r)=>{let i=Object.entries(e),a=i.length;for(let e=0;e{let n=e,r=t.split(_e),i=r.length;for(let e=0;e`isThemeGetter`in e&&e.isThemeGetter===!0,Me=e=>{if(e<1)return{get:()=>void 0,set:()=>{}};let t=0,n=Object.create(null),r=Object.create(null),i=(i,a)=>{n[i]=a,t++,t>e&&(t=0,r=n,n=Object.create(null))};return{get(e){let t=n[e];if(t!==void 0)return t;if((t=r[e])!==void 0)return i(e,t),t},set(e,t){e in n?n[e]=t:i(e,t)}}},Ne=`!`,Pe=`:`,Fe=[],Ie=(e,t,n,r,i)=>({modifiers:e,hasImportantModifier:t,baseClassName:n,maybePostfixModifierPosition:r,isExternal:i}),Le=e=>{let{prefix:t,experimentalParseClassName:n}=e,r=e=>{let t=[],n=0,r=0,i=0,a,o=e.length;for(let s=0;si?a-i:void 0;return Ie(t,l,c,u)};if(t){let e=t+Pe,n=r;r=t=>t.startsWith(e)?n(t.slice(e.length)):Ie(Fe,!1,t,void 0,!0)}if(n){let e=r;r=t=>n({className:t,parseClassName:e})}return r},Re=e=>{let t=new Map;return e.orderSensitiveModifiers.forEach((e,n)=>{t.set(e,1e6+n)}),e=>{let n=[],r=[];for(let i=0;i0&&(r.sort(),n.push(...r),r=[]),n.push(a)):r.push(a)}return r.length>0&&(r.sort(),n.push(...r)),n}},ze=e=>({cache:Me(e.cacheSize),parseClassName:Le(e),sortModifiers:Re(e),postfixLookupClassGroupIds:Be(e),...be(e)}),Be=e=>{let t=Object.create(null),n=e.postfixLookupClassGroups;if(n)for(let e=0;e{let{parseClassName:n,getClassGroupId:r,getConflictingClassGroupIds:i,sortModifiers:a,postfixLookupClassGroupIds:o}=t,s=[],c=e.trim().split(Ve),l=``;for(let e=c.length-1;e>=0;--e){let t=c[e],{isExternal:u,modifiers:d,hasImportantModifier:f,baseClassName:p,maybePostfixModifierPosition:m}=n(t);if(u){l=t+(l.length>0?` `+l:l);continue}let h=!!m,g;if(h){g=r(p.substring(0,m));let e=g&&o[g]?r(p):void 0;e&&e!==g&&(g=e,h=!1)}else g=r(p);if(!g){if(!h){l=t+(l.length>0?` `+l:l);continue}if(g=r(p),!g){l=t+(l.length>0?` `+l:l);continue}h=!1}let _=d.length===0?``:d.length===1?d[0]:a(d).join(`:`),v=f?_+Ne:_,y=v+g;if(s.indexOf(y)>-1)continue;s.push(y);let b=i(g,h);for(let e=0;e0?` `+l:l)}return l},Ue=(...e)=>{let t=0,n,r,i=``;for(;t{if(typeof e==`string`)return e;let t,n=``;for(let r=0;r{let n,r,i,a,o=o=>(n=ze(t.reduce((e,t)=>t(e),e())),r=n.cache.get,i=n.cache.set,a=s,s(o)),s=e=>{let t=r(e);if(t)return t;let a=He(e,n);return i(e,a),a};return a=o,(...e)=>a(Ue(...e))},Ke=[],qe=e=>{let t=t=>t[e]||Ke;return t.isThemeGetter=!0,t},Je=/^\[(?:(\w[\w-]*):)?(.+)\]$/i,Ye=/^\((?:(\w[\w-]*):)?(.+)\)$/i,Xe=/^\d+(?:\.\d+)?\/\d+(?:\.\d+)?$/,Ze=/^(\d+(\.\d+)?)?(xs|sm|md|lg|xl)$/,Qe=/\d+(%|px|r?em|[sdl]?v([hwib]|min|max)|pt|pc|in|cm|mm|cap|ch|ex|r?lh|cq(w|h|i|b|min|max))|\b(calc|min|max|clamp)\(.+\)|^0$/,$e=/^(rgba?|hsla?|hwb|(ok)?(lab|lch)|color-mix)\(.+\)$/,et=/^(inset_)?-?((\d+)?\.?(\d+)[a-z]+|0)_-?((\d+)?\.?(\d+)[a-z]+|0)/,tt=/^(url|image|image-set|cross-fade|element|(repeating-)?(linear|radial|conic)-gradient)\(.+\)$/,nt=e=>Xe.test(e),j=e=>!!e&&!Number.isNaN(Number(e)),rt=e=>!!e&&Number.isInteger(Number(e)),it=e=>e.endsWith(`%`)&&j(e.slice(0,-1)),at=e=>Ze.test(e),ot=()=>!0,st=e=>Qe.test(e)&&!$e.test(e),ct=()=>!1,lt=e=>et.test(e),ut=e=>tt.test(e),dt=e=>!M(e)&&!N(e),ft=e=>e.startsWith(`@container`)&&(e[10]===`/`&&e[11]!==void 0||e[11]===`s`&&e[16]!==void 0&&e.startsWith(`-size/`,10)||e[11]===`n`&&e[18]!==void 0&&e.startsWith(`-normal/`,10)),pt=e=>Tt(e,kt,ct),M=e=>Je.test(e),mt=e=>Tt(e,At,st),ht=e=>Tt(e,jt,j),gt=e=>Tt(e,Nt,ot),_t=e=>Tt(e,Mt,ct),vt=e=>Tt(e,Dt,ct),yt=e=>Tt(e,Ot,ut),bt=e=>Tt(e,Pt,lt),N=e=>Ye.test(e),xt=e=>Et(e,At),St=e=>Et(e,Mt),Ct=e=>Et(e,Dt),P=e=>Et(e,kt),F=e=>Et(e,Ot),I=e=>Et(e,Pt,!0),wt=e=>Et(e,Nt,!0),Tt=(e,t,n)=>{let r=Je.exec(e);return r?r[1]?t(r[1]):n(r[2]):!1},Et=(e,t,n=!1)=>{let r=Ye.exec(e);return r?r[1]?t(r[1]):n:!1},Dt=e=>e===`position`||e===`percentage`,Ot=e=>e===`image`||e===`url`,kt=e=>e===`length`||e===`size`||e===`bg-size`,At=e=>e===`length`,jt=e=>e===`number`,Mt=e=>e===`family-name`,Nt=e=>e===`number`||e===`weight`,Pt=e=>e===`shadow`,Ft=Ge(()=>{let e=qe(`color`),t=qe(`font`),n=qe(`text`),r=qe(`font-weight`),i=qe(`tracking`),a=qe(`leading`),o=qe(`breakpoint`),s=qe(`container`),c=qe(`spacing`),l=qe(`radius`),u=qe(`shadow`),d=qe(`inset-shadow`),f=qe(`text-shadow`),p=qe(`drop-shadow`),m=qe(`blur`),h=qe(`perspective`),g=qe(`aspect`),_=qe(`ease`),v=qe(`animate`),y=()=>[`auto`,`avoid`,`all`,`avoid-page`,`page`,`left`,`right`,`column`],b=()=>[`center`,`top`,`bottom`,`left`,`right`,`top-left`,`left-top`,`top-right`,`right-top`,`bottom-right`,`right-bottom`,`bottom-left`,`left-bottom`],x=()=>[...b(),N,M],ee=()=>[`auto`,`hidden`,`clip`,`visible`,`scroll`],te=()=>[`auto`,`contain`,`none`],S=()=>[N,M,c],C=()=>[nt,`full`,`auto`,...S()],ne=()=>[rt,`none`,`subgrid`,N,M],re=()=>[`auto`,{span:[`full`,rt,N,M]},rt,N,M],ie=()=>[rt,`auto`,N,M],ae=()=>[`auto`,`min`,`max`,`fr`,N,M],oe=()=>[`start`,`end`,`center`,`between`,`around`,`evenly`,`stretch`,`baseline`,`center-safe`,`end-safe`],se=()=>[`start`,`end`,`center`,`stretch`,`center-safe`,`end-safe`],ce=()=>[`auto`,...S()],w=()=>[nt,`auto`,`full`,`dvw`,`dvh`,`lvw`,`lvh`,`svw`,`svh`,`min`,`max`,`fit`,...S()],T=()=>[nt,`screen`,`full`,`dvw`,`lvw`,`svw`,`min`,`max`,`fit`,...S()],le=()=>[nt,`screen`,`full`,`lh`,`dvh`,`lvh`,`svh`,`min`,`max`,`fit`,...S()],E=()=>[e,N,M],ue=()=>[...b(),Ct,vt,{position:[N,M]}],de=()=>[`no-repeat`,{repeat:[``,`x`,`y`,`space`,`round`]}],fe=()=>[`auto`,`cover`,`contain`,P,pt,{size:[N,M]}],pe=()=>[it,xt,mt],D=()=>[``,`none`,`full`,l,N,M],O=()=>[``,j,xt,mt],k=()=>[`solid`,`dashed`,`dotted`,`double`],me=()=>[`normal`,`multiply`,`screen`,`overlay`,`darken`,`lighten`,`color-dodge`,`color-burn`,`hard-light`,`soft-light`,`difference`,`exclusion`,`hue`,`saturation`,`color`,`luminosity`],A=()=>[j,it,Ct,vt],he=()=>[``,`none`,m,N,M],ge=()=>[`none`,j,N,M],_e=()=>[`none`,j,N,M],ve=()=>[j,N,M],ye=()=>[nt,`full`,...S()];return{cacheSize:500,theme:{animate:[`spin`,`ping`,`pulse`,`bounce`],aspect:[`video`],blur:[at],breakpoint:[at],color:[ot],container:[at],"drop-shadow":[at],ease:[`in`,`out`,`in-out`],font:[dt],"font-weight":[`thin`,`extralight`,`light`,`normal`,`medium`,`semibold`,`bold`,`extrabold`,`black`],"inset-shadow":[at],leading:[`none`,`tight`,`snug`,`normal`,`relaxed`,`loose`],perspective:[`dramatic`,`near`,`normal`,`midrange`,`distant`,`none`],radius:[at],shadow:[at],spacing:[`px`,j],text:[at],"text-shadow":[at],tracking:[`tighter`,`tight`,`normal`,`wide`,`wider`,`widest`]},classGroups:{aspect:[{aspect:[`auto`,`square`,nt,M,N,g]}],container:[`container`],"container-type":[{"@container":[``,`normal`,`size`,N,M]}],"container-named":[ft],columns:[{columns:[j,M,N,s]}],"break-after":[{"break-after":y()}],"break-before":[{"break-before":y()}],"break-inside":[{"break-inside":[`auto`,`avoid`,`avoid-page`,`avoid-column`]}],"box-decoration":[{"box-decoration":[`slice`,`clone`]}],box:[{box:[`border`,`content`]}],display:[`block`,`inline-block`,`inline`,`flex`,`inline-flex`,`table`,`inline-table`,`table-caption`,`table-cell`,`table-column`,`table-column-group`,`table-footer-group`,`table-header-group`,`table-row-group`,`table-row`,`flow-root`,`grid`,`inline-grid`,`contents`,`list-item`,`hidden`],sr:[`sr-only`,`not-sr-only`],float:[{float:[`right`,`left`,`none`,`start`,`end`]}],clear:[{clear:[`left`,`right`,`both`,`none`,`start`,`end`]}],isolation:[`isolate`,`isolation-auto`],"object-fit":[{object:[`contain`,`cover`,`fill`,`none`,`scale-down`]}],"object-position":[{object:x()}],overflow:[{overflow:ee()}],"overflow-x":[{"overflow-x":ee()}],"overflow-y":[{"overflow-y":ee()}],overscroll:[{overscroll:te()}],"overscroll-x":[{"overscroll-x":te()}],"overscroll-y":[{"overscroll-y":te()}],position:[`static`,`fixed`,`absolute`,`relative`,`sticky`],inset:[{inset:C()}],"inset-x":[{"inset-x":C()}],"inset-y":[{"inset-y":C()}],start:[{"inset-s":C(),start:C()}],end:[{"inset-e":C(),end:C()}],"inset-bs":[{"inset-bs":C()}],"inset-be":[{"inset-be":C()}],top:[{top:C()}],right:[{right:C()}],bottom:[{bottom:C()}],left:[{left:C()}],visibility:[`visible`,`invisible`,`collapse`],z:[{z:[rt,`auto`,N,M]}],basis:[{basis:[nt,`full`,`auto`,s,...S()]}],"flex-direction":[{flex:[`row`,`row-reverse`,`col`,`col-reverse`]}],"flex-wrap":[{flex:[`nowrap`,`wrap`,`wrap-reverse`]}],flex:[{flex:[j,nt,`auto`,`initial`,`none`,M]}],grow:[{grow:[``,j,N,M]}],shrink:[{shrink:[``,j,N,M]}],order:[{order:[rt,`first`,`last`,`none`,N,M]}],"grid-cols":[{"grid-cols":ne()}],"col-start-end":[{col:re()}],"col-start":[{"col-start":ie()}],"col-end":[{"col-end":ie()}],"grid-rows":[{"grid-rows":ne()}],"row-start-end":[{row:re()}],"row-start":[{"row-start":ie()}],"row-end":[{"row-end":ie()}],"grid-flow":[{"grid-flow":[`row`,`col`,`dense`,`row-dense`,`col-dense`]}],"auto-cols":[{"auto-cols":ae()}],"auto-rows":[{"auto-rows":ae()}],gap:[{gap:S()}],"gap-x":[{"gap-x":S()}],"gap-y":[{"gap-y":S()}],"justify-content":[{justify:[...oe(),`normal`]}],"justify-items":[{"justify-items":[...se(),`normal`]}],"justify-self":[{"justify-self":[`auto`,...se()]}],"align-content":[{content:[`normal`,...oe()]}],"align-items":[{items:[...se(),{baseline:[``,`last`]}]}],"align-self":[{self:[`auto`,...se(),{baseline:[``,`last`]}]}],"place-content":[{"place-content":oe()}],"place-items":[{"place-items":[...se(),`baseline`]}],"place-self":[{"place-self":[`auto`,...se()]}],p:[{p:S()}],px:[{px:S()}],py:[{py:S()}],ps:[{ps:S()}],pe:[{pe:S()}],pbs:[{pbs:S()}],pbe:[{pbe:S()}],pt:[{pt:S()}],pr:[{pr:S()}],pb:[{pb:S()}],pl:[{pl:S()}],m:[{m:ce()}],mx:[{mx:ce()}],my:[{my:ce()}],ms:[{ms:ce()}],me:[{me:ce()}],mbs:[{mbs:ce()}],mbe:[{mbe:ce()}],mt:[{mt:ce()}],mr:[{mr:ce()}],mb:[{mb:ce()}],ml:[{ml:ce()}],"space-x":[{"space-x":S()}],"space-x-reverse":[`space-x-reverse`],"space-y":[{"space-y":S()}],"space-y-reverse":[`space-y-reverse`],size:[{size:w()}],"inline-size":[{inline:[`auto`,...T()]}],"min-inline-size":[{"min-inline":[`auto`,...T()]}],"max-inline-size":[{"max-inline":[`none`,...T()]}],"block-size":[{block:[`auto`,...le()]}],"min-block-size":[{"min-block":[`auto`,...le()]}],"max-block-size":[{"max-block":[`none`,...le()]}],w:[{w:[s,`screen`,...w()]}],"min-w":[{"min-w":[s,`screen`,`none`,...w()]}],"max-w":[{"max-w":[s,`screen`,`none`,`prose`,{screen:[o]},...w()]}],h:[{h:[`screen`,`lh`,...w()]}],"min-h":[{"min-h":[`screen`,`lh`,`none`,...w()]}],"max-h":[{"max-h":[`screen`,`lh`,...w()]}],"font-size":[{text:[`base`,n,xt,mt]}],"font-smoothing":[`antialiased`,`subpixel-antialiased`],"font-style":[`italic`,`not-italic`],"font-weight":[{font:[r,wt,gt]}],"font-stretch":[{"font-stretch":[`ultra-condensed`,`extra-condensed`,`condensed`,`semi-condensed`,`normal`,`semi-expanded`,`expanded`,`extra-expanded`,`ultra-expanded`,it,M]}],"font-family":[{font:[St,_t,t]}],"font-features":[{"font-features":[M]}],"fvn-normal":[`normal-nums`],"fvn-ordinal":[`ordinal`],"fvn-slashed-zero":[`slashed-zero`],"fvn-figure":[`lining-nums`,`oldstyle-nums`],"fvn-spacing":[`proportional-nums`,`tabular-nums`],"fvn-fraction":[`diagonal-fractions`,`stacked-fractions`],tracking:[{tracking:[i,N,M]}],"line-clamp":[{"line-clamp":[j,`none`,N,ht]}],leading:[{leading:[a,...S()]}],"list-image":[{"list-image":[`none`,N,M]}],"list-style-position":[{list:[`inside`,`outside`]}],"list-style-type":[{list:[`disc`,`decimal`,`none`,N,M]}],"text-alignment":[{text:[`left`,`center`,`right`,`justify`,`start`,`end`]}],"placeholder-color":[{placeholder:E()}],"text-color":[{text:E()}],"text-decoration":[`underline`,`overline`,`line-through`,`no-underline`],"text-decoration-style":[{decoration:[...k(),`wavy`]}],"text-decoration-thickness":[{decoration:[j,`from-font`,`auto`,N,mt]}],"text-decoration-color":[{decoration:E()}],"underline-offset":[{"underline-offset":[j,`auto`,N,M]}],"text-transform":[`uppercase`,`lowercase`,`capitalize`,`normal-case`],"text-overflow":[`truncate`,`text-ellipsis`,`text-clip`],"text-wrap":[{text:[`wrap`,`nowrap`,`balance`,`pretty`]}],indent:[{indent:S()}],"tab-size":[{tab:[rt,N,M]}],"vertical-align":[{align:[`baseline`,`top`,`middle`,`bottom`,`text-top`,`text-bottom`,`sub`,`super`,N,M]}],whitespace:[{whitespace:[`normal`,`nowrap`,`pre`,`pre-line`,`pre-wrap`,`break-spaces`]}],break:[{break:[`normal`,`words`,`all`,`keep`]}],wrap:[{wrap:[`break-word`,`anywhere`,`normal`]}],hyphens:[{hyphens:[`none`,`manual`,`auto`]}],content:[{content:[`none`,N,M]}],"bg-attachment":[{bg:[`fixed`,`local`,`scroll`]}],"bg-clip":[{"bg-clip":[`border`,`padding`,`content`,`text`]}],"bg-origin":[{"bg-origin":[`border`,`padding`,`content`]}],"bg-position":[{bg:ue()}],"bg-repeat":[{bg:de()}],"bg-size":[{bg:fe()}],"bg-image":[{bg:[`none`,{linear:[{to:[`t`,`tr`,`r`,`br`,`b`,`bl`,`l`,`tl`]},rt,N,M],radial:[``,N,M],conic:[rt,N,M]},F,yt]}],"bg-color":[{bg:E()}],"gradient-from-pos":[{from:pe()}],"gradient-via-pos":[{via:pe()}],"gradient-to-pos":[{to:pe()}],"gradient-from":[{from:E()}],"gradient-via":[{via:E()}],"gradient-to":[{to:E()}],rounded:[{rounded:D()}],"rounded-s":[{"rounded-s":D()}],"rounded-e":[{"rounded-e":D()}],"rounded-t":[{"rounded-t":D()}],"rounded-r":[{"rounded-r":D()}],"rounded-b":[{"rounded-b":D()}],"rounded-l":[{"rounded-l":D()}],"rounded-ss":[{"rounded-ss":D()}],"rounded-se":[{"rounded-se":D()}],"rounded-ee":[{"rounded-ee":D()}],"rounded-es":[{"rounded-es":D()}],"rounded-tl":[{"rounded-tl":D()}],"rounded-tr":[{"rounded-tr":D()}],"rounded-br":[{"rounded-br":D()}],"rounded-bl":[{"rounded-bl":D()}],"border-w":[{border:O()}],"border-w-x":[{"border-x":O()}],"border-w-y":[{"border-y":O()}],"border-w-s":[{"border-s":O()}],"border-w-e":[{"border-e":O()}],"border-w-bs":[{"border-bs":O()}],"border-w-be":[{"border-be":O()}],"border-w-t":[{"border-t":O()}],"border-w-r":[{"border-r":O()}],"border-w-b":[{"border-b":O()}],"border-w-l":[{"border-l":O()}],"divide-x":[{"divide-x":O()}],"divide-x-reverse":[`divide-x-reverse`],"divide-y":[{"divide-y":O()}],"divide-y-reverse":[`divide-y-reverse`],"border-style":[{border:[...k(),`hidden`,`none`]}],"divide-style":[{divide:[...k(),`hidden`,`none`]}],"border-color":[{border:E()}],"border-color-x":[{"border-x":E()}],"border-color-y":[{"border-y":E()}],"border-color-s":[{"border-s":E()}],"border-color-e":[{"border-e":E()}],"border-color-bs":[{"border-bs":E()}],"border-color-be":[{"border-be":E()}],"border-color-t":[{"border-t":E()}],"border-color-r":[{"border-r":E()}],"border-color-b":[{"border-b":E()}],"border-color-l":[{"border-l":E()}],"divide-color":[{divide:E()}],"outline-style":[{outline:[...k(),`none`,`hidden`]}],"outline-offset":[{"outline-offset":[j,N,M]}],"outline-w":[{outline:[``,j,xt,mt]}],"outline-color":[{outline:E()}],shadow:[{shadow:[``,`none`,u,I,bt]}],"shadow-color":[{shadow:E()}],"inset-shadow":[{"inset-shadow":[`none`,d,I,bt]}],"inset-shadow-color":[{"inset-shadow":E()}],"ring-w":[{ring:O()}],"ring-w-inset":[`ring-inset`],"ring-color":[{ring:E()}],"ring-offset-w":[{"ring-offset":[j,mt]}],"ring-offset-color":[{"ring-offset":E()}],"inset-ring-w":[{"inset-ring":O()}],"inset-ring-color":[{"inset-ring":E()}],"text-shadow":[{"text-shadow":[`none`,f,I,bt]}],"text-shadow-color":[{"text-shadow":E()}],opacity:[{opacity:[j,N,M]}],"mix-blend":[{"mix-blend":[...me(),`plus-darker`,`plus-lighter`]}],"bg-blend":[{"bg-blend":me()}],"mask-clip":[{"mask-clip":[`border`,`padding`,`content`,`fill`,`stroke`,`view`]},`mask-no-clip`],"mask-composite":[{mask:[`add`,`subtract`,`intersect`,`exclude`]}],"mask-image-linear-pos":[{"mask-linear":[j]}],"mask-image-linear-from-pos":[{"mask-linear-from":A()}],"mask-image-linear-to-pos":[{"mask-linear-to":A()}],"mask-image-linear-from-color":[{"mask-linear-from":E()}],"mask-image-linear-to-color":[{"mask-linear-to":E()}],"mask-image-t-from-pos":[{"mask-t-from":A()}],"mask-image-t-to-pos":[{"mask-t-to":A()}],"mask-image-t-from-color":[{"mask-t-from":E()}],"mask-image-t-to-color":[{"mask-t-to":E()}],"mask-image-r-from-pos":[{"mask-r-from":A()}],"mask-image-r-to-pos":[{"mask-r-to":A()}],"mask-image-r-from-color":[{"mask-r-from":E()}],"mask-image-r-to-color":[{"mask-r-to":E()}],"mask-image-b-from-pos":[{"mask-b-from":A()}],"mask-image-b-to-pos":[{"mask-b-to":A()}],"mask-image-b-from-color":[{"mask-b-from":E()}],"mask-image-b-to-color":[{"mask-b-to":E()}],"mask-image-l-from-pos":[{"mask-l-from":A()}],"mask-image-l-to-pos":[{"mask-l-to":A()}],"mask-image-l-from-color":[{"mask-l-from":E()}],"mask-image-l-to-color":[{"mask-l-to":E()}],"mask-image-x-from-pos":[{"mask-x-from":A()}],"mask-image-x-to-pos":[{"mask-x-to":A()}],"mask-image-x-from-color":[{"mask-x-from":E()}],"mask-image-x-to-color":[{"mask-x-to":E()}],"mask-image-y-from-pos":[{"mask-y-from":A()}],"mask-image-y-to-pos":[{"mask-y-to":A()}],"mask-image-y-from-color":[{"mask-y-from":E()}],"mask-image-y-to-color":[{"mask-y-to":E()}],"mask-image-radial":[{"mask-radial":[N,M]}],"mask-image-radial-from-pos":[{"mask-radial-from":A()}],"mask-image-radial-to-pos":[{"mask-radial-to":A()}],"mask-image-radial-from-color":[{"mask-radial-from":E()}],"mask-image-radial-to-color":[{"mask-radial-to":E()}],"mask-image-radial-shape":[{"mask-radial":[`circle`,`ellipse`]}],"mask-image-radial-size":[{"mask-radial":[{closest:[`side`,`corner`],farthest:[`side`,`corner`]}]}],"mask-image-radial-pos":[{"mask-radial-at":b()}],"mask-image-conic-pos":[{"mask-conic":[j]}],"mask-image-conic-from-pos":[{"mask-conic-from":A()}],"mask-image-conic-to-pos":[{"mask-conic-to":A()}],"mask-image-conic-from-color":[{"mask-conic-from":E()}],"mask-image-conic-to-color":[{"mask-conic-to":E()}],"mask-mode":[{mask:[`alpha`,`luminance`,`match`]}],"mask-origin":[{"mask-origin":[`border`,`padding`,`content`,`fill`,`stroke`,`view`]}],"mask-position":[{mask:ue()}],"mask-repeat":[{mask:de()}],"mask-size":[{mask:fe()}],"mask-type":[{"mask-type":[`alpha`,`luminance`]}],"mask-image":[{mask:[`none`,N,M]}],filter:[{filter:[``,`none`,N,M]}],blur:[{blur:he()}],brightness:[{brightness:[j,N,M]}],contrast:[{contrast:[j,N,M]}],"drop-shadow":[{"drop-shadow":[``,`none`,p,I,bt]}],"drop-shadow-color":[{"drop-shadow":E()}],grayscale:[{grayscale:[``,j,N,M]}],"hue-rotate":[{"hue-rotate":[j,N,M]}],invert:[{invert:[``,j,N,M]}],saturate:[{saturate:[j,N,M]}],sepia:[{sepia:[``,j,N,M]}],"backdrop-filter":[{"backdrop-filter":[``,`none`,N,M]}],"backdrop-blur":[{"backdrop-blur":he()}],"backdrop-brightness":[{"backdrop-brightness":[j,N,M]}],"backdrop-contrast":[{"backdrop-contrast":[j,N,M]}],"backdrop-grayscale":[{"backdrop-grayscale":[``,j,N,M]}],"backdrop-hue-rotate":[{"backdrop-hue-rotate":[j,N,M]}],"backdrop-invert":[{"backdrop-invert":[``,j,N,M]}],"backdrop-opacity":[{"backdrop-opacity":[j,N,M]}],"backdrop-saturate":[{"backdrop-saturate":[j,N,M]}],"backdrop-sepia":[{"backdrop-sepia":[``,j,N,M]}],"border-collapse":[{border:[`collapse`,`separate`]}],"border-spacing":[{"border-spacing":S()}],"border-spacing-x":[{"border-spacing-x":S()}],"border-spacing-y":[{"border-spacing-y":S()}],"table-layout":[{table:[`auto`,`fixed`]}],caption:[{caption:[`top`,`bottom`]}],transition:[{transition:[``,`all`,`colors`,`opacity`,`shadow`,`transform`,`none`,N,M]}],"transition-behavior":[{transition:[`normal`,`discrete`]}],duration:[{duration:[j,`initial`,N,M]}],ease:[{ease:[`linear`,`initial`,_,N,M]}],delay:[{delay:[j,N,M]}],animate:[{animate:[`none`,v,N,M]}],backface:[{backface:[`hidden`,`visible`]}],perspective:[{perspective:[h,N,M]}],"perspective-origin":[{"perspective-origin":x()}],rotate:[{rotate:ge()}],"rotate-x":[{"rotate-x":ge()}],"rotate-y":[{"rotate-y":ge()}],"rotate-z":[{"rotate-z":ge()}],scale:[{scale:_e()}],"scale-x":[{"scale-x":_e()}],"scale-y":[{"scale-y":_e()}],"scale-z":[{"scale-z":_e()}],"scale-3d":[`scale-3d`],skew:[{skew:ve()}],"skew-x":[{"skew-x":ve()}],"skew-y":[{"skew-y":ve()}],transform:[{transform:[N,M,``,`none`,`gpu`,`cpu`]}],"transform-origin":[{origin:x()}],"transform-style":[{transform:[`3d`,`flat`]}],translate:[{translate:ye()}],"translate-x":[{"translate-x":ye()}],"translate-y":[{"translate-y":ye()}],"translate-z":[{"translate-z":ye()}],"translate-none":[`translate-none`],zoom:[{zoom:[rt,N,M]}],accent:[{accent:E()}],appearance:[{appearance:[`none`,`auto`]}],"caret-color":[{caret:E()}],"color-scheme":[{scheme:[`normal`,`dark`,`light`,`light-dark`,`only-dark`,`only-light`]}],cursor:[{cursor:[`auto`,`default`,`pointer`,`wait`,`text`,`move`,`help`,`not-allowed`,`none`,`context-menu`,`progress`,`cell`,`crosshair`,`vertical-text`,`alias`,`copy`,`no-drop`,`grab`,`grabbing`,`all-scroll`,`col-resize`,`row-resize`,`n-resize`,`e-resize`,`s-resize`,`w-resize`,`ne-resize`,`nw-resize`,`se-resize`,`sw-resize`,`ew-resize`,`ns-resize`,`nesw-resize`,`nwse-resize`,`zoom-in`,`zoom-out`,N,M]}],"field-sizing":[{"field-sizing":[`fixed`,`content`]}],"pointer-events":[{"pointer-events":[`auto`,`none`]}],resize:[{resize:[`none`,``,`y`,`x`]}],"scroll-behavior":[{scroll:[`auto`,`smooth`]}],"scrollbar-thumb-color":[{"scrollbar-thumb":E()}],"scrollbar-track-color":[{"scrollbar-track":E()}],"scrollbar-gutter":[{"scrollbar-gutter":[`auto`,`stable`,`both`]}],"scrollbar-w":[{scrollbar:[`auto`,`thin`,`none`]}],"scroll-m":[{"scroll-m":S()}],"scroll-mx":[{"scroll-mx":S()}],"scroll-my":[{"scroll-my":S()}],"scroll-ms":[{"scroll-ms":S()}],"scroll-me":[{"scroll-me":S()}],"scroll-mbs":[{"scroll-mbs":S()}],"scroll-mbe":[{"scroll-mbe":S()}],"scroll-mt":[{"scroll-mt":S()}],"scroll-mr":[{"scroll-mr":S()}],"scroll-mb":[{"scroll-mb":S()}],"scroll-ml":[{"scroll-ml":S()}],"scroll-p":[{"scroll-p":S()}],"scroll-px":[{"scroll-px":S()}],"scroll-py":[{"scroll-py":S()}],"scroll-ps":[{"scroll-ps":S()}],"scroll-pe":[{"scroll-pe":S()}],"scroll-pbs":[{"scroll-pbs":S()}],"scroll-pbe":[{"scroll-pbe":S()}],"scroll-pt":[{"scroll-pt":S()}],"scroll-pr":[{"scroll-pr":S()}],"scroll-pb":[{"scroll-pb":S()}],"scroll-pl":[{"scroll-pl":S()}],"snap-align":[{snap:[`start`,`end`,`center`,`align-none`]}],"snap-stop":[{snap:[`normal`,`always`]}],"snap-type":[{snap:[`none`,`x`,`y`,`both`]}],"snap-strictness":[{snap:[`mandatory`,`proximity`]}],touch:[{touch:[`auto`,`none`,`manipulation`]}],"touch-x":[{"touch-pan":[`x`,`left`,`right`]}],"touch-y":[{"touch-pan":[`y`,`up`,`down`]}],"touch-pz":[`touch-pinch-zoom`],select:[{select:[`none`,`text`,`all`,`auto`]}],"will-change":[{"will-change":[`auto`,`scroll`,`contents`,`transform`,N,M]}],fill:[{fill:[`none`,...E()]}],"stroke-w":[{stroke:[j,xt,mt,ht]}],stroke:[{stroke:[`none`,...E()]}],"forced-color-adjust":[{"forced-color-adjust":[`auto`,`none`]}]},conflictingClassGroups:{"container-named":[`container-type`],overflow:[`overflow-x`,`overflow-y`],overscroll:[`overscroll-x`,`overscroll-y`],inset:[`inset-x`,`inset-y`,`inset-bs`,`inset-be`,`start`,`end`,`top`,`right`,`bottom`,`left`],"inset-x":[`right`,`left`],"inset-y":[`top`,`bottom`],flex:[`basis`,`grow`,`shrink`],gap:[`gap-x`,`gap-y`],p:[`px`,`py`,`ps`,`pe`,`pbs`,`pbe`,`pt`,`pr`,`pb`,`pl`],px:[`pr`,`pl`],py:[`pt`,`pb`],m:[`mx`,`my`,`ms`,`me`,`mbs`,`mbe`,`mt`,`mr`,`mb`,`ml`],mx:[`mr`,`ml`],my:[`mt`,`mb`],size:[`w`,`h`],"font-size":[`leading`],"fvn-normal":[`fvn-ordinal`,`fvn-slashed-zero`,`fvn-figure`,`fvn-spacing`,`fvn-fraction`],"fvn-ordinal":[`fvn-normal`],"fvn-slashed-zero":[`fvn-normal`],"fvn-figure":[`fvn-normal`],"fvn-spacing":[`fvn-normal`],"fvn-fraction":[`fvn-normal`],"line-clamp":[`display`,`overflow`],rounded:[`rounded-s`,`rounded-e`,`rounded-t`,`rounded-r`,`rounded-b`,`rounded-l`,`rounded-ss`,`rounded-se`,`rounded-ee`,`rounded-es`,`rounded-tl`,`rounded-tr`,`rounded-br`,`rounded-bl`],"rounded-s":[`rounded-ss`,`rounded-es`],"rounded-e":[`rounded-se`,`rounded-ee`],"rounded-t":[`rounded-tl`,`rounded-tr`],"rounded-r":[`rounded-tr`,`rounded-br`],"rounded-b":[`rounded-br`,`rounded-bl`],"rounded-l":[`rounded-tl`,`rounded-bl`],"border-spacing":[`border-spacing-x`,`border-spacing-y`],"border-w":[`border-w-x`,`border-w-y`,`border-w-s`,`border-w-e`,`border-w-bs`,`border-w-be`,`border-w-t`,`border-w-r`,`border-w-b`,`border-w-l`],"border-w-x":[`border-w-r`,`border-w-l`],"border-w-y":[`border-w-t`,`border-w-b`],"border-color":[`border-color-x`,`border-color-y`,`border-color-s`,`border-color-e`,`border-color-bs`,`border-color-be`,`border-color-t`,`border-color-r`,`border-color-b`,`border-color-l`],"border-color-x":[`border-color-r`,`border-color-l`],"border-color-y":[`border-color-t`,`border-color-b`],translate:[`translate-x`,`translate-y`,`translate-none`],"translate-none":[`translate`,`translate-x`,`translate-y`,`translate-z`],"scroll-m":[`scroll-mx`,`scroll-my`,`scroll-ms`,`scroll-me`,`scroll-mbs`,`scroll-mbe`,`scroll-mt`,`scroll-mr`,`scroll-mb`,`scroll-ml`],"scroll-mx":[`scroll-mr`,`scroll-ml`],"scroll-my":[`scroll-mt`,`scroll-mb`],"scroll-p":[`scroll-px`,`scroll-py`,`scroll-ps`,`scroll-pe`,`scroll-pbs`,`scroll-pbe`,`scroll-pt`,`scroll-pr`,`scroll-pb`,`scroll-pl`],"scroll-px":[`scroll-pr`,`scroll-pl`],"scroll-py":[`scroll-pt`,`scroll-pb`],touch:[`touch-x`,`touch-y`,`touch-pz`],"touch-x":[`touch`],"touch-y":[`touch`],"touch-pz":[`touch`]},conflictingClassGroupModifiers:{"font-size":[`leading`]},postfixLookupClassGroups:[`container-type`],orderSensitiveModifiers:[`*`,`**`,`after`,`backdrop`,`before`,`details-content`,`file`,`first-letter`,`first-line`,`marker`,`placeholder`,`selection`]}});function L(...e){return Ft(D(e))}var It=e((e=>{var t=Symbol.for(`react.transitional.element`),n=Symbol.for(`react.fragment`);function r(e,n,r){var i=null;if(r!==void 0&&(i=``+r),n.key!==void 0&&(i=``+n.key),`key`in n)for(var a in r={},n)a!==`key`&&(r[a]=n[a]);else r=n;return n=r.ref,{$$typeof:t,type:e,key:i,ref:n===void 0?null:n,props:r}}e.Fragment=n,e.jsx=r,e.jsxs=r})),R=e(((e,t)=>{t.exports=It()}))(),Lt=me(`inline-flex min-h-[22px] items-center rounded-full border px-2 py-0.5 text-[11px] font-extrabold uppercase leading-tight`,{variants:{variant:{neutral:`border-border bg-secondary text-muted-foreground`,succeeded:`border-emerald-400/35 bg-emerald-500/15 text-emerald-300`,failed:`border-red-400/40 bg-red-500/15 text-red-300`,dead:`border-red-400/40 bg-red-500/15 text-red-300`,missing:`border-red-400/40 bg-red-500/15 text-red-300`,running:`border-amber-400/40 bg-amber-500/15 text-amber-300`,queued:`border-teal-400/40 bg-teal-500/15 text-teal-200`,canceled:`border-border bg-secondary text-muted-foreground`}},defaultVariants:{variant:`neutral`}});function z({className:e,variant:t,...n}){return(0,R.jsx)(`span`,{"data-slot":`badge`,className:L(Lt({variant:t,className:e})),...n})}var Rt=me(`inline-flex min-h-9 shrink-0 items-center justify-center gap-2 whitespace-nowrap rounded-md border text-sm font-semibold transition-colors focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 [&_svg]:shrink-0`,{variants:{variant:{default:`border-primary bg-primary text-primary-foreground hover:bg-primary/90`,secondary:`border-border bg-secondary text-secondary-foreground hover:bg-secondary/80`,outline:`border-border bg-background hover:bg-accent hover:text-accent-foreground`,ghost:`border-transparent hover:bg-accent hover:text-accent-foreground`,destructive:`border-destructive bg-destructive text-white hover:bg-destructive/90`},size:{default:`h-9 px-4 py-2`,sm:`h-8 rounded-md px-3 text-xs`,icon:`size-9`}},defaultVariants:{variant:`secondary`,size:`default`}});function B({className:e,variant:t,size:n,type:r=`button`,...i}){return(0,R.jsx)(`button`,{"data-slot":`button`,type:r,className:L(Rt({variant:t,size:n,className:e})),...i})}function V({className:e,...t}){return(0,R.jsx)(`div`,{"data-slot":`card`,className:L(`rounded-lg border bg-card text-card-foreground shadow-[0_18px_44px_rgb(0_0_0/0.22)]`,e),...t})}function H({className:e,...t}){return(0,R.jsx)(`div`,{"data-slot":`card-header`,className:L(`flex items-center justify-between gap-3 border-b px-4 py-3`,e),...t})}function zt({className:e,...t}){return(0,R.jsx)(`h2`,{"data-slot":`card-title`,className:L(`text-[15px] font-bold`,e),...t})}function Bt({className:e,...t}){return(0,R.jsx)(`div`,{"data-slot":`card-content`,className:L(`p-4`,e),...t})}function U({className:e,type:t,...n}){return(0,R.jsx)(`input`,{"data-slot":`input`,type:t,className:L(`flex min-h-9 w-full rounded-md border border-input bg-background px-3 py-2 text-sm text-foreground shadow-xs transition-colors placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50`,e),...n})}function W({className:e,...t}){return(0,R.jsx)(`label`,{"data-slot":`label`,className:L(`grid gap-1.5 text-xs font-bold text-muted-foreground`,e),...t})}function Vt({className:e,...t}){return(0,R.jsx)(`select`,{"data-slot":`select`,className:L(`flex min-h-9 w-full rounded-md border border-input bg-background px-3 py-2 text-sm text-foreground shadow-xs transition-colors focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50`,e),...t})}function Ht({className:e,...t}){return(0,R.jsx)(`table`,{"data-slot":`table`,className:L(`w-full border-collapse text-sm`,e),...t})}function Ut({className:e,...t}){return(0,R.jsx)(`thead`,{"data-slot":`table-header`,className:e,...t})}function Wt({className:e,...t}){return(0,R.jsx)(`tbody`,{"data-slot":`table-body`,className:e,...t})}function Gt({className:e,...t}){return(0,R.jsx)(`tr`,{"data-slot":`table-row`,className:L(`border-b transition-colors hover:bg-muted/45`,e),...t})}function G({className:e,...t}){return(0,R.jsx)(`th`,{"data-slot":`table-head`,className:L(`bg-secondary px-3 py-3 text-left align-middle text-xs font-extrabold text-muted-foreground`,e),...t})}function K({className:e,...t}){return(0,R.jsx)(`td`,{"data-slot":`table-cell`,className:L(`px-3 py-3 align-middle text-sm`,e),...t})}var Kt={pending:`Needs review`,selected:`Assigned to onboarder`,reachingout:`Reaching out`,awaitingcontribution:`Awaiting contribution`,onboarded:`Onboarded`,waitlist:`Waitlist`,rejected:`Rejected`};function qt(e){if(!e)return``;let t=new Date(e);return Number.isNaN(t.getTime())?e:t.toLocaleString(void 0,{year:`numeric`,month:`short`,day:`numeric`,hour:`2-digit`,minute:`2-digit`})}function Jt(e,t=new Date){if(!e)return null;let n=new Date(e);if(Number.isNaN(n.getTime()))return null;let r=t.getTime()-n.getTime();return r<0?0:Math.floor(r/864e5)}function Yt(e){return e==null?``:JSON.stringify(e,null,2)}function Xt(e){return e.onboarding_state||e.onboardingState||e.cOnboardingState||``}function Zt(e){let t=String(e||``).trim();if(!t)return`No status`;let n=t.toLowerCase();return Kt[n]?Kt[n]:t.replace(/[-_]+/g,` `).replace(/\s+/g,` `).trim().replace(/\b\w/g,e=>e.toUpperCase())}function Qt(e){let t=String(e||``).trim().toLowerCase();return!t||t===`pending`?`neutral`:t===`selected`?`queued`:t===`rejected`?`failed`:t===`onboarded`?`succeeded`:t===`waitlist`?`running`:`queued`}function $t(e){let t=String(e||``).trim();return!t||t.toLowerCase()===`none`?``:t}function en(e){let t=String(e||``).trim();return t?/^https?:\/\//i.test(t)?t:`https://${t.replace(/^\/+/,``)}`:``}function tn(e){try{return new URL(en(e))}catch{return null}}function nn(e,t){let n=e.toLowerCase();return n===t||n.endsWith(`.${t}`)}function rn(e){return e.split(`/`).filter(Boolean).map(e=>encodeURIComponent(e)).join(`/`)}function an(e){let t=String(e||``).trim();if(!t)return``;let n=tn(t);if(n&&nn(n.hostname,`linkedin.com`))return n.href;if(/^https?:\/\//i.test(t))return``;let r=t.replace(/^@/,``).replace(/^\/+|\/+$/g,``).replace(/^in\//i,``);return r?`https://www.linkedin.com/in/${rn(r)}`:``}function on(e){let t=String(e||``).trim().replace(/^@/,``);if(!t)return``;let n=tn(t);if(n&&nn(n.hostname,`github.com`))return n.href;if(/^https?:\/\//i.test(t))return``;let r=t.replace(/^\/+|\/+$/g,``);return r?`https://github.com/${rn(r)}`:``}var sn=[{category:`CRM`,label:`CRM`,description:`EspoCRM connection settings used by the API, worker, and Discord bot.`},{category:`Projects`,label:`Projects`,description:`ERPNext credentials and project workflow settings.`},{category:`Onboarding`,label:`Onboarding`,description:`Editable onboarding integrations such as DocuSeal, Outline, and onboarding email SMTP.`},{category:`AI`,label:`AI Providers`,description:`Provider credentials, base URLs, and model defaults.`},{category:`Agent`,label:`Agent Runtime`,description:`Planner, fallback, and tiered model routing for agent workflows.`},{category:`Observability`,label:`Observability`,description:`Telemetry and request tracing integrations.`},{category:`Intake`,label:`Intake`,description:`Resume and mailbox intake limits and parser defaults.`},{category:`Operations`,label:`Operations`,description:`Queue, sync, GitHub, and notification behavior.`},{category:`Legacy`,label:`Legacy`,description:`Older integrations retained for compatibility.`}],cn=new Map(sn.map((e,t)=>[e.category,{...e,index:t}]));function ln(e){return`configurationGroup-${e.replace(/[^a-zA-Z0-9_-]+/g,`-`)}`}function un(e){return e.key.startsWith(`ONBOARDING_EMAIL_`)||e.is_secret||e.value_type===`url`||e.key.endsWith(`_MODEL`)||e.key.endsWith(`_API_USER`)||e.key.endsWith(`_BASE_URL`)}var dn={people:`/dashboard/people`,gigs:`/dashboard/gigs`,projects:`/dashboard/projects`,onboarding:`/dashboard/onboarding`,jobs:`/dashboard/jobs`,agent:`/dashboard/agent`,audit:`/dashboard/audit`,configuration:`/dashboard/configuration`},fn={people:`people:read`,gigs:`gigs:read`,projects:`projects:read`,onboarding:`onboarding:read`,jobs:`jobs:read`,agent:`audit:read`,audit:`audit:read`,configuration:`configuration:read`},pn={discord:{label:`Discord`,options:[[`linked`,`Linked`],[`missing`,`Missing`]]},email_508:{label:`508 email`,options:[[`present`,`Present`],[`missing`,`Missing`]]},resume:{label:`Resume`,options:[[`present`,`Present`],[`missing`,`Missing`]]},skills:{label:`Skills`,options:[[`present`,`Parsed`],[`missing`,`Not parsed`]]},sync_status:{label:`Sync status`,options:[[`active`,`Active`],[`conflict`,`Conflict`],[`missing_in_crm`,`Missing in CRM`]]}},mn=[[`pending`,`Needs review`],[`selected`,`Assigned to onboarder`],[`reachingout`,`Reaching out`],[`awaitingcontribution`,`Awaiting contribution`],[`onboarded`,`Onboarded`],[`waitlist`,`Waitlist`],[`rejected`,`Rejected`]],hn=mn.slice(0,4),gn=new Set([`onboarded`,`waitlist`,`rejected`]);function _n(e){return String(e||``).trim().toLowerCase().replace(/[-_\s]+/g,``)}var vn=class extends Error{status;statusText;payload;url;method;constructor(e,t,n,r,i,a){super(e),this.name=`ApiRequestError`,this.status=t,this.statusText=n,this.payload=r,this.url=i,this.method=a}};function yn(e,t){let n=e.detail;if(typeof n==`string`&&n.trim())return n;let r=e.error;return typeof r==`string`?r===`person_not_found`?`No CRM person, ERPNext user, or ERPNext supplier matched "${typeof e.person==`string`&&e.person.trim()?e.person:`that person`}". Try an email address or an exact name from CRM/ERPNext.`:r===`candidate_not_found`?`The selected person record is no longer available. Search again and choose one of the current matches.`:r===`invalid_crm_profile`?`Paste a valid CRM Contact profile URL or Contact id.`:r===`crm_profile_not_found`?`That CRM Contact profile was not found.`:r===`crm_profile_mismatch`?`CRM returned a different Contact than the profile requested. Check the profile URL and try again.`:r===`crm_profile_lookup_failed`?`CRM profile lookup failed. Try again after CRM is reachable.`:r===`ambiguous_person`?`Multiple people matched. Choose the matching person record.`:r||t:t}function bn(e,t){return typeof e==`string`&&e.trim()?e:e instanceof Error&&e.message.trim()?e.message:t}function xn(){return window.location.pathname.split(`/`).filter(Boolean)[1]||``}function Sn(){let e=xn();return Object.hasOwn(dn,e)?e:`people`}function Cn(e=`gigs`){let[,t,n]=window.location.pathname.split(`/`).filter(Boolean);if(t!==e||!n)return``;try{return decodeURIComponent(n)}catch{return``}}async function q(e,t={}){let n=String(t.method||`GET`).toUpperCase(),r=new Headers(t.headers);r.set(`Accept`,`application/json`);let i;try{i=await fetch(e,{credentials:`same-origin`,...t,headers:r})}catch(t){throw new vn(bn(t,`Network request failed`),0,`Network request failed`,null,e,n)}if(i.status===401){let t=`${window.location.pathname}${window.location.search}`||`/dashboard`;throw window.location.assign(`/auth/login?next=${encodeURIComponent(t)}`),new vn(`Session expired`,i.status,i.statusText,null,e,n)}if(!i.ok){let t=i.statusText,r=null;try{r=await i.json(),r&&typeof r==`object`&&(t=yn(r,String(t||`Request failed`)))}catch{t=i.statusText}throw new vn(typeof t==`string`?t:JSON.stringify(t),i.status,i.statusText,r,e,n)}return i.json()}function wn(e,t,n){if(e===`gigs`){let e=t;if(n===`title`)return e.title||``;if(n===`status`)return e.status||``;if(n===`applications`)return Number(e.application_count||0);if(n===`activity`)return zn(e)}if(e===`projects`){let e=t;if(n===`display_name`)return e.display_name||``;if(n===`customer`)return e.customer||``;if(n===`status`)return e.source_status||``;if(n===`roster_count`)return Number(e.roster_count||0);if(n===`modified`)return e.source_modified_at||e.last_synced_at||``}if(e===`onboarding`){let e=t,r=e.profile_status||{};if(n===`name`)return e.name||e.email_508||e.email||``;if(n===`onboarding_state`){let t=Xt(e);return t.toLowerCase()===`pending`?`zzz-${t}`:t}if(n===`onboarder`)return e.onboarder||``;if(n===`updated`)return e.onboarding_updated_at||``;if(n===`profile_gaps`)return[!r.discord_linked,!r.latest_resume,Number(r.skills_count||0)<=0].filter(Boolean).length}if(e===`people`){let e=t,r=e.profile_status||{};if(n===`name`)return e.name||e.email_508||e.email||``;if(n===`status`)return[r.crm_active,r.is_member,r.discord_linked,r.email_508,r.latest_resume].filter(Boolean).length;if(n===`discord`)return e.discord_username||e.discord_user_id||``;if(n===`resume`)return e.latest_resume_name||e.latest_resume_id||``}if(e===`audit`){let e=t;if(n===`actor`)return e.actor_display_name||e.actor_subject||e.actor_provider||``}return t[n]??``}function Tn(e,t,n){let r=n.direction===`asc`?1:-1;return[...t].sort((t,i)=>{let a=wn(e,t,n.key),o=wn(e,i,n.key);return typeof a==`number`&&typeof o==`number`?(a-o)*r:String(a).localeCompare(String(o),void 0,{numeric:!0})*r})}function En({label:e,scope:t,sort:n,sortKey:r,onSort:i}){let a=n.key===r,o=n.direction===`asc`?`↑`:`↓`;return(0,R.jsx)(`button`,{type:`button`,"data-sort-scope":t,"data-sort-key":r,className:`text-left font-[inherit] text-inherit hover:text-foreground`,onClick:()=>i(t,r),children:a?`${e} ${o}`:e})}function Dn({className:e,label:t,scope:n,sort:r,sortKey:i,onSort:a}){return(0,R.jsx)(G,{className:e,"aria-sort":r.key===i?r.direction===`asc`?`ascending`:`descending`:`none`,children:(0,R.jsx)(En,{label:t,scope:n,sort:r,sortKey:i,onSort:a})})}function On({label:e,value:t,id:n}){return(0,R.jsxs)(V,{className:`p-4`,children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:e}),(0,R.jsx)(`strong`,{id:n,className:`block text-2xl`,children:t})]})}function kn({children:e,hidden:t}){return t?null:(0,R.jsx)(`div`,{className:`px-4 py-7 text-center text-sm text-muted-foreground`,children:e})}function An({value:e,query:t}){let n=t.trim().toLowerCase();if(!n)return(0,R.jsx)(R.Fragment,{children:e});let r=e.toLowerCase(),i=[],a=0,o=r.indexOf(n);for(;o>=0;){o>a&&i.push(e.slice(a,o));let t=o+n.length;i.push((0,R.jsx)(`mark`,{className:`rounded-sm bg-amber-200 px-0.5 text-inherit dark:bg-amber-500/35`,children:e.slice(o,t)},`${o}-${t}`)),a=t,o=r.indexOf(n,a)}return avoid 0);function bt(e){return s.includes(e)}function N(e){return s.includes(`${e}:dry_run`)}function xt(e){return bt(e)||N(e)}function St(e){return bt(fn[e])}function Ct(){return Object.keys(dn).find(e=>St(e))||`people`}function P(e,t){o({message:e,tone:t})}function F(e,t){P(bn(e,t),`error`)}function I(e,t){Ee(n=>({...n,[e]:t}))}function wt(e,t=!1){let n=e;St(n)||(P(`${n[0].toUpperCase()}${n.slice(1)} requires SSO validation`,`error`),n=Ct()),n!==`gigs`&&le(``),n!==`projects`&&ue(``),n===`gigs`&&t&&le(``),n===`projects`&&t&&ue(``),i(n),t?window.history.pushState({view:n},``,dn[n]):(!Object.hasOwn(dn,xn())||n!==e)&&window.history.replaceState({view:n},``,dn[n])}yt.current=wt;function Tt(e){return!u||!e?``:`${u}/#Contact/view/${encodeURIComponent(e)}`}function Et(e){return!u||!e?``:`${u}/api/v1/Attachment/file/${encodeURIComponent(e)}`}function Dt(e,t){Me(n=>{let r=n[e];return{...n,[e]:{key:t,direction:r.key===t&&r.direction===`asc`?`desc`:`asc`}}})}function Ot(e){le(e),w(h.find(t=>t.id===e)||null),i(`gigs`),window.history.pushState({view:`gigs`,gigId:e},``,`/dashboard/gigs/${encodeURIComponent(e)}`)}function kt(){le(``),w(null),window.history.replaceState({view:`gigs`},``,dn.gigs)}function At(e){ue(e),i(`projects`),window.history.pushState({view:`projects`,projectId:e},``,`/dashboard/projects/${encodeURIComponent(e)}`)}function jt(){ue(``),window.history.replaceState({view:`projects`},``,dn.projects)}async function Mt(){let e=await q(`/dashboard/api/me`);n(e);let t=Array.isArray(e.permissions)?e.permissions:[];return c(t),d((e.crm_base_url||``).replace(/\/+$/,``)),t}function Nt(){let e=new URLSearchParams({minutes:Ne,limit:`100`});return Fe&&e.set(`status`,Fe),Le.trim()&&e.set(`type`,Le.trim()),`/dashboard/api/jobs?${e.toString()}`}function Pt(){let e=new URLSearchParams({limit:String(Ge)});return ze&&e.set(`status`,ze),Ve.trim()&&e.set(`query`,Ve.trim()),Ue&&e.set(`include_historical`,`true`),`/dashboard/api/gigs?${e.toString()}`}function Ft(){let e=new URLSearchParams({limit:`100`,status:Ye});return qe.trim()&&e.set(`query`,qe.trim()),`/dashboard/api/projects?${e.toString()}`}async function It(){I(`jobs`,!0),P(`Loading background tasks`);try{let e=await q(Nt());p(e),P(`Loaded ${e.length} background task${e.length===1?``:`s`}`,`ok`)}catch(e){F(e,`Unable to load background tasks`)}finally{I(`jobs`,!1)}}async function Lt(){I(`gigs`,!0);try{let e=await q(Pt());y(e),P(`Loaded ${e.length} gig${e.length===1?``:`s`}`,`ok`),Jt()}catch(e){F(e,`Unable to load gigs`)}finally{I(`gigs`,!1)}}async function z(){I(`projects`,!0);try{let e=await q(Ft());S(e.projects||[]),ne(e.summary||{}),P(`Loaded ${(e.projects||[]).length} project${(e.projects||[]).length===1?``:`s`}`,`ok`)}catch(e){F(e,`Unable to load projects`)}finally{I(`projects`,!1)}}async function Rt(){I(`syncProjects`,!0),P(`Queueing project sync`);try{let e=await q(`/dashboard/api/sync/projects`,{method:`POST`});e.dry_run?P(`Dry run only: would queue ${e.would_enqueue?.job_type||`project sync`}`,`warning`):P(`Queued project sync ${e.job_id}`,`ok`)}catch(e){F(e,`Unable to queue project sync`)}finally{I(`syncProjects`,!1)}}async function V(e){let t=e.trim();if(t.length<2)return[];try{return(await q(`/dashboard/api/erpnext/customers?${new URLSearchParams({query:t}).toString()}`)).customers||[]}catch(e){return P(e instanceof Error?e.message:`Unable to search customers`,`error`),[]}}async function H(e){let t=e.trim();if(t.length<2)return[];try{return(await q(`/dashboard/api/erpnext/contacts?${new URLSearchParams({query:t}).toString()}`)).contacts||[]}catch(e){return P(e instanceof Error?e.message:`Unable to search contacts`,`error`),[]}}async function zt(e){let t=e.trim();if(t.length<2)return[];try{return(await q(`/dashboard/api/erpnext/account-managers?${new URLSearchParams({query:t}).toString()}`)).users||[]}catch(e){return P(e instanceof Error?e.message:`Unable to search account managers`,`error`),[]}}async function Bt(){try{let e=(await q(`/dashboard/api/erpnext/cost-centers`)).cost_centers||[];return e.length?e:[{name:`Projects - 5`,cost_center_name:`Projects`}]}catch(e){return P(e instanceof Error?e.message:`Unable to load cost centers`,`error`),[{name:`Projects - 5`,cost_center_name:`Projects`}]}}async function U(e){I(`createProject`,!0);try{let t=await q(`/dashboard/api/projects/create`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify(e)});return t.project.id?(S(e=>e.some(e=>e.id===t.project.id)?e.map(e=>e.id===t.project.id?t.project:e):[t.project,...e]),P(t.setup_warnings?.length?t.setup_warning_message||`Created ERP project setup; account manager setup needs follow-up`:`Created ERP project setup`,t.setup_warnings?.length?`warning`:`ok`),At(t.project.id)):(P([t.cache_refresh_message||`Created ERP project in ERPNext; local sync is pending`,t.setup_warnings?.length?t.setup_warning_message||`Account manager setup needs follow-up`:``].filter(Boolean).join(` `),t.setup_warnings?.length?`warning`:`ok`),z()),!0}catch(e){return P(e instanceof Error?e.message:`Unable to create project`,`error`),!1}finally{I(`createProject`,!1)}}async function W(e,t){I(`project:${e}:status`,!0);try{let n=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/status`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:t})});S(t=>t.map(t=>t.id===e?n.project:t)),P(`Updated project status`,`ok`)}catch(e){F(e,`Unable to update project`)}finally{I(`project:${e}:status`,!1)}}async function Vt(e,t){if(e.length===0)return!1;I(`projectsBulkUpdate`,!0);try{let n=await q(`/dashboard/api/projects/bulk`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({project_ids:e,...t})}),r=n.projects||[];S(e=>e.map(e=>r.find(t=>t.id===e.id)||e));let i=n.failures||[];return P(i.length?`Updated ${r.length}; ${i.length} failed`:`Updated ${r.length} project${r.length===1?``:`s`}`,i.length?`error`:`ok`),i.length===0}catch(e){return F(e,`Unable to bulk update projects`),!1}finally{I(`projectsBulkUpdate`,!1)}}async function Ht(e,t,n,r){let i=t.trim(),a=n.trim();if(!i||!a)return!1;I(`project:${e}:user`,!0);try{let t=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/users`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({user:i,candidate_id:a,...r||{}})});return S(n=>n.map(n=>n.id===e?t.project:n)),P(t.activity_cost_error?`Added project user; rate failed`:t.activity_cost?`Added project user and rate`:`Added project user`,t.activity_cost_error?`error`:`ok`),!0}catch(e){return F(e,`Unable to add project user`),!1}finally{I(`project:${e}:user`,!1)}}async function Ut(e,t){let n=t.trim();if(!n)return!1;I(`project:${e}:user`,!0);try{let t=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/users/remove`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({user:n})});return S(n=>n.map(n=>n.id===e?t.project:n)),P(`Removed project user`,`ok`),!0}catch(e){return P(e instanceof Error?e.message:`Unable to remove project user`,`error`),!1}finally{I(`project:${e}:user`,!1)}}async function Wt(e,t,n){let r=t.trim();if(!r)return!1;I(`project:${e}:historical`,!0);try{let t=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/historical-members`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({person:r,candidate_id:n})});return S(n=>n.map(n=>n.id===e?t.project:n)),Ae(null),P(`Added historical project member`,`ok`),!0}catch(t){if(t instanceof vn&&t.status===409){let n=t.payload?.candidates||[];if(n.length>0)return Ae({projectId:e,person:r,candidates:n}),P(`Choose the matching person record`,`error`),!1}return F(t,`Unable to add historical member`),!1}finally{I(`project:${e}:historical`,!1)}}async function Gt(e,t){let n=t.trim();if(!n)return!1;I(`project:${e}:historical`,!0);try{let t=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/historical-members/remove`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({source_user_id:n})});return S(n=>n.map(n=>n.id===e?t.project:n)),P(`Removed historical project member`,`ok`),!0}catch(e){return P(e instanceof Error?e.message:`Unable to remove historical member`,`error`),!1}finally{I(`project:${e}:historical`,!1)}}async function G(e,t,n){I(`project:${e}:wiki`,!0);try{await q(`/dashboard/api/projects/${encodeURIComponent(e)}/wiki-match`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:t,row_key:n})}),P(t===`no_row`?`Marked as no wiki row`:`Confirmed wiki match`,`ok`),await K()}catch(e){F(e,`Unable to save wiki match`)}finally{I(`project:${e}:wiki`,!1)}}async function K(){I(`wikiMatches`,!0);try{oe(await q(`/dashboard/api/projects/wiki-matches`)),P(`Loaded wiki match preview`,`ok`)}catch(e){F(e,`Unable to load wiki matches`)}finally{I(`wikiMatches`,!1)}}async function Kt(e){I(`gig:${e}:detail`,!0);try{w(await q(`/dashboard/api/gigs/${encodeURIComponent(e)}`))}catch(e){w(null),F(e,`Unable to load gig`)}finally{I(`gig:${e}:detail`,!1)}}async function qt(){await Lt(),T&&await Kt(T)}async function Jt(){if(bt(`gigs:read`)){I(`notifications`,!0);try{let e=await q(`/dashboard/api/notifications?limit=20`);Qe(e.stale_days||7),fe(e.notifications||[])}catch(e){F(e,`Unable to load notifications`)}finally{I(`notifications`,!1)}}}async function Yt(e,t){I(`gig:${e}:status`,!0);try{let n=(await q(`/dashboard/api/gigs/${encodeURIComponent(e)}/status`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:t})})).discord_title_sync?.status;P(n===`error`?`Updated gig status; Discord title sync failed`:`Updated gig status`,n===`error`?`error`:`ok`),await Lt(),T===e&&await Kt(e)}catch(e){F(e,`Unable to update gig`)}finally{I(`gig:${e}:status`,!1)}}async function Xt(e,t,n){I(`application:${t}:status`,!0);try{await q(`/dashboard/api/gigs/${encodeURIComponent(e)}/applications/${encodeURIComponent(t)}/status`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:n})}),P(`Updated candidate status`,`ok`),await Lt(),T===e&&await Kt(e)}catch(e){F(e,`Unable to update candidate`)}finally{I(`application:${t}:status`,!1)}}async function Qt(e,t){let n=t.trim();if(!n)return P(`Paste a CRM Contact profile first`,`warning`),!1;I(`gig:${e}:addCandidate`,!0);try{return await q(`/dashboard/api/gigs/${encodeURIComponent(e)}/applications`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({crm_profile:n})}),P(`Added candidate`,`ok`),await Lt(),T===e&&await Kt(e),!0}catch(e){return F(e,`Unable to add candidate`),!1}finally{I(`gig:${e}:addCandidate`,!1)}}function $t(){let e=new URLSearchParams({limit:`25`});$e.trim()&&e.set(`query`,$e.trim()),tt&&e.set(`is_member`,tt);for(let[t,n]of Object.entries(j))n&&e.set(t,n);return`/dashboard/api/people?${e.toString()}`}async function en(){I(`people`,!0);try{k(await q($t()))}catch(e){F(e,`Unable to load people`)}finally{I(`people`,!1)}}function tn(){let e=new URLSearchParams({limit:`25`});ct.trim()&&e.set(`query`,ct.trim()),ut&&e.set(`onboarding_state`,ut),ft.trim()&&e.set(`onboarder`,ft.trim());for(let[t,n]of Object.entries(M))n&&e.set(t,n);return`/dashboard/api/onboarding?${e.toString()}`}async function nn(){I(`onboarding`,!0);try{A(await q(tn()))}catch(e){F(e,`Unable to load onboarding`)}finally{I(`onboarding`,!1)}}async function rn(e,t){if(!e)return P(`Missing CRM contact`,`error`),null;let n=`onboarding-email-draft:${e}`;I(n,!0);try{let n=await q(`/dashboard/api/onboarding/${encodeURIComponent(e)}/email/draft`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify(t)});return P(`Drafted onboarding email`,`ok`),n}catch(e){return F(e,`Unable to draft onboarding email`),null}finally{I(n,!1)}}async function an(e,t,n){if(!e)return P(`Missing CRM contact`,`error`),null;let r=`onboarding-email-send:${e}`;I(r,!0);try{let r=await q(`/dashboard/api/onboarding/${encodeURIComponent(e)}/email/send`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({...t,markdown_body:n})});return A(t=>t.map(t=>t.crm_contact_id===e?{...t,onboarding_email_sent_at:r.onboarding_email_sent_at||t.onboarding_email_sent_at,onboarding_email_sent_by:r.onboarding_email_sent_by||t.onboarding_email_sent_by,onboarding_email_recipient:r.onboarding_email_recipient||r.recipient_email||t.onboarding_email_recipient}:t)),P(`Sent onboarding email`,`ok`),r}catch(e){return F(e,`Unable to send onboarding email`),null}finally{I(r,!1)}}async function on(){I(`audit`,!0);try{ge(await q(`/dashboard/api/audit-events?limit=25`))}catch(e){F(e,`Unable to load audit events`)}finally{I(`audit`,!1)}}async function sn(){I(`agent`,!0);try{ve(await q(`/dashboard/api/agent?limit=100`))}catch(e){F(e,`Unable to load agent report`)}finally{I(`agent`,!1)}}async function cn(){I(`configuration`,!0);try{be((await q(`/dashboard/api/configuration`)).items)}catch(e){F(e,`Unable to load configuration`)}finally{I(`configuration`,!1)}}async function ln(e,t){I(`configuration:${e}`,!0);try{be((await q(`/dashboard/api/configuration/${encodeURIComponent(e)}`,{method:`PUT`,headers:{"Content-Type":`application/json`},body:JSON.stringify({value:t})})).items),P(`Saved ${e}`,`ok`)}catch(t){F(t,`Unable to save ${e}`)}finally{I(`configuration:${e}`,!1)}}async function un(e){I(`configuration:${e}`,!0);try{be((await q(`/dashboard/api/configuration/${encodeURIComponent(e)}`,{method:`PUT`,headers:{"Content-Type":`application/json`},body:JSON.stringify({clear:!0})})).items),P(`Cleared ${e}`,`ok`)}catch(t){F(t,`Unable to clear ${e}`)}finally{I(`configuration:${e}`,!1)}}async function mn(e){I(`detail:${e}`,!0),P(`Loading ${e}`);try{we(await q(`/dashboard/api/jobs/${encodeURIComponent(e)}`)),P(`Loaded ${e}`,`ok`)}catch(e){F(e,`Unable to load task detail`)}finally{I(`detail:${e}`,!1)}}async function hn(e){I(`rerun:${e}`,!0),P(`Rerunning ${e}`);try{let t=await q(`/dashboard/api/jobs/${encodeURIComponent(e)}/rerun`,{method:`POST`});t.dry_run?P(`Dry run only: would rerun ${t.would_enqueue?.job_type||e}`,`warning`):(P(`Queued rerun ${t.job_id}`,`ok`),await It())}catch(e){F(e,`Unable to rerun task`)}finally{I(`rerun:${e}`,!1)}}async function yn(){I(`syncPeople`,!0),P(`Queueing people sync`);try{let e=await q(`/dashboard/api/sync/people`,{method:`POST`});e.dry_run?P(`Dry run only: would queue ${e.would_enqueue?.job_type||`people sync`}`,`warning`):P(`Queued people sync ${e.job_id}`,`ok`)}catch(e){F(e,`Unable to queue people sync`)}finally{I(`syncPeople`,!1)}}async function wn(e,t){let n=String(e||``).trim(),r=t.trim();if(!n){P(`Missing CRM contact id`,`error`);return}if(!r){P(`Enter a 508 username`,`error`);return}I(`onboarder:${n}`,!0),P(`Assigning ${r}`);try{let e=await q(`/dashboard/api/onboarding/${encodeURIComponent(n)}/onboarder`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({onboarder:r})});A(t=>t.map(t=>t.crm_contact_id===e.contact_id?{...t,onboarder:e.onboarder,onboarding_state:e.state_updated&&e.onboarding_state?e.onboarding_state:t.onboarding_state,onboarding_status_label:e.onboarding_status_label||(e.state_updated?void 0:t.onboarding_status_label)}:t)),P(`Assigned ${e.onboarder}`,`ok`)}catch(e){F(e,`Unable to assign onboarder`)}finally{I(`onboarder:${n}`,!1)}}async function En(e,t){let n=String(e||``).trim(),r=t.trim();if(!n){P(`Missing CRM contact id`,`error`);return}if(!r){P(`Choose an onboarding status`,`error`);return}I(`onboarding-status:${n}`,!0),P(`Updating onboarding status`);try{let e=await q(`/dashboard/api/onboarding/${encodeURIComponent(n)}/status`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:r})}),t=_n(e.onboarding_state),i=e.onboarding_status_label||Zt(t);A(n=>n.map(n=>n.crm_contact_id===e.contact_id?{...n,onboarding_state:t,onboarding_status_label:i}:n).filter(n=>n.crm_contact_id!==e.contact_id||!gn.has(t))),P(`Status set to ${i}`,`ok`)}catch(e){F(e,`Unable to update onboarding status`)}finally{I(`onboarding-status:${n}`,!1)}}async function Dn(e){let t=e.email.trim().toLowerCase(),n=e.first_name.trim();if(!t?.endsWith(`@508.dev`))return P(`Enter the engineer's @508.dev email`,`error`),null;if(!n)return P(`Enter the engineer name`,`error`),null;I(`engineerSetup`,!0);try{let r=await q(`/dashboard/api/onboarding/engineers`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({...e,email:t,first_name:n})});return P(`Set up ${r.employee_name||r.user||t}`,`ok`),r}catch(e){if(e instanceof vn&&e.status===409){let t=e.payload&&typeof e.payload==`object`?e.payload:null,n=(Array.isArray(t?.matches)?t.matches:[]).map(e=>e?.label||e?.email).filter(Boolean).slice(0,2).join(`, `);P(n?`Similar account exists: ${n}`:`Similar account exists; confirm before creating`,`error`)}else P(e instanceof Error?e.message:`Unable to set up engineer`,`error`);return null}finally{I(`engineerSetup`,!1)}}async function On(){I(`logout`,!0);try{let e=await q(`/auth/logout`,{method:`POST`});window.location.assign(e.end_session_url||`/dashboard`)}catch(e){F(e,`Unable to log out`),I(`logout`,!1)}}(0,l.useEffect)(()=>{Mt().then(e=>{let t=Sn(),n=e.includes(fn[t])?t:Object.keys(dn).find(t=>e.includes(fn[t]))||`people`;le(n===`gigs`?Cn():``),ue(n===`projects`?Cn(`projects`):``),i(n),(!Object.hasOwn(dn,xn())||n!==t)&&window.history.replaceState({view:n},``,dn[n])}).catch(e=>{F(e,`Dashboard failed to load`)})},[]),(0,l.useEffect)(()=>{let e=()=>{le(Cn()),ue(Cn(`projects`)),yt.current(Sn(),!1)};return window.addEventListener(`popstate`,e),()=>window.removeEventListener(`popstate`,e)},[]),(0,l.useEffect)(()=>{if(!a.message)return;let e=window.setTimeout(()=>o({message:``}),4500);return()=>window.clearTimeout(e)},[a.message]),(0,l.useEffect)(()=>{},[]),(0,l.useEffect)(()=>{s.length!==0&&(bt(`gigs:read`)&&Jt(),r===`people`&&en(),r===`gigs`&&Lt(),r===`projects`&&z(),r===`onboarding`&&nn(),r===`jobs`&&It(),r===`agent`&&sn(),r===`audit`&&on(),r===`configuration`&&cn())},[r]),(0,l.useEffect)(()=>{s.length!==0&&(bt(`gigs:read`)&&Jt(),r===`people`&&en(),r===`gigs`&&Lt(),r===`projects`&&z(),r===`onboarding`&&nn(),r===`jobs`&&It(),r===`agent`&&sn(),r===`audit`&&on(),r===`configuration`&&cn())},[s]),(0,l.useEffect)(()=>{r===`jobs`&&s.length>0&&It()},[Ne,Fe]),(0,l.useEffect)(()=>{r===`gigs`&&s.length>0&&Lt()},[ze,Ue,Ge]),(0,l.useEffect)(()=>{r===`projects`&&s.length>0&&z()},[Ye]),(0,l.useEffect)(()=>{r===`gigs`&&T&&s.length>0&&Kt(T)},[r,T,s]),(0,l.useEffect)(()=>{r===`people`&&s.length>0&&en()},[tt]),(0,l.useEffect)(()=>{r===`people`&&s.length>0&&en()},[j]),(0,l.useEffect)(()=>{r===`onboarding`&&s.length>0&&nn()},[ut]),(0,l.useEffect)(()=>{r===`onboarding`&&s.length>0&&nn()},[M]);let kn=(0,l.useMemo)(()=>Tn(`jobs`,f,je.jobs),[f,je.jobs]),An=(0,l.useMemo)(()=>Tn(`people`,O,je.people),[O,je.people]),jn=(0,l.useMemo)(()=>Tn(`onboarding`,me,je.onboarding),[me,je.onboarding]),Fn=(0,l.useMemo)(()=>Tn(`gigs`,h,je.gigs),[h,je.gigs]),In=(0,l.useMemo)(()=>Tn(`projects`,te,je.projects),[te,je.projects]),Ln=(0,l.useMemo)(()=>se?.id===T?se:Fn.find(e=>e.id===T)||null,[se,T,Fn]),Rn=(0,l.useMemo)(()=>In.find(e=>e.id===E)||null,[E,In]),zn=(0,l.useMemo)(()=>Tn(`audit`,he,je.audit),[he,je.audit]),Bn=(0,l.useMemo)(()=>f.reduce((e,t)=>(e[t.status]=(e[t.status]||0)+1,e),{}),[f]),Vn=Object.keys(pn).filter(e=>!j[e]),Hn=Object.keys(pn).filter(e=>e!==`sync_status`&&e!==`email_508`&&!M[e]);function Un(e){if(e.type===`stale_recruiting_gig`){let t=e.engagement_id||(e.id.startsWith(`stale-recruiting:`)?e.id.slice(17):``);t?Ot(t):(Be(`recruiting`),wt(`gigs`,!0))}D(!1)}(0,l.useEffect)(()=>{!Vn.includes(it)&&Vn[0]&&at(Vn[0])},[Vn,it]),(0,l.useEffect)(()=>{let e=pn[it]?.options;e?.[0]&&!e.some(([e])=>e===ot)&&st(e[0][0])},[it,ot]),(0,l.useEffect)(()=>{!Hn.includes(ht)&&Hn[0]&>(Hn[0])},[Hn,ht]),(0,l.useEffect)(()=>{let e=pn[ht]?.options;e?.[0]&&!e.some(([e])=>e===_t)&&vt(e[0][0])},[ht,_t]);let Wn=[t?.email,t?.crm_contact_id?`CRM ${t.crm_contact_id}`:``,t?.actor_provider].filter(Boolean).join(` | `);return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsx)(`header`,{className:`sticky top-0 z-20 border-b bg-background/90 backdrop-blur`,children:(0,R.jsxs)(`div`,{className:`mx-auto flex max-w-7xl flex-col gap-4 px-5 py-4 md:flex-row md:items-center md:justify-between`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`h1`,{className:`text-xl font-bold`,children:`508 Operations Dashboard`}),(0,R.jsx)(`p`,{className:`text-sm text-muted-foreground`,children:`Operations view for authenticated 508 operators.`})]}),(0,R.jsxs)(`div`,{className:`flex min-w-0 items-center gap-3`,children:[bt(`gigs:read`)?(0,R.jsx)(`div`,{className:`relative`,children:(0,R.jsxs)(B,{id:`notifications`,type:`button`,variant:`outline`,size:`icon`,"aria-label":`Notifications`,"aria-expanded":pe,onClick:()=>D(e=>!e),children:[(0,R.jsx)(g,{}),de.length>0?(0,R.jsx)(`span`,{className:`absolute -right-1 -top-1 grid min-h-5 min-w-5 place-items-center rounded-full bg-red-500 px-1 text-[11px] font-bold text-white`,children:de.length}):null]})}):null,(0,R.jsxs)(`div`,{className:`grid min-w-0 gap-0.5 text-right text-sm text-muted-foreground`,children:[(0,R.jsx)(`strong`,{id:`userName`,className:`truncate text-foreground`,children:t?.display_name||t?.email||t?.subject||`Loading user`}),(0,R.jsx)(`span`,{id:`userMeta`,className:`truncate`,children:Wn||`Checking session`})]}),(0,R.jsxs)(B,{id:`logout`,type:`button`,variant:`outline`,onClick:On,disabled:Te.logout,children:[(0,R.jsx)(ee,{}),`Log out`]})]})]})}),(0,R.jsx)(Mn,{open:pe,notifications:de,loading:Te.notifications,onClose:()=>D(!1),onRefresh:Jt,onOpenNotification:Un}),(0,R.jsx)(Pn,{toast:a}),null,(0,R.jsx)(Nn,{choice:ke,loading:!!(ke&&Te[`project:${ke.projectId}:historical`]),crmContactUrl:Tt,onClose:()=>Ae(null),onChoose:e=>{ke&&Wt(ke.projectId,ke.person,e)}}),(0,R.jsxs)(`main`,{className:`mx-auto grid max-w-7xl grid-cols-1 gap-5 px-5 py-5 md:grid-cols-[190px_minmax(0,1fr)]`,children:[(0,R.jsx)(`nav`,{className:`grid content-start gap-1 md:sticky md:top-24`,"aria-label":`Dashboard sections`,children:[[`people`,`People`,ce],[`gigs`,`Gigs`,_],[`projects`,`Projects`,x],[`onboarding`,`Onboarding`,v],[`jobs`,`Background tasks`,m],[`agent`,`Agent`,ae],[`audit`,`Audit`,b],[`configuration`,`Configuration`,ie]].filter(([e])=>St(e)).map(([e,t,n])=>(0,R.jsxs)(`a`,{className:L(`flex min-h-10 items-center gap-2 rounded-md border border-transparent px-3 text-sm font-extrabold text-muted-foreground hover:border-border hover:bg-secondary hover:text-foreground`,r===e&&`border-primary bg-accent text-accent-foreground`),"data-view-link":e,"data-permission":fn[e],href:dn[e],"aria-current":r===e?`page`:void 0,onClick:t=>{t.preventDefault(),wt(e,!0)},children:[(0,R.jsx)(n,{className:`size-4`}),t]},e))}),(0,R.jsxs)(`div`,{className:`grid min-w-0 gap-5`,children:[r===`people`?(0,R.jsx)($n,{crmBaseUrl:u,people:An,sort:je.people,canSync:xt(`people:sync`),loading:Te,peopleQuery:$e,peopleMember:tt,peopleFilters:j,peopleFilterKind:it,peopleFilterValue:ot,peopleFilterKeys:Vn,onSearch:en,onSync:yn,onSort:e=>Dt(`people`,e),setPeopleQuery:et,setPeopleMember:nt,setPeopleFilterKind:at,setPeopleFilterValue:st,addFilter:()=>{rt(e=>({...e,[it]:ot}))},removeFilter:e=>{rt(t=>{let n={...t};return delete n[e],n})},crmContactUrl:Tt,crmAttachmentUrl:Et}):null,r===`gigs`?(0,R.jsx)(Jn,{gigs:Fn,selectedGig:Ln,selectedGigId:T,sort:je.gigs,loading:Te,status:ze,query:Ve,includeHistorical:Ue,limit:Ge,staleDays:Ze,canWrite:bt(`gigs:write`),canIncludeHistorical:bt(`people:read`),crmContactUrl:Tt,crmAttachmentUrl:Et,setStatus:Be,setQuery:He,setIncludeHistorical:We,setLimit:Ke,onRefresh:qt,onSort:e=>Dt(`gigs`,e),onOpenGig:Ot,onCloseGig:kt,onUpdateStatus:Yt,onAddApplication:Qt,onUpdateApplicationStatus:Xt}):null,r===`projects`?(0,R.jsx)(Gn,{projects:In,selectedProject:Rn,selectedProjectId:E,summary:C,wikiMatches:re,sort:je.projects,loading:Te,query:qe,status:Ye,canSync:xt(`projects:sync`),canWrite:bt(`projects:write`),crmContactUrl:Tt,setQuery:Je,setStatus:Xe,onSearch:z,onSync:Rt,onSearchCustomers:V,onSearchContacts:H,onSearchAccountManagers:zt,onLoadCostCenters:Bt,onCreateProject:U,onUpdateStatus:W,onBulkUpdate:Vt,onAddUser:Ht,onRemoveUser:Ut,onAddHistoricalMember:Wt,onRemoveHistoricalMember:Gt,onUpdateWikiMatch:G,onWikiMatches:K,onOpenProject:At,onCloseProject:jt,onSort:e=>Dt(`projects`,e)}):null,r===`onboarding`?(0,R.jsx)(er,{people:jn,sort:je.onboarding,loading:Te,onboardingQuery:ct,onboardingState:ut,onboarderFilter:ft,onboardingFilters:M,onboardingFilterKind:ht,onboardingFilterValue:_t,onboardingFilterKeys:Hn,onSearch:nn,onSort:e=>Dt(`onboarding`,e),onAssign:wn,onStatusChange:En,onDraftEmail:rn,onSendEmail:an,onSetupEngineer:Dn,setOnboardingQuery:lt,setOnboardingState:dt,setOnboarderFilter:pt,setOnboardingFilterKind:gt,setOnboardingFilterValue:vt,addFilter:()=>{mt(e=>({...e,[ht]:_t}))},removeFilter:e=>{mt(t=>{let n={...t};return delete n[e],n})},crmContactUrl:Tt,crmAttachmentUrl:Et,canWrite:bt(`onboarding:write`),canConfigure:bt(`configuration:write`),onOpenConfiguration:()=>{Se({category:`Onboarding`,nonce:Date.now()}),wt(`configuration`,!0)}}):null,r===`jobs`?(0,R.jsx)(cr,{jobs:kn,jobDetail:Ce,sort:je.jobs,loading:Te,minutes:Ne,status:Fe,jobType:Le,jobCounts:Bn,canWrite:xt(`jobs:write`),setMinutes:Pe,setStatus:Ie,setJobType:Re,onSearch:It,onSort:e=>Dt(`jobs`,e),onDetail:mn,onRerun:hn}):null,r===`audit`?(0,R.jsx)(lr,{events:zn,sort:je.audit,loading:Te,onRefresh:on,onSort:e=>Dt(`audit`,e)}):null,r===`agent`?(0,R.jsx)(ur,{report:_e,loading:Te,onRefresh:sn}):null,r===`configuration`?(0,R.jsx)(dr,{items:ye,loading:Te,canWrite:bt(`configuration:write`),focusCategory:xe?.category,focusNonce:xe?.nonce,onRefresh:cn,onSave:ln,onClear:un}):null]})]})]})}function Mn({open:e,notifications:t,loading:n,onClose:r,onRefresh:i,onOpenNotification:a}){return e?(0,R.jsxs)(`div`,{className:`fixed inset-0 z-40`,"aria-labelledby":`notificationsTitle`,"aria-modal":`true`,role:`dialog`,children:[(0,R.jsx)(`button`,{type:`button`,className:`absolute inset-0 cursor-default bg-black/45`,"aria-label":`Close notifications`,onClick:r}),(0,R.jsxs)(`aside`,{className:`absolute right-0 top-0 grid h-full w-full max-w-md grid-rows-[auto_minmax(0,1fr)] border-l bg-background shadow-2xl`,children:[(0,R.jsxs)(`div`,{className:`flex items-center justify-between gap-3 border-b p-4`,children:[(0,R.jsxs)(`div`,{className:`grid gap-0.5`,children:[(0,R.jsx)(`strong`,{id:`notificationsTitle`,className:`text-base`,children:`Notifications`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:t.length===0?`No active notifications`:`${t.length} active`})]}),(0,R.jsxs)(`div`,{className:`flex items-center gap-2`,children:[(0,R.jsxs)(B,{type:`button`,variant:`outline`,size:`sm`,onClick:i,disabled:n,children:[(0,R.jsx)(C,{}),`Refresh`]}),(0,R.jsx)(B,{type:`button`,variant:`ghost`,size:`icon`,"aria-label":`Close`,onClick:r,children:(0,R.jsx)(w,{})})]})]}),(0,R.jsx)(`div`,{className:`min-h-0 overflow-auto p-4`,children:t.length===0?(0,R.jsx)(`div`,{className:`rounded-md border border-dashed p-6 text-sm text-muted-foreground`,children:`No active notifications.`}):(0,R.jsx)(`div`,{className:`grid gap-3`,children:t.map(e=>(0,R.jsxs)(`button`,{type:`button`,className:`grid gap-2 rounded-md border p-3 text-left hover:bg-secondary`,onClick:()=>a(e),children:[(0,R.jsx)(`span`,{className:`text-sm font-bold`,children:e.title}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.message})]},e.id))})})]})]}):null}function Nn({choice:e,loading:t,crmContactUrl:n,onClose:r,onChoose:i}){return e?(0,R.jsxs)(`div`,{className:`fixed inset-0 z-50 grid place-items-center p-4`,"aria-labelledby":`historicalPersonChoiceTitle`,"aria-modal":`true`,role:`dialog`,children:[(0,R.jsx)(`button`,{type:`button`,className:`absolute inset-0 cursor-default bg-black/45`,"aria-label":`Close person selection`,onClick:r}),(0,R.jsxs)(`div`,{className:`relative grid w-full max-w-2xl gap-4 rounded-md border bg-background p-5 shadow-2xl`,children:[(0,R.jsxs)(`div`,{className:`flex items-start justify-between gap-3`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`strong`,{id:`historicalPersonChoiceTitle`,className:`block text-base`,children:`Choose person record`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.person})]}),(0,R.jsx)(B,{type:`button`,variant:`ghost`,size:`icon`,"aria-label":`Close person selection`,onClick:r,children:(0,R.jsx)(w,{})})]}),(0,R.jsx)(`div`,{className:`grid gap-2`,children:e.candidates.map(e=>(0,R.jsxs)(`div`,{className:`grid gap-3 rounded-md border p-3 md:grid-cols-[minmax(0,1fr)_auto] md:items-center`,children:[(0,R.jsxs)(`div`,{className:`min-w-0`,children:[(0,R.jsx)(`strong`,{className:`block truncate`,children:e.label||e.full_name||e.email||`Person`}),(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-x-3 gap-y-1 text-sm text-muted-foreground`,children:[e.email?(0,R.jsx)(`span`,{children:e.email}):null,e.sources?.length?(0,R.jsx)(`span`,{children:e.sources.join(`, `)}):null,e.erpnext_user_id?(0,R.jsxs)(`span`,{children:[`ERP `,e.erpnext_user_id]}):null,e.supplier_erpnext_id?(0,R.jsxs)(`span`,{children:[`Supplier `,e.supplier_erpnext_id]}):null,e.crm_contact_id&&n(e.crm_contact_id)?(0,R.jsx)(`a`,{className:`font-semibold text-primary underline-offset-4 hover:underline`,href:n(e.crm_contact_id),target:`_blank`,rel:`noreferrer`,children:`CRM`}):null]})]}),(0,R.jsx)(B,{type:`button`,disabled:t,onClick:()=>i(e.candidate_id),children:`Select`})]},e.candidate_id))})]})]}):null}function Pn({toast:e}){return e.message?(0,R.jsx)(`div`,{id:`toast`,role:`status`,className:L(`fixed bottom-5 right-5 z-50 max-w-sm rounded-md border bg-background px-4 py-3 text-sm font-semibold shadow-lg`,e.tone===`ok`&&`border-emerald-500/40 text-emerald-300`,e.tone===`warning`&&`border-amber-500/40 text-amber-200`,e.tone===`error`&&`border-red-500/40 text-red-300`),children:e.message}):null}function Fn({filters:e,onRemove:t,suffix:n=`filter`}){return(0,R.jsx)(`fieldset`,{className:`m-0 flex min-h-7 flex-wrap gap-2 border-0 p-0`,"aria-label":`Active filters`,children:Object.entries(e).map(([e,r])=>{let i=pn[e],a=i.options.find(([e])=>e===r),o=`${i.label}: ${a?a[1]:r}`;return(0,R.jsxs)(B,{type:`button`,variant:`outline`,size:`sm`,className:`rounded-full`,"aria-label":`Remove ${o} ${n}`,onClick:()=>t(e),children:[o,` x`]},e)})})}var In=[`recruiting`,`filled`,`unknown`,`lost`,`outdated`],Ln=[`suggested`,`interested`,`reviewing`,`contacted`,`accepted`,`unavailable`,`rejected`,`withdrawn`];function Rn(e){return String(e||``).replace(/[-_]+/g,` `).replace(/\s+/g,` `).trim().replace(/\b\w/g,e=>e.toUpperCase())}function zn(e){let t=[e.last_activity_at,e.last_status_changed_at,e.posted_at,e.created_at].map(e=>e?new Date(e).getTime():NaN).filter(e=>!Number.isNaN(e));return t.length>0?new Date(Math.max(...t)).toISOString():``}function Bn(e,t){if(e.status!==`recruiting`)return null;let n=Jt(zn(e));return n===null||ne.projects.map(e=>e.id),[e.projects]),m=(0,l.useMemo)(()=>new Set(p),[p]),g=n.filter(e=>m.has(e)),_=e.projects.length>0&&g.length===e.projects.length;(0,l.useEffect)(()=>{r(e=>e.filter(e=>m.has(e)))},[m]);function v(e,t){r(n=>t?Array.from(new Set([...n,e])):n.filter(t=>t!==e))}async function b(){let t={};i&&(t.status=i),o&&(t.project_type=o),await e.onBulkUpdate(g,t)&&(r([]),a(``),s(``),u(!1))}let x=(0,R.jsxs)(V,{className:`grid gap-3 p-4 md:grid-cols-[minmax(0,1fr)_180px_auto_auto_auto] md:items-end`,children:[(0,R.jsxs)(W,{children:[`Search projects`,(0,R.jsx)(U,{id:`projectQuery`,value:e.query,autoComplete:`off`,placeholder:`Project, customer, ERP id`,onChange:t=>e.setQuery(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onSearch()})]}),(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{id:`projectStatus`,value:e.status,onChange:t=>e.setStatus(t.target.value),children:[(0,R.jsx)(`option`,{value:`Open`,children:`Open`}),(0,R.jsx)(`option`,{value:``,children:`Any status`})]})]}),(0,R.jsxs)(B,{id:`refreshProjects`,type:`button`,onClick:e.onSearch,disabled:e.loading.projects,children:[(0,R.jsx)(C,{}),`Refresh`]}),e.canSync?(0,R.jsxs)(B,{id:`syncProjects`,type:`button`,variant:`outline`,onClick:e.onSync,disabled:e.loading.syncProjects,children:[(0,R.jsx)(C,{}),`Sync ERP`]}):null,(0,R.jsxs)(B,{id:`wikiProjectMatches`,type:`button`,variant:`outline`,onClick:e.onWikiMatches,disabled:e.loading.wikiMatches,children:[(0,R.jsx)(ne,{}),`Wiki match`]})]});return e.selectedProjectId&&!e.selectedProject&&e.loading.projects?(0,R.jsxs)(R.Fragment,{children:[x,(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Project detail`})}),(0,R.jsx)(Bt,{className:`text-sm text-muted-foreground`,children:`Loading project.`})]})]}):e.selectedProjectId&&!e.selectedProject?(0,R.jsxs)(R.Fragment,{children:[x,(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Project detail`})}),(0,R.jsxs)(Bt,{className:`grid gap-3`,children:[(0,R.jsx)(`p`,{className:`text-sm text-muted-foreground`,children:`This project is not in the current result set. Clear filters or refresh the project list.`}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,onClick:e.onCloseProject,children:[(0,R.jsx)(h,{}),`Back to projects`]})]})]})]}):e.selectedProject?(0,R.jsxs)(R.Fragment,{children:[x,(0,R.jsx)(qn,{project:e.selectedProject,loading:e.loading,canWrite:e.canWrite,crmContactUrl:e.crmContactUrl,onBack:e.onCloseProject,onUpdateStatus:e.onUpdateStatus,onAddUser:e.onAddUser,onRemoveUser:e.onRemoveUser,onAddHistoricalMember:e.onAddHistoricalMember,onRemoveHistoricalMember:e.onRemoveHistoricalMember})]}):(0,R.jsxs)(R.Fragment,{children:[x,(0,R.jsxs)(`section`,{className:`grid gap-3 md:grid-cols-2`,"aria-label":`Project summary`,children:[(0,R.jsx)(On,{id:`projectMetricOpen`,label:`Open`,value:e.summary.open_project_count||0}),(0,R.jsx)(On,{id:`projectMetricTotal`,label:`Projects`,value:e.summary.project_count||0})]}),e.canWrite?(0,R.jsxs)(V,{className:`flex flex-wrap items-center justify-between gap-3 p-4`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Selected`}),(0,R.jsxs)(`strong`,{className:`block`,children:[g.length,` project(s)`]})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-2`,children:[(0,R.jsxs)(B,{type:`button`,onClick:()=>f(!0),children:[(0,R.jsx)(S,{}),`New project`]}),(0,R.jsx)(B,{type:`button`,variant:`outline`,disabled:g.length===0,onClick:()=>u(!0),children:`Bulk edit`})]})]}):null,d?(0,R.jsx)(Kn,{loading:e.loading.createProject,onClose:()=>f(!1),onSearchCustomers:e.onSearchCustomers,onSearchContacts:e.onSearchContacts,onSearchAccountManagers:e.onSearchAccountManagers,onLoadCostCenters:e.onLoadCostCenters,onCreateProject:e.onCreateProject}):null,c?(0,R.jsxs)(`div`,{className:`fixed inset-0 z-50 grid place-items-center p-4`,"aria-labelledby":`bulkProjectEditTitle`,"aria-modal":`true`,role:`dialog`,children:[(0,R.jsx)(`button`,{type:`button`,className:`absolute inset-0 cursor-default bg-black/45`,"aria-label":`Close bulk project edit`,onClick:()=>u(!1)}),(0,R.jsxs)(`div`,{className:`relative grid w-full max-w-lg gap-4 rounded-md border bg-background p-5 shadow-2xl`,children:[(0,R.jsxs)(`div`,{className:`flex items-start justify-between gap-3`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`strong`,{id:`bulkProjectEditTitle`,className:`block text-base`,children:`Bulk edit projects`}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[g.length,` selected`]})]}),(0,R.jsx)(B,{type:`button`,variant:`ghost`,size:`icon`,"aria-label":`Close bulk project edit`,onClick:()=>u(!1),children:(0,R.jsx)(w,{})})]}),(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsx)(`strong`,{className:`text-sm`,children:`Changes`}),(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{value:i,onChange:e=>a(e.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`No change`}),(0,R.jsx)(`option`,{value:`Open`,children:`Open`}),(0,R.jsx)(`option`,{value:`Completed`,children:`Completed`}),(0,R.jsx)(`option`,{value:`Cancelled`,children:`Cancelled`})]})]}),(0,R.jsxs)(W,{children:[`ERP Type`,(0,R.jsxs)(Vt,{value:o,onChange:e=>s(e.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`No change`}),(0,R.jsx)(`option`,{value:`Internal`,children:`Internal`}),(0,R.jsx)(`option`,{value:`External`,children:`External`})]})]})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-2`,children:[(0,R.jsx)(B,{type:`button`,variant:`outline`,onClick:()=>u(!1),children:`Cancel`}),(0,R.jsx)(B,{type:`button`,disabled:e.loading.projectsBulkUpdate||g.length===0||!i&&!o,onClick:()=>void b(),children:`Apply changes`})]})]})]}):null,(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`ERP projects`}),(0,R.jsx)(`span`,{id:`projectsStatus`,className:`text-sm text-muted-foreground`,children:e.loading.projects?`Loading`:`${e.projects.length} shown | synced ${qt(e.summary.last_synced_at)}`})]}),(0,R.jsx)(kn,{hidden:e.projects.length!==0,children:`No projects match this view. Sync ERP projects if the cache is empty.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`projectsTable`,className:L(`min-w-[1100px]`,e.projects.length===0&&`hidden`),"aria-label":`ERP projects`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[e.canWrite?(0,R.jsx)(G,{className:`w-[48px]`,children:(0,R.jsx)(`input`,{type:`checkbox`,"aria-label":`Select all visible projects`,checked:_,onChange:e=>{r(e.target.checked?p:[])}})}):null,(0,R.jsx)(Dn,{className:`w-[24%]`,label:`Project`,scope:`projects`,sort:e.sort,sortKey:`display_name`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[16%]`,label:`Customer`,scope:`projects`,sort:e.sort,sortKey:`customer`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[10%]`,label:`Status`,scope:`projects`,sort:e.sort,sortKey:`status`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(G,{className:`w-[16%]`,children:`Timeline`}),(0,R.jsx)(Dn,{className:`w-[10%]`,label:`Roster`,scope:`projects`,sort:e.sort,sortKey:`roster_count`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[14%]`,label:`Modified`,scope:`projects`,sort:e.sort,sortKey:`modified`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(G,{children:`ERP`})]})}),(0,R.jsx)(Wt,{id:`projectsBody`,children:e.projects.map(t=>{let n=t.roster_members||[];return(0,R.jsxs)(Gt,{children:[e.canWrite?(0,R.jsx)(K,{children:(0,R.jsx)(`input`,{type:`checkbox`,"aria-label":`Select ${t.display_name}`,checked:g.includes(t.id),onChange:e=>v(t.id,e.target.checked)})}):null,(0,R.jsxs)(K,{children:[(0,R.jsx)(`button`,{type:`button`,className:`text-left font-bold text-primary underline-offset-4 hover:underline`,onClick:()=>e.onOpenProject(t.id),children:t.display_name}),(0,R.jsxs)(`div`,{className:`mt-1 flex flex-wrap items-center gap-1.5`,children:[t.project_type?(0,R.jsx)(z,{variant:`neutral`,children:t.project_type}):null,t.linked_engagement_count?(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[t.linked_engagement_count,` linked gig`]}):null]})]}),(0,R.jsx)(K,{children:t.customer_erpnext_url?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-semibold text-primary underline-offset-4 hover:underline`,href:t.customer_erpnext_url,target:`_blank`,rel:`noreferrer`,children:[t.customer,(0,R.jsx)(y,{className:`size-3.5`})]}):t.customer||`None`}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:Vn(t.source_status),children:t.source_status||`Unknown`})}),(0,R.jsx)(K,{children:[t.actual_start_date,t.actual_end_date].filter(Boolean).map(e=>Hn(e)).join(` to `)||`Not set`}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-1`,children:[(0,R.jsx)(`strong`,{children:n.length}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[n.map(Un).slice(0,4).join(`, `)||`No ERP roster`,n.length>4?` +${n.length-4}`:``]})]})}),(0,R.jsx)(K,{children:qt(t.source_modified_at)}),(0,R.jsx)(K,{className:`text-xs`,children:t.erpnext_project_url?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-mono font-semibold text-primary underline-offset-4 hover:underline`,href:t.erpnext_project_url,target:`_blank`,rel:`noreferrer`,children:[t.erpnext_project_id,(0,R.jsx)(y,{className:`size-3.5`})]}):(0,R.jsx)(`span`,{className:`font-mono`,children:`Unlinked`})})]},t.id)})})]})})]}),e.wikiMatches?(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Wiki match preview`}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[e.wikiMatches.document?.title||`Client & Project Info`,` |`,` `,qt(e.wikiMatches.document?.updatedAt)]})]}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`wikiMatchesTable`,className:`min-w-[920px]`,"aria-label":`Wiki matches`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{children:`ERP project`}),(0,R.jsx)(G,{children:`Best wiki row`}),(0,R.jsx)(G,{children:`Confidence`}),(0,R.jsx)(G,{children:`Section`}),(0,R.jsx)(G,{children:`Decision`})]})}),(0,R.jsx)(Wt,{children:t.map((t,n)=>{let r=t.project,i=t.best_match?.row||{},a=t.manual_match?.match_status||``,o=r?.id||i.row_key||[i.section,i.Client].filter(Boolean).join(`:`)||`wiki-match-${n}`;return(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:r?.display_name||`Unknown`}),(0,R.jsxs)(K,{children:[(0,R.jsx)(`strong`,{children:i.Client||`No match`}),(0,R.jsx)(`div`,{className:`text-sm text-muted-foreground`,children:[i.DRI,i.Members].filter(Boolean).join(` | `)})]}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:t.best_match?.confidence===`high`?`succeeded`:t.best_match?.confidence===`medium`?`running`:`neutral`,children:t.best_match?`${t.best_match.confidence} ${t.best_match.score}`:`none`})}),(0,R.jsx)(K,{children:i.section||``}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-2`,children:[a?(0,R.jsx)(z,{variant:a===`confirmed`?`succeeded`:`neutral`,children:a===`no_row`?`No wiki row`:`Confirmed`}):null,e.canWrite&&r?.id?(0,R.jsxs)(R.Fragment,{children:[i.row_key?(0,R.jsx)(B,{type:`button`,variant:`outline`,size:`sm`,disabled:e.loading[`project:${r.id}:wiki`],onClick:()=>void e.onUpdateWikiMatch(r.id,`confirmed`,i.row_key),children:`Confirm`}):null,(0,R.jsx)(B,{type:`button`,variant:`outline`,size:`sm`,disabled:e.loading[`project:${r.id}:wiki`],onClick:()=>void e.onUpdateWikiMatch(r.id,`no_row`),children:`No row`})]}):null]})})]},o)})})]})})]}):null]})}function Kn(e){let[t,n]=(0,l.useState)(``),[r,i]=(0,l.useState)(`new`),[a,o]=(0,l.useState)(``),[s,c]=(0,l.useState)(``),[u,d]=(0,l.useState)(``),[f,p]=(0,l.useState)([]),[m,h]=(0,l.useState)(``),[g,_]=(0,l.useState)(``),[v,y]=(0,l.useState)([]),[b,x]=(0,l.useState)(`USD`),[ee,te]=(0,l.useState)(``),[S,C]=(0,l.useState)(``),[ne,re]=(0,l.useState)(``),[ie,ae]=(0,l.useState)(``),[oe,se]=(0,l.useState)(``),[ce,T]=(0,l.useState)(``),[le,E]=(0,l.useState)(`United States`),[ue,de]=(0,l.useState)(``),[fe,pe]=(0,l.useState)(`new`),[D,O]=(0,l.useState)(``),[k,me]=(0,l.useState)(``),[A,he]=(0,l.useState)([]),[ge,_e]=(0,l.useState)(``),[ve,ye]=(0,l.useState)(``),[be,xe]=(0,l.useState)(``),[Se,Ce]=(0,l.useState)(``),[we,Te]=(0,l.useState)(``),[Ee,De]=(0,l.useState)(!1),[Oe,ke]=(0,l.useState)([{name:`Projects - 5`,cost_center_name:`Projects`}]),[Ae,je]=(0,l.useState)(`Projects - 5`),[Me,Ne]=(0,l.useState)(``),[Pe,Fe]=(0,l.useState)(!1),Ie=(0,l.useRef)(e.onSearchCustomers),Le=(0,l.useRef)(e.onSearchContacts),Re=(0,l.useRef)(e.onSearchAccountManagers),ze=(0,l.useRef)(e.onLoadCostCenters),Be=(0,l.useRef)(0),Ve=(0,l.useRef)(0),He=(0,l.useRef)(0),Ue=(0,l.useRef)(0),We=t.trim()?`Engineering for ${t.trim()}`.slice(0,140):``,Ge=[ne,ie,oe,ce,ue].some(e=>e.trim()),Ke=[ge,ve,be,Se,we].some(e=>e.trim()),qe=t.trim()&&(r===`new`?a.trim():u.trim())&&!e.loading;(0,l.useEffect)(()=>{Ie.current=e.onSearchCustomers},[e.onSearchCustomers]),(0,l.useEffect)(()=>{Le.current=e.onSearchContacts},[e.onSearchContacts]),(0,l.useEffect)(()=>{Re.current=e.onSearchAccountManagers},[e.onSearchAccountManagers]),(0,l.useEffect)(()=>{ze.current=e.onLoadCostCenters},[e.onLoadCostCenters]),(0,l.useEffect)(()=>{let e=!0,t=Be.current+1;return Be.current=t,ze.current().then(n=>{!e||Be.current!==t||(ke(n),je(e=>n.some(t=>t.name===e)?e:`Projects - 5`))}),()=>{e=!1}},[]),(0,l.useEffect)(()=>{if(r!==`existing`){Ve.current+=1,p([]);return}let e=!0,t=Ve.current+1;Ve.current=t;let n=window.setTimeout(()=>{Ie.current(s).then(n=>{!e||Ve.current!==t||p(n)})},250);return()=>{e=!1,window.clearTimeout(n)}},[r,s]),(0,l.useEffect)(()=>{if(r!==`new`){He.current+=1,y([]);return}let e=!0,t=He.current+1;He.current=t;let n=window.setTimeout(()=>{Re.current(m).then(n=>{!e||He.current!==t||y(n)})},250);return()=>{e=!1,window.clearTimeout(n)}},[r,m]),(0,l.useEffect)(()=>{if(r!==`new`||fe!==`existing`){Ue.current+=1,he([]);return}let e=!0,t=Ue.current+1;Ue.current=t;let n=window.setTimeout(()=>{Le.current(D).then(n=>{!e||Ue.current!==t||he(n)})},250);return()=>{e=!1,window.clearTimeout(n)}},[r,fe,D]);async function Je(){qe&&await e.onCreateProject({project_name:t.trim(),customer_mode:r,customer_name:r===`new`?a.trim():void 0,customer:r===`existing`?u.trim():void 0,account_manager:r===`new`&&g.trim()||void 0,default_billing_currency:r===`new`?b.trim()||`USD`:void 0,default_cost_center:Ae.trim()||`Projects - 5`,activity_type:Pe&&Me.trim()||void 0,customer_details:r===`new`&&ee.trim()||void 0,customer_website:r===`new`&&S.trim()||void 0,address_line1:r===`new`&&ne.trim()||void 0,address_line2:r===`new`&&ie.trim()||void 0,address_city:r===`new`&&oe.trim()||void 0,address_state:r===`new`&&ce.trim()||void 0,address_country:r===`new`&&ne.trim()?le.trim()||`United States`:void 0,address_postal_code:r===`new`&&ue.trim()||void 0,contact:r===`new`&&fe===`existing`&&k.trim()||void 0,contact_first_name:r===`new`&&fe===`new`&&ge.trim()||void 0,contact_last_name:r===`new`&&fe===`new`&&ve.trim()||void 0,contact_email:r===`new`&&fe===`new`&&be.trim()||void 0,contact_phone:r===`new`&&fe===`new`&&Se.trim()||void 0,contact_mobile:r===`new`&&fe===`new`&&we.trim()||void 0})&&e.onClose()}return(0,R.jsxs)(`div`,{className:`fixed inset-0 z-50 grid place-items-center p-4`,"aria-labelledby":`createProjectTitle`,"aria-modal":`true`,role:`dialog`,children:[(0,R.jsx)(`button`,{type:`button`,className:`absolute inset-0 cursor-default bg-black/45`,"aria-label":`Close project creation`,onClick:e.onClose}),(0,R.jsxs)(`div`,{className:`relative grid max-h-[90vh] w-full max-w-2xl gap-4 overflow-y-auto rounded-md border bg-background p-5 shadow-2xl`,children:[(0,R.jsxs)(`div`,{className:`flex items-start justify-between gap-3`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`strong`,{id:`createProjectTitle`,className:`block text-base`,children:`New ERP project`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:`Creates a project and links a new or existing customer.`})]}),(0,R.jsx)(B,{type:`button`,variant:`ghost`,size:`icon`,"aria-label":`Close project creation`,onClick:e.onClose,children:(0,R.jsx)(w,{})})]}),(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsxs)(W,{children:[`Project name *`,(0,R.jsx)(U,{value:t,autoComplete:`off`,maxLength:140,placeholder:`Acme Portal`,onChange:e=>n(e.target.value)})]}),(0,R.jsxs)(`div`,{className:`grid gap-2`,children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Customer`}),(0,R.jsx)(`div`,{className:`grid grid-cols-2 gap-2`,children:[`new`,`existing`].map(e=>(0,R.jsx)(B,{type:`button`,variant:r===e?`default`:`outline`,onClick:()=>i(e),children:e===`new`?`New customer`:`Existing customer`},e))})]}),r===`new`?(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Customer name *`,(0,R.jsx)(U,{value:a,autoComplete:`off`,maxLength:140,placeholder:`Acme`,onChange:e=>o(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Account manager`,(0,R.jsx)(U,{value:m,autoComplete:`off`,placeholder:`Search @508.dev user`,onChange:e=>{h(e.target.value),_(``)}})]}),m.trim().length>=2?(0,R.jsx)(`div`,{className:`grid max-h-40 gap-2 overflow-y-auto rounded-md border p-2 md:col-span-2`,children:v.length?v.map(e=>{let t=e.email||e.name||``;return(0,R.jsxs)(`label`,{className:`flex cursor-pointer items-start gap-2 rounded-sm px-2 py-1.5 hover:bg-secondary`,children:[(0,R.jsx)(`input`,{type:`radio`,name:`erpAccountManager`,value:t,checked:g===t,onChange:()=>{_(t),h(t)}}),(0,R.jsxs)(`span`,{className:`grid gap-0.5 text-sm`,children:[(0,R.jsx)(`strong`,{children:e.full_name||t}),(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:t})]})]},t)}):(0,R.jsx)(`span`,{className:`px-2 py-3 text-sm text-muted-foreground`,children:`No enabled @508.dev users found.`})}):null]}):(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsxs)(W,{children:[`Find customer *`,(0,R.jsx)(U,{value:s,autoComplete:`off`,placeholder:`Search customer`,onChange:e=>c(e.target.value)})]}),(0,R.jsx)(`div`,{className:`grid max-h-48 gap-2 overflow-y-auto rounded-md border p-2`,children:f.length?f.map(e=>{let t=e.name||e.customer_name||``;return(0,R.jsxs)(`label`,{className:`flex cursor-pointer items-start gap-2 rounded-sm px-2 py-1.5 hover:bg-secondary`,children:[(0,R.jsx)(`input`,{type:`radio`,name:`erpCustomer`,value:t,checked:u===t,onChange:()=>d(t)}),(0,R.jsxs)(`span`,{className:`grid gap-0.5 text-sm`,children:[(0,R.jsx)(`strong`,{children:e.customer_name||t}),(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:[t,e.default_currency].filter(Boolean).join(` | `)})]})]},t)}):(0,R.jsx)(`span`,{className:`px-2 py-3 text-sm text-muted-foreground`,children:`Search at least two characters.`})})]}),r===`new`?(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Customer details`,(0,R.jsx)(`textarea`,{value:ee,className:`min-h-20 w-full rounded-md border border-input bg-background px-3 py-2 text-sm text-foreground shadow-xs transition-colors placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]`,maxLength:2e3,placeholder:`More information`,onChange:e=>te(e.target.value)})]}),(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Website`,(0,R.jsx)(U,{value:S,autoComplete:`url`,placeholder:`https://example.com`,onChange:e=>C(e.target.value)})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsxs)(`div`,{className:`flex items-center justify-between gap-3`,children:[(0,R.jsx)(`strong`,{className:`text-sm text-foreground`,children:`Contact`}),(0,R.jsx)(`div`,{className:`grid grid-cols-2 gap-2`,children:[`new`,`existing`].map(e=>(0,R.jsx)(B,{type:`button`,size:`sm`,variant:fe===e?`default`:`outline`,onClick:()=>pe(e),children:e===`new`?`New`:`Existing`},e))})]}),fe===`new`?(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[(0,R.jsxs)(W,{children:[`First name `,Ke?`*`:``,(0,R.jsx)(U,{value:ge,autoComplete:`given-name`,onChange:e=>_e(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Last name`,(0,R.jsx)(U,{value:ve,autoComplete:`family-name`,onChange:e=>ye(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Email`,(0,R.jsx)(U,{value:be,type:`email`,autoComplete:`email`,onChange:e=>xe(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Phone`,(0,R.jsx)(U,{value:Se,type:`tel`,autoComplete:`tel`,onChange:e=>Ce(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Mobile`,(0,R.jsx)(U,{value:we,type:`tel`,autoComplete:`tel`,onChange:e=>Te(e.target.value)})]})]}):(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsxs)(W,{children:[`Find contact`,(0,R.jsx)(U,{value:D,autoComplete:`off`,placeholder:`Search name or email`,onChange:e=>O(e.target.value)})]}),(0,R.jsx)(`div`,{className:`grid max-h-48 gap-2 overflow-y-auto rounded-md border p-2`,children:A.length?A.map(e=>{let t=e.name||``,n=e.full_name||t,r=[{key:`company`,value:e.company_name},{key:`email`,value:e.email_id},{key:`phone`,value:e.phone},{key:`mobile`,value:e.mobile_no}].filter(e=>!!e.value);return(0,R.jsxs)(`label`,{className:`flex cursor-pointer items-start gap-2 rounded-sm px-2 py-1.5 hover:bg-secondary`,children:[(0,R.jsx)(`input`,{type:`radio`,name:`erpContact`,value:t,checked:k===t,onChange:()=>me(t)}),(0,R.jsxs)(`span`,{className:`grid gap-0.5 text-sm`,children:[(0,R.jsx)(`strong`,{children:(0,R.jsx)(An,{value:n,query:D})}),r.length?(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:r.map((e,t)=>(0,R.jsxs)(`span`,{children:[t>0?` | `:``,(0,R.jsx)(An,{value:e.value,query:D})]},e.key))}):null]})]},t)}):(0,R.jsx)(`span`,{className:`px-2 py-3 text-sm text-muted-foreground`,children:`Search at least two characters.`})})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[(0,R.jsx)(`strong`,{className:`text-sm text-foreground md:col-span-2`,children:`Address`}),(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Address line 1 `,Ge?`*`:``,(0,R.jsx)(U,{value:ne,autoComplete:`address-line1`,onChange:e=>re(e.target.value)})]}),(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Address line 2`,(0,R.jsx)(U,{value:ie,autoComplete:`address-line2`,onChange:e=>ae(e.target.value)})]}),(0,R.jsxs)(W,{children:[`City`,(0,R.jsx)(U,{value:oe,autoComplete:`address-level2`,onChange:e=>se(e.target.value)})]}),(0,R.jsxs)(W,{children:[`State`,(0,R.jsx)(U,{value:ce,autoComplete:`address-level1`,onChange:e=>T(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Postal code`,(0,R.jsx)(U,{value:ue,autoComplete:`postal-code`,onChange:e=>de(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Country`,(0,R.jsx)(U,{value:le,autoComplete:`country-name`,onChange:e=>E(e.target.value)})]})]})]}):null,(0,R.jsxs)(`div`,{className:`grid gap-3 rounded-md border p-3`,children:[(0,R.jsx)(B,{type:`button`,variant:`outline`,onClick:()=>De(e=>!e),children:Ee?`Hide advanced`:`Show advanced`}),Ee?(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[r===`new`?(0,R.jsxs)(W,{children:[`Billing currency`,(0,R.jsx)(U,{value:b,autoComplete:`off`,maxLength:3,onChange:e=>x(e.target.value.toUpperCase())})]}):null,(0,R.jsxs)(W,{children:[`Cost center`,(0,R.jsx)(Vt,{value:Ae,onChange:e=>je(e.target.value),children:Oe.map(e=>{let t=e.name||``;return(0,R.jsx)(`option`,{value:t,children:[t,e.company].filter(Boolean).join(` | `)},t)})})]}),(0,R.jsxs)(W,{children:[`Activity type`,(0,R.jsx)(U,{value:Pe?Me:We,autoComplete:`off`,maxLength:140,placeholder:We||`Engineering for project`,onChange:e=>{Fe(!0),Ne(e.target.value)}})]})]}):null]})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-2`,children:[(0,R.jsx)(B,{type:`button`,variant:`outline`,onClick:e.onClose,children:`Cancel`}),(0,R.jsx)(B,{type:`button`,disabled:!qe,onClick:()=>void Je(),children:`Create project`})]})]})]})}function qn(e){let t=e.project,n=t.roster_members||[],[r,i]=(0,l.useState)(``),[a,o]=(0,l.useState)([]),[s,c]=(0,l.useState)(``),[u,d]=(0,l.useState)(``),[f,p]=(0,l.useState)(``),[m,g]=(0,l.useState)(``),_=[t.actual_start_date||t.expected_start_date,t.actual_end_date||t.expected_end_date].filter(Boolean).map(e=>Hn(e)).join(` to `)||`Not set`,v=typeof t.percent_complete==`number`?`${Math.round(t.percent_complete)}%`:`Not set`,b=a.find(e=>e.candidate_id===s),x=r.trim().includes(`@`)?r.trim().length>=5:r.trim().length>=3,ee=!!(u.trim()||f.trim()||m.trim()),te=Wn(f),S=Wn(m),C=!!((f.trim()||m.trim())&&!u.trim()),ne=!!(u.trim()&&(!f.trim()||!m.trim())),re=!!(f.trim()&&te===void 0)||!!(m.trim()&&S===void 0),ie=C||ne||re||te!==void 0&&te<0||S!==void 0&&S<0,ae=ee&&!ie?{activity_type:u.trim(),billing_rate:te,costing_rate:S}:void 0;(0,l.useEffect)(()=>{if(!e.canWrite)return;let t=r.trim();if(s&&b&&t===(b.email||b.label||``))return;if(s&&c(``),!(t.includes(`@`)?t.length>=5:t.length>=3)){o([]);return}let n=new AbortController,i=window.setTimeout(()=>{q(`/dashboard/api/project-member-candidates?query=${encodeURIComponent(t)}`,{signal:n.signal}).then(e=>o(e)).catch(e=>{e instanceof DOMException&&e.name===`AbortError`||o([])})},500);return()=>{n.abort(),window.clearTimeout(i)}},[r,e.canWrite,b,s]);function se(e){c(e.candidate_id),i(e.email||e.label||e.full_name||r)}return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-[auto_minmax(0,1fr)_auto] md:items-start`,children:[(0,R.jsxs)(B,{type:`button`,variant:`outline`,onClick:e.onBack,children:[(0,R.jsx)(h,{}),`Projects`]}),(0,R.jsxs)(`div`,{className:`min-w-0`,children:[(0,R.jsx)(zt,{children:t.display_name}),(0,R.jsxs)(`div`,{className:`mt-2 flex flex-wrap items-center gap-2 text-sm text-muted-foreground`,children:[(0,R.jsx)(z,{variant:Vn(t.source_status),children:t.source_status||`Unknown`}),t.erpnext_project_id?(0,R.jsx)(`span`,{className:`font-mono`,children:t.erpnext_project_id}):null,t.last_synced_at?(0,R.jsxs)(`span`,{children:[`Synced `,qt(t.last_synced_at)]}):null]})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-start gap-2 md:justify-end`,children:[e.canWrite?(0,R.jsxs)(Vt,{className:`w-[160px]`,"aria-label":`Status for ${t.display_name}`,value:t.source_status||``,disabled:e.loading[`project:${t.id}:status`],onChange:n=>e.onUpdateStatus(t.id,n.target.value),children:[(0,R.jsx)(`option`,{value:`Open`,children:`Open`}),(0,R.jsx)(`option`,{value:`Completed`,children:`Completed`}),(0,R.jsx)(`option`,{value:`Cancelled`,children:`Cancelled`})]}):null,t.erpnext_project_url?(0,R.jsxs)(`a`,{className:`inline-flex min-h-9 items-center justify-center gap-2 rounded-md border bg-secondary px-3 text-sm font-semibold`,href:t.erpnext_project_url,target:`_blank`,rel:`noreferrer`,children:[(0,R.jsx)(y,{className:`size-4`}),`ERP project`]}):null,t.customer_erpnext_url?(0,R.jsxs)(`a`,{className:`inline-flex min-h-9 items-center justify-center gap-2 rounded-md border bg-secondary px-3 text-sm font-semibold`,href:t.customer_erpnext_url,target:`_blank`,rel:`noreferrer`,children:[(0,R.jsx)(y,{className:`size-4`}),`ERP customer`]}):null]})]})}),(0,R.jsxs)(Bt,{className:`grid gap-4 md:grid-cols-2 lg:grid-cols-4`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Customer`}),(0,R.jsx)(`strong`,{className:`block`,children:t.customer||`None`})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Timeline`}),(0,R.jsx)(`strong`,{className:`block`,children:_})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Progress`}),(0,R.jsx)(`strong`,{className:`block`,children:v})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Linked Gigs`}),(0,R.jsx)(`strong`,{className:`block`,children:t.linked_engagement_count||0})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`ERP Type`}),(0,R.jsx)(`div`,{className:`mt-1`,children:t.project_type?(0,R.jsx)(z,{variant:`neutral`,children:t.project_type}):(0,R.jsx)(`strong`,{className:`block`,children:`Not set`})})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`ERP Modified`}),(0,R.jsx)(`strong`,{className:`block`,children:qt(t.source_modified_at)})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Cache ID`}),(0,R.jsx)(`strong`,{className:`block break-all font-mono text-xs`,children:t.id})]})]})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Project roster`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:n.length?`${n.length} synced ERP user${n.length===1?``:`s`}`:`No ERP roster`})]}),e.canWrite?(0,R.jsxs)(Bt,{className:`grid gap-3 border-b md:grid-cols-[minmax(260px,1fr)_minmax(180px,.7fr)_minmax(130px,.45fr)_minmax(130px,.45fr)_auto_auto] md:items-end`,children:[(0,R.jsxs)(`div`,{className:`relative`,children:[(0,R.jsxs)(W,{children:[`Person search`,(0,R.jsx)(U,{value:r,autoComplete:`off`,placeholder:`Search @508.dev person`,onChange:e=>i(e.target.value),onKeyDown:e=>{e.key===`Enter`&&(e.preventDefault(),a.length===1&&se(a[0]))}})]}),x&&!s?(0,R.jsx)(`div`,{className:`absolute z-20 mt-1 max-h-64 w-full overflow-auto rounded-md border bg-background shadow-lg`,children:a.length?a.map(e=>(0,R.jsxs)(`button`,{type:`button`,className:`grid w-full gap-0.5 px-3 py-2 text-left hover:bg-secondary focus:bg-secondary focus:outline-none`,onClick:()=>se(e),children:[(0,R.jsx)(`span`,{className:`truncate text-sm font-bold`,children:e.label||e.full_name||e.email||`Person`}),(0,R.jsx)(`span`,{className:`truncate text-xs text-muted-foreground`,children:[e.email,e.sources?.join(`, `)].filter(Boolean).join(` | `)})]},e.candidate_id)):(0,R.jsx)(`div`,{className:`px-3 py-2 text-sm text-muted-foreground`,children:`No verified @508.dev results`})}):null]}),(0,R.jsxs)(W,{children:[`Activity Type`,(0,R.jsx)(U,{value:u,autoComplete:`off`,placeholder:`Optional rate step`,onChange:e=>d(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Billing rate`,(0,R.jsx)(U,{value:f,inputMode:`decimal`,autoComplete:`off`,placeholder:`USD/hr`,onChange:e=>p(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Costing rate`,(0,R.jsx)(U,{value:m,inputMode:`decimal`,autoComplete:`off`,placeholder:`USD/hr`,onChange:e=>g(e.target.value)})]}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,disabled:e.loading[`project:${t.id}:user`]||!s||!b?.email||ie,onClick:()=>void e.onAddUser(t.id,b?.email||r,s,ae).then(e=>{e&&(i(``),o([]),c(``),d(``),p(``),g(``))}),children:[(0,R.jsx)(ce,{}),`Add ERP user`]}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,disabled:e.loading[`project:${t.id}:historical`]||!r.trim(),onClick:()=>void e.onAddHistoricalMember(t.id,r).then(e=>{e&&i(``)}),children:[(0,R.jsx)(ce,{}),`Add historical`]})]}):null,(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{className:`min-w-[860px]`,"aria-label":`Project roster`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{children:`Name`}),(0,R.jsx)(G,{children:`Email`}),(0,R.jsx)(G,{children:`ERP user`}),(0,R.jsx)(G,{children:`Links`}),(0,R.jsx)(G,{children:`Source`}),(0,R.jsx)(G,{children:`Last seen`}),e.canWrite?(0,R.jsx)(G,{children:`Actions`}):null]})}),(0,R.jsx)(Wt,{children:n.length?n.map(n=>{let r=Un(n),i=n.source_user_id||n.email||``,a=n.roster_kind===`historical`||n.source===`manual`;return(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:(0,R.jsx)(`strong`,{children:n.full_name||n.email||n.source_user_id})}),(0,R.jsx)(K,{children:n.email||`None`}),(0,R.jsx)(K,{className:`font-mono text-xs`,children:n.erpnext_user_url?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-semibold text-primary underline-offset-4 hover:underline`,href:n.erpnext_user_url,target:`_blank`,rel:`noreferrer`,children:[n.source_user_id||`ERP user`,(0,R.jsx)(y,{className:`size-3.5`})]}):n.source_user_id||`Unknown`}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-2`,children:[n.supplier_erpnext_url?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-semibold text-primary underline-offset-4 hover:underline`,href:n.supplier_erpnext_url,target:`_blank`,rel:`noreferrer`,children:[`Supplier`,(0,R.jsx)(y,{className:`size-3.5`})]}):null,n.crm_contact_id&&e.crmContactUrl(n.crm_contact_id)?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-semibold text-primary underline-offset-4 hover:underline`,href:e.crmContactUrl(n.crm_contact_id),target:`_blank`,rel:`noreferrer`,children:[`CRM`,(0,R.jsx)(y,{className:`size-3.5`})]}):null,!n.supplier_erpnext_url&&!(n.crm_contact_id&&e.crmContactUrl(n.crm_contact_id))?(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:`None`}):null]})}),(0,R.jsx)(K,{children:n.roster_kind||n.source||`ERP`}),(0,R.jsx)(K,{children:qt(n.last_seen_at)}),e.canWrite?(0,R.jsx)(K,{children:(0,R.jsxs)(B,{type:`button`,variant:`outline`,size:`sm`,disabled:!i||e.loading[`project:${t.id}:${a?`historical`:`user`}`],onClick:()=>{window.confirm(`Remove ${r} from this project roster?`)&&(a?e.onRemoveHistoricalMember(t.id,i):e.onRemoveUser(t.id,i))},children:[(0,R.jsx)(oe,{}),`Remove`]})}):null]},`${n.source||``}:${n.source_user_id||n.email}`)}):(0,R.jsx)(Gt,{children:(0,R.jsx)(K,{colSpan:e.canWrite?7:6,className:`text-sm text-muted-foreground`,children:`No roster rows have been synced for this project.`})})})]})})]})]})}function Jn(e){let t=e.gigs.reduce((t,n)=>(t.total+=1,t.applications+=Number(n.application_count||0),t.interested+=Number(n.interested_count||0),Bn(n,e.staleDays)!==null&&(t.stale+=1),t),{total:0,applications:0,interested:0,stale:0}),n=(0,R.jsxs)(V,{className:`grid gap-3 p-4 md:grid-cols-[minmax(140px,.75fr)_minmax(220px,1.25fr)_auto_auto_auto] md:items-end`,children:[(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{id:`gigStatus`,value:e.status,onChange:t=>e.setStatus(t.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Any status`}),In.map(e=>(0,R.jsx)(`option`,{value:e,children:Rn(e)},e))]})]}),(0,R.jsxs)(W,{children:[`Search gigs`,(0,R.jsx)(U,{id:`gigQuery`,value:e.query,autoComplete:`off`,placeholder:`Title, gig text, #tag, @poster`,onChange:t=>e.setQuery(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onRefresh()})]}),e.canIncludeHistorical?(0,R.jsxs)(`label`,{className:`flex min-h-9 items-center gap-2 text-xs font-bold text-muted-foreground`,children:[(0,R.jsx)(`input`,{type:`checkbox`,checked:e.includeHistorical,onChange:t=>e.setIncludeHistorical(t.target.checked)}),`Include historical`]}):null,(0,R.jsxs)(B,{id:`searchGigs`,type:`button`,onClick:e.onRefresh,disabled:e.loading.gigs,children:[(0,R.jsx)(ne,{}),`Search`]}),(0,R.jsxs)(B,{id:`refreshGigs`,type:`button`,variant:`outline`,onClick:e.onRefresh,disabled:e.loading.gigs,children:[(0,R.jsx)(C,{}),`Refresh`]}),e.gigs.length>=e.limit?(0,R.jsx)(B,{type:`button`,variant:`outline`,onClick:()=>e.setLimit(Math.min(e.limit+100,500)),disabled:e.loading.gigs||e.limit>=500,children:`Load more`}):null]}),r=e.selectedGigId?e.loading[`gig:${e.selectedGigId}:detail`]:!1;return e.selectedGigId&&!e.selectedGig&&(e.loading.gigs||r)?(0,R.jsxs)(R.Fragment,{children:[n,(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Gig detail`})}),(0,R.jsx)(Bt,{className:`text-sm text-muted-foreground`,children:`Loading gig.`})]})]}):e.selectedGigId&&!e.selectedGig?(0,R.jsxs)(R.Fragment,{children:[n,(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Gig detail`})}),(0,R.jsxs)(Bt,{className:`grid gap-3`,children:[(0,R.jsx)(`p`,{className:`text-sm text-muted-foreground`,children:`This gig is not in the current result set. Clear filters or refresh the gig list.`}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,onClick:e.onCloseGig,children:[(0,R.jsx)(h,{}),`Back to gigs`]})]})]})]}):e.selectedGig?(0,R.jsxs)(R.Fragment,{children:[n,(0,R.jsx)(Zn,{gig:e.selectedGig,loading:e.loading,canWrite:e.canWrite,crmContactUrl:e.crmContactUrl,crmAttachmentUrl:e.crmAttachmentUrl,staleDays:e.staleDays,onBack:e.onCloseGig,onUpdateStatus:e.onUpdateStatus,onAddApplication:e.onAddApplication,onUpdateApplicationStatus:e.onUpdateApplicationStatus})]}):(0,R.jsxs)(R.Fragment,{children:[n,(0,R.jsxs)(`section`,{className:`grid gap-3 md:grid-cols-4`,"aria-label":`Gig summary`,children:[(0,R.jsx)(On,{id:`gigMetricTotal`,label:`Gigs`,value:t.total}),(0,R.jsx)(On,{id:`gigMetricCandidates`,label:`Candidates`,value:t.applications}),(0,R.jsx)(On,{id:`gigMetricInterested`,label:`Interested`,value:t.interested}),(0,R.jsx)(On,{id:`gigMetricStale`,label:`Stale recruiting`,value:t.stale})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Discord gigs`}),(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center justify-end gap-2`,children:[(0,R.jsxs)(B,{type:`button`,variant:`ghost`,size:`sm`,onClick:()=>e.onSort(`activity`),"aria-label":`Sort gigs by activity`,children:[`Activity`,` `,e.sort.key===`activity`?e.sort.direction===`asc`?`↑`:`↓`:``]}),(0,R.jsxs)(B,{type:`button`,variant:`ghost`,size:`sm`,onClick:()=>e.onSort(`title`),"aria-label":`Sort gigs by title`,children:[`Title `,e.sort.key===`title`?e.sort.direction===`asc`?`↑`:`↓`:``]}),(0,R.jsx)(`span`,{id:`gigsStatus`,className:`text-sm text-muted-foreground`,children:e.loading.gigs?`Loading`:`${e.gigs.length} shown`})]})]}),(0,R.jsx)(kn,{hidden:e.gigs.length!==0,children:`No gigs match this view.`}),(0,R.jsx)(`div`,{id:`gigsBody`,className:L(`grid gap-3 p-4`,e.gigs.length===0&&`hidden`),children:e.gigs.map(t=>(0,R.jsx)(Yn,{gig:t,loading:e.loading,canWrite:e.canWrite,staleDays:e.staleDays,onOpenGig:e.onOpenGig,onUpdateStatus:e.onUpdateStatus},t.id))})]})]})}function Yn({gig:e,loading:t,canWrite:n,onOpenGig:r,onUpdateStatus:i,staleDays:a}){let o=Array.isArray(e.applications)?e.applications:[],s=e.status===`recruiting`,c=e.discord_guild_id&&e.discord_thread_id?`https://discord.com/channels/${encodeURIComponent(e.discord_guild_id)}/${encodeURIComponent(e.discord_thread_id)}`:``,l=Bn(e,a);return(0,R.jsxs)(`article`,{className:L(`grid gap-4 rounded-md border bg-background p-4 lg:grid-cols-[minmax(0,1fr)_220px_180px] lg:items-start`,!s&&`border-l-4 border-l-muted-foreground/60 bg-secondary/45`),children:[(0,R.jsxs)(`div`,{className:`min-w-0`,children:[(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-2`,children:[(0,R.jsx)(`a`,{className:`text-base font-extrabold text-primary`,href:`/dashboard/gigs/${encodeURIComponent(e.id)}`,onClick:t=>{t.preventDefault(),r(e.id)},children:e.title||`Untitled gig`}),(0,R.jsx)(z,{variant:e.status===`filled`?`succeeded`:e.status===`lost`?`failed`:s?`queued`:`neutral`,children:e.status_label||Rn(e.status)}),s?null:(0,R.jsx)(z,{variant:`neutral`,children:`Not recruiting`}),l===null?null:(0,R.jsxs)(z,{variant:`running`,children:[l,`d stale`]})]}),(0,R.jsxs)(`div`,{className:`mt-2 flex flex-wrap gap-1.5`,children:[e.posting_type?(0,R.jsx)(z,{variant:`neutral`,children:Rn(e.posting_type)}):null,e.discord_channel_name?(0,R.jsxs)(z,{variant:`neutral`,children:[`#`,e.discord_channel_name]}):null,(e.required_skills||[]).slice(0,5).map(e=>(0,R.jsx)(z,{variant:`queued`,children:e},e)),(e.preferred_skills||[]).slice(0,3).map(e=>(0,R.jsx)(z,{variant:`neutral`,children:e},e))]}),(0,R.jsxs)(`div`,{className:`mt-3 flex flex-wrap gap-x-4 gap-y-1 text-sm text-muted-foreground`,children:[(0,R.jsxs)(`span`,{children:[`Activity `,qt(zn(e))||`unknown`]}),(0,R.jsxs)(`span`,{children:[`Posted `,qt(e.posted_at)||`unknown`]}),c?(0,R.jsx)(`a`,{className:`font-extrabold text-primary`,href:c,target:`_blank`,rel:`noreferrer`,children:`Open Discord thread`}):null]})]}),(0,R.jsxs)(`div`,{className:`grid grid-cols-2 gap-2 text-sm lg:grid-cols-1`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`block text-xs font-bold text-muted-foreground`,children:`People`}),(0,R.jsx)(`strong`,{children:e.application_count||o.length}),(0,R.jsxs)(`span`,{className:`ml-2 text-muted-foreground`,children:[Number(e.interested_count||0),` interested`]})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`block text-xs font-bold text-muted-foreground`,children:`Top candidates`}),(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:o.slice(0,3).map(e=>Xn(e)).join(`, `)||`None yet`})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-2`,children:[n?(0,R.jsx)(Vt,{"aria-label":`Status for ${e.title||`gig`}`,value:e.status,disabled:t[`gig:${e.id}:status`],onChange:t=>i(e.id,t.target.value),children:In.map(e=>(0,R.jsx)(`option`,{value:e,children:Rn(e)},e))}):null,(0,R.jsx)(B,{type:`button`,onClick:()=>r(e.id),children:`Manage people`})]})]})}function Xn(e){return e.name||e.email_508||e.discord_username||(typeof e.evaluation?.discord_username==`string`?e.evaluation.discord_username:``)||`Candidate`}function Zn({gig:e,loading:t,canWrite:n,crmContactUrl:r,crmAttachmentUrl:i,staleDays:a,onBack:o,onUpdateStatus:s,onAddApplication:c,onUpdateApplicationStatus:u}){let[d,f]=(0,l.useState)(``),p=Array.isArray(e.applications)?e.applications:[],m=e.status===`recruiting`,g=e.discord_guild_id&&e.discord_thread_id?`https://discord.com/channels/${encodeURIComponent(e.discord_guild_id)}/${encodeURIComponent(e.discord_thread_id)}`:``,_=Bn(e,a);return(0,R.jsxs)(`div`,{className:`grid gap-5`,children:[(0,R.jsxs)(V,{className:L(!m&&`border-l-4 border-l-muted-foreground/60 bg-secondary/35`),children:[(0,R.jsxs)(H,{className:`items-start`,children:[(0,R.jsxs)(`div`,{className:`grid gap-2`,children:[(0,R.jsxs)(B,{type:`button`,variant:`ghost`,size:`sm`,className:`w-fit`,onClick:o,children:[(0,R.jsx)(h,{}),`Back to gigs`]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(zt,{className:`text-xl`,children:e.title||`Untitled gig`}),(0,R.jsxs)(`div`,{className:`mt-2 flex flex-wrap gap-1.5`,children:[(0,R.jsx)(z,{variant:e.status===`filled`?`succeeded`:e.status===`lost`?`failed`:m?`queued`:`neutral`,children:e.status_label||Rn(e.status)}),m?null:(0,R.jsx)(z,{variant:`neutral`,children:`Not recruiting`}),_===null?null:(0,R.jsxs)(z,{variant:`running`,children:[_,`d stale`]}),e.posting_type?(0,R.jsx)(z,{variant:`neutral`,children:Rn(e.posting_type)}):null,e.discord_channel_name?(0,R.jsxs)(z,{variant:`neutral`,children:[`#`,e.discord_channel_name]}):null,(e.required_skills||[]).map(e=>(0,R.jsx)(z,{variant:`queued`,children:e},e)),(e.preferred_skills||[]).map(e=>(0,R.jsx)(z,{variant:`neutral`,children:e},e))]})]})]}),(0,R.jsxs)(`div`,{className:`grid min-w-[190px] gap-2`,children:[n?(0,R.jsxs)(W,{children:[`Gig status`,(0,R.jsx)(Vt,{"aria-label":`Status for ${e.title||`gig`}`,value:e.status,disabled:t[`gig:${e.id}:status`],onChange:t=>s(e.id,t.target.value),children:In.map(e=>(0,R.jsx)(`option`,{value:e,children:Rn(e)},e))})]}):null,g?(0,R.jsxs)(`a`,{className:`inline-flex min-h-9 items-center justify-center gap-2 rounded-md border bg-secondary px-3 text-sm font-semibold`,href:g,target:`_blank`,rel:`noreferrer`,children:[(0,R.jsx)(y,{className:`size-4`}),`Discord thread`]}):null]})]}),(0,R.jsxs)(Bt,{className:`grid gap-4 lg:grid-cols-[1fr_1fr_1fr]`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Activity`}),(0,R.jsx)(`strong`,{className:`block`,children:qt(zn(e))||`unknown`}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[`Posted `,qt(e.posted_at)||`unknown`]})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`People`}),(0,R.jsx)(`strong`,{className:`block`,children:e.application_count||p.length}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[Number(e.interested_count||0),` interested`]})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Discord`}),(0,R.jsx)(`strong`,{className:`block`,children:e.discord_channel_name||`No channel`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.discord_thread_id?`Thread ${e.discord_thread_id}`:`No thread`})]})]})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`People`}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[p.length,` candidate`,p.length===1?``:`s`]})]}),n?(0,R.jsxs)(`form`,{className:`grid gap-2 border-t p-4 md:grid-cols-[minmax(220px,1fr)_auto]`,onSubmit:t=>{t.preventDefault(),c(e.id,d).then(e=>{e&&f(``)})},children:[(0,R.jsxs)(W,{className:`min-w-0`,children:[`CRM profile`,(0,R.jsx)(U,{value:d,onChange:e=>f(e.target.value),placeholder:`https://crm.508.dev/#Contact/view/...`,"aria-label":`CRM profile for candidate`})]}),(0,R.jsxs)(B,{type:`submit`,className:`self-end`,disabled:t[`gig:${e.id}:addCandidate`]||!d.trim(),children:[(0,R.jsx)(se,{}),`Add candidate`]})]}):null,(0,R.jsx)(kn,{hidden:p.length!==0,children:`No suggested or interested people yet.`}),(0,R.jsx)(`div`,{className:L(`grid gap-3 p-4`,p.length===0&&`hidden`),children:p.map(a=>(0,R.jsx)(Qn,{gigId:e.id,application:a,loading:t,canWrite:n,crmContactUrl:r,crmAttachmentUrl:i,onUpdateApplicationStatus:u},a.id))})]})]})}function Qn({gigId:e,application:t,loading:n,canWrite:r,crmContactUrl:i,crmAttachmentUrl:a,onUpdateApplicationStatus:o}){let s=Xn(t),c=i(t.crm_contact_id),l=a(t.latest_resume_id),u=typeof t.fit_score==`number`?`${Math.round(t.fit_score)}/100`:typeof t.match_score==`number`?t.match_score.toFixed(1):``,d=typeof t.evaluation?.llm_summary==`string`?t.evaluation.llm_summary:``;return(0,R.jsxs)(`div`,{className:`grid gap-2 rounded-md border bg-background p-2`,children:[(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-2`,children:[c?(0,R.jsx)(`a`,{className:`font-extrabold text-primary`,href:c,target:`_blank`,rel:`noopener noreferrer`,children:s}):(0,R.jsx)(`strong`,{children:s}),(0,R.jsx)(z,{variant:t.status===`interested`?`succeeded`:`neutral`,children:Rn(t.status)}),(0,R.jsx)(z,{variant:`neutral`,children:Rn(t.source||`manual_add`)}),u?(0,R.jsxs)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:[`Fit `,u]}):null,c?(0,R.jsx)(`a`,{className:`text-xs font-extrabold text-primary`,href:c,target:`_blank`,rel:`noopener noreferrer`,"aria-label":`Open ${s} CRM profile`,children:`CRM profile`}):null,l?(0,R.jsx)(`a`,{className:`text-xs font-extrabold text-primary`,href:l,target:`_blank`,rel:`noopener noreferrer`,children:`Resume`}):null]}),d?(0,R.jsx)(`div`,{className:`text-xs text-muted-foreground`,children:d}):null,r?(0,R.jsx)(Vt,{"aria-label":`Candidate status for ${s}`,value:t.status||`suggested`,disabled:n[`application:${t.id}:status`],onChange:n=>o(e,t.id,n.target.value),children:Ln.map(e=>(0,R.jsx)(`option`,{value:e,children:Rn(e)},e))}):null]})}function $n(e){let t=pn[e.peopleFilterKind]?.options||[];return(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`People lookup`}),(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center justify-end gap-2`,children:[e.canSync?(0,R.jsxs)(B,{id:`syncPeople`,"data-permission":`people:sync`,type:`button`,onClick:e.onSync,disabled:e.loading.syncPeople,children:[(0,R.jsx)(C,{}),`Sync people`]}):null,e.crmBaseUrl?(0,R.jsx)(`a`,{id:`crmHomeLink`,className:`text-sm font-extrabold text-primary`,href:e.crmBaseUrl,target:`_blank`,rel:`noreferrer`,children:`Open CRM`}):null,(0,R.jsx)(`span`,{id:`peopleStatus`,className:`text-sm text-muted-foreground`,children:e.loading.people?`Loading`:`${e.people.length} shown`})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 border-b p-4 md:grid-cols-[minmax(0,1fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`Search CRM people cache`,(0,R.jsx)(U,{id:`peopleQuery`,value:e.peopleQuery,autoComplete:`off`,placeholder:`Name, email, CRM id, Discord, resume`,onChange:t=>e.setPeopleQuery(t.target.value),onKeyDown:t=>{t.key===`Enter`&&e.onSearch()}})]}),(0,R.jsxs)(B,{id:`searchPeople`,type:`button`,onClick:e.onSearch,disabled:e.loading.people,children:[(0,R.jsx)(ne,{}),`Search`]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 border-b bg-background p-4 md:grid-cols-[minmax(120px,.7fr)_minmax(150px,1fr)_minmax(150px,1fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`Member`,(0,R.jsxs)(Vt,{id:`peopleMember`,value:e.peopleMember,onChange:t=>e.setPeopleMember(t.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Any`}),(0,R.jsx)(`option`,{value:`true`,children:`Member`}),(0,R.jsx)(`option`,{value:`false`,children:`Not member`})]})]}),(0,R.jsxs)(W,{children:[`Add filter`,(0,R.jsx)(Vt,{id:`peopleFilterKind`,value:e.peopleFilterKind,disabled:e.peopleFilterKeys.length===0,onChange:t=>e.setPeopleFilterKind(t.target.value),children:e.peopleFilterKeys.map(e=>(0,R.jsx)(`option`,{value:e,children:pn[e].label},e))})]}),(0,R.jsxs)(W,{children:[`Value`,(0,R.jsx)(Vt,{id:`peopleFilterValue`,value:e.peopleFilterValue,onChange:t=>e.setPeopleFilterValue(t.target.value),children:t.map(([e,t])=>(0,R.jsx)(`option`,{value:e,children:t},e))})]}),(0,R.jsx)(B,{id:`addPeopleFilter`,type:`button`,onClick:e.addFilter,disabled:e.peopleFilterKeys.length===0,children:`Add filter`}),(0,R.jsx)(`div`,{id:`activePeopleFilters`,className:`md:col-span-4`,children:(0,R.jsx)(Fn,{filters:e.peopleFilters,onRemove:e.removeFilter})})]}),(0,R.jsx)(kn,{hidden:e.people.length!==0,children:`No people match this lookup.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`peopleTable`,className:L(`min-w-[900px]`,e.people.length===0&&`hidden`),"aria-label":`People lookup results`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(Dn,{className:`w-[27%]`,label:`Name`,scope:`people`,sort:e.sort,sortKey:`name`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[28%]`,label:`Status`,scope:`people`,sort:e.sort,sortKey:`status`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[20%]`,label:`Discord`,scope:`people`,sort:e.sort,sortKey:`discord`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[25%]`,label:`Resume / skills`,scope:`people`,sort:e.sort,sortKey:`resume`,onSort:(t,n)=>e.onSort(n)})]})}),(0,R.jsx)(Wt,{id:`peopleBody`,children:e.people.map(t=>{let n=t.name||t.email_508||t.email||`CRM contact`,r=e.crmContactUrl(t.crm_contact_id),i=t.profile_status||{},a=Number(i.skills_count||0),o=e.crmAttachmentUrl(t.latest_resume_id);return(0,R.jsxs)(Gt,{children:[(0,R.jsxs)(K,{children:[r?(0,R.jsx)(`a`,{className:`font-extrabold text-primary`,href:r,target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${n} in CRM`,children:n}):(0,R.jsx)(`strong`,{children:n}),(0,R.jsx)(`div`,{className:`text-sm text-muted-foreground`,children:[t.email_508||t.email,t.contact_type].filter(Boolean).join(` | `)})]}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-1.5`,children:[i.crm_active?null:(0,R.jsx)(z,{variant:`missing`,children:t.sync_status||`CRM sync issue`}),(0,R.jsx)(z,{variant:i.is_member?`succeeded`:`missing`,children:i.is_member?`Member`:`Missing Member`}),(0,R.jsx)(z,{variant:i.discord_linked?`succeeded`:`missing`,children:i.discord_linked?`Discord`:`Missing Discord`}),(0,R.jsx)(z,{variant:i.email_508?`succeeded`:`missing`,children:i.email_508?`508 email`:`Missing 508 email`}),i.latest_resume?null:(0,R.jsx)(z,{variant:`missing`,children:`Missing Resume`})]})}),(0,R.jsx)(K,{children:[t.discord_username,t.discord_user_id].filter(Boolean).join(` | `)||`Not linked`}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-1.5`,children:[o?(0,R.jsx)(`a`,{className:`inline-flex min-h-7 items-center rounded-md border bg-secondary px-2 text-xs font-extrabold`,href:o,target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${n} resume`,children:`Resume`}):(0,R.jsx)(`span`,{children:t.latest_resume_name||t.latest_resume_id||`No resume`}),(0,R.jsx)(z,{variant:a>0?`succeeded`:`missing`,children:a>0?`Skills parsed`:`Skills not parsed`})]})})]},t.crm_contact_id||n)})})]})})]})}function er(e){let t=pn[e.onboardingFilterKind]?.options||[];return(0,R.jsxs)(R.Fragment,{children:[e.canWrite?(0,R.jsx)(or,{loading:e.loading.engineerSetup,onSetup:e.onSetupEngineer}):null,(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Onboarding queue`}),(0,R.jsx)(`span`,{id:`onboardingStatus`,className:`text-sm text-muted-foreground`,children:e.loading.onboarding?`Loading`:`${e.people.length} shown`})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 border-b p-4 md:grid-cols-[minmax(0,1fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`Search prospects`,(0,R.jsx)(U,{id:`onboardingQuery`,value:e.onboardingQuery,autoComplete:`off`,placeholder:`Name, email, Discord, onboarder`,onChange:t=>e.setOnboardingQuery(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onSearch()})]}),(0,R.jsxs)(B,{id:`searchOnboarding`,type:`button`,onClick:e.onSearch,disabled:e.loading.onboarding,children:[(0,R.jsx)(ne,{}),`Search`]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 border-b bg-background p-4 md:grid-cols-[minmax(140px,.8fr)_minmax(150px,1fr)_minmax(150px,1fr)_minmax(120px,.7fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{id:`onboardingState`,value:e.onboardingState,onChange:t=>e.setOnboardingState(t.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Any state`}),hn.map(([e,t])=>(0,R.jsx)(`option`,{value:e,children:t},e))]})]}),(0,R.jsxs)(W,{children:[`Onboarder`,(0,R.jsx)(U,{id:`onboarderFilter`,value:e.onboarderFilter,autoComplete:`off`,placeholder:`Any onboarder`,onChange:t=>e.setOnboarderFilter(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onSearch()})]}),(0,R.jsxs)(W,{children:[`Add filter`,(0,R.jsx)(Vt,{id:`onboardingFilterKind`,value:e.onboardingFilterKind,disabled:e.onboardingFilterKeys.length===0,onChange:t=>e.setOnboardingFilterKind(t.target.value),children:e.onboardingFilterKeys.map(e=>(0,R.jsx)(`option`,{value:e,children:pn[e].label},e))})]}),(0,R.jsxs)(W,{children:[`Value`,(0,R.jsx)(Vt,{id:`onboardingFilterValue`,value:e.onboardingFilterValue,onChange:t=>e.setOnboardingFilterValue(t.target.value),children:t.map(([e,t])=>(0,R.jsx)(`option`,{value:e,children:t},e))})]}),(0,R.jsx)(B,{id:`addOnboardingFilter`,type:`button`,onClick:e.addFilter,disabled:e.onboardingFilterKeys.length===0,children:`Add filter`}),(0,R.jsx)(`div`,{id:`activeOnboardingFilters`,className:`md:col-span-5`,children:(0,R.jsx)(Fn,{filters:e.onboardingFilters,onRemove:e.removeFilter,suffix:`onboarding filter`})})]}),(0,R.jsx)(kn,{hidden:e.people.length!==0,children:`No prospects match this queue view.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`onboardingTable`,className:L(`min-w-[1340px]`,e.people.length===0&&`hidden`),"aria-label":`Onboarding queue`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(Dn,{className:`w-[18%]`,label:`Name`,scope:`onboarding`,sort:e.sort,sortKey:`name`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[12%]`,label:`Status`,scope:`onboarding`,sort:e.sort,sortKey:`onboarding_state`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[20%]`,label:`Onboarder`,scope:`onboarding`,sort:e.sort,sortKey:`onboarder`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[12%]`,label:`Updated`,scope:`onboarding`,sort:e.sort,sortKey:`updated`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(G,{className:`w-[15%]`,children:`Email`}),(0,R.jsx)(G,{className:`w-[12%]`,children:`Links`}),(0,R.jsx)(Dn,{className:`w-[11%]`,label:`Needs`,scope:`onboarding`,sort:e.sort,sortKey:`profile_gaps`,onSort:(t,n)=>e.onSort(n)})]})}),(0,R.jsx)(Wt,{id:`onboardingBody`,children:e.people.map(t=>(0,R.jsx)(sr,{person:t,loading:e.loading,canWrite:e.canWrite,onAssign:e.onAssign,onStatusChange:e.onStatusChange,onDraftEmail:e.onDraftEmail,onSendEmail:e.onSendEmail,crmContactUrl:e.crmContactUrl,crmAttachmentUrl:e.crmAttachmentUrl,canConfigure:e.canConfigure,onOpenConfiguration:e.onOpenConfiguration},t.crm_contact_id||t.name))})]})})]})]})}var tr=[`Female`,`Genderqueer`,`Male`,`Non-Conforming`,`Other`,`Prefer not to say`,`Transgender`],nr=[`Company Email`,`Personal Email`,`User ID`];function rr(e){let t=(e||``).trim().split(/\s+/).filter(Boolean);return t.length===0?{first:``,middle:``,last:``}:t.length===1?{first:t[0],middle:``,last:``}:t.length===2?{first:t[0],middle:``,last:t[1]}:{first:t[0],middle:t.slice(1,-1).join(` `),last:t[t.length-1]}}function ir(e){let t=(e.email||``).trim();return!t||t.toLowerCase().endsWith(`@508.dev`)?``:t}function ar(e){let t=(e.email_508||``).trim();if(t)return t;let n=(e.email||``).trim();return n.toLowerCase().endsWith(`@508.dev`)?n:``}function or({loading:e,onSetup:t}){let[n,r]=(0,l.useState)(``),[i,a]=(0,l.useState)([]),[o,s]=(0,l.useState)(!1),[c,u]=(0,l.useState)(``),[d,f]=(0,l.useState)(``),[p,m]=(0,l.useState)(``),[h,g]=(0,l.useState)(``),[_,v]=(0,l.useState)(``),[y,b]=(0,l.useState)(``),[x,ee]=(0,l.useState)(``),[te,S]=(0,l.useState)(``),[C,re]=(0,l.useState)(``),[ie,ae]=(0,l.useState)(``),[oe,ce]=(0,l.useState)(``);function w(e){let t=rr(e.name);m(t.first),g(t.middle),v(t.last),f(ar(e)),ae(ir(e)),b(e.address_country||``),r(e.name||e.email_508||e.email||``),a([]),u(``)}async function T(){let e=n.trim();if(e){s(!0),u(``);try{a(await q(`/dashboard/api/people?${new URLSearchParams({limit:`8`,query:e}).toString()}`))}catch(e){u(bn(e,`Unable to search people`)),a([])}finally{s(!1)}}}async function le(){let e={email:d,first_name:p,middle_name:h,last_name:_,country:y,personal_email:ie};x.trim()&&(e.gender=x),te.trim()&&(e.date_of_birth=te),C.trim()&&(e.date_of_joining=C),oe.trim()&&(e.prefered_email=oe),await t(e)&&(r(``),a([]),f(``),m(``),g(``),v(``),b(``),ee(``),S(``),re(``),ae(``),ce(``))}return(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Engineer setup`})}),(0,R.jsx)(Bt,{children:(0,R.jsxs)(`form`,{className:`grid gap-3`,onSubmit:e=>{e.preventDefault(),le()},children:[(0,R.jsxs)(`div`,{className:`grid gap-3 border-b pb-3 md:grid-cols-[minmax(0,1fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`CRM person`,(0,R.jsx)(U,{value:n,autoComplete:`off`,placeholder:`Search name or email`,onChange:e=>r(e.target.value),onKeyDown:e=>{e.key===`Enter`&&(e.preventDefault(),T())}})]}),(0,R.jsxs)(B,{type:`button`,onClick:T,disabled:o||!n.trim(),children:[(0,R.jsx)(ne,{}),`Search`]}),c?(0,R.jsx)(`span`,{className:`text-sm font-semibold text-destructive`,children:c}):null,i.length>0?(0,R.jsx)(`div`,{className:`grid gap-2 md:col-span-2`,children:i.map(e=>{let t=e.name||e.email_508||e.email||e.crm_contact_id,n=[e.email_508||e.email,e.contact_type].filter(Boolean).join(` | `);return(0,R.jsxs)(`button`,{type:`button`,className:`grid rounded-md border bg-background px-3 py-2 text-left text-sm hover:border-primary`,onClick:()=>w(e),children:[(0,R.jsx)(`strong`,{children:t}),n?(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:n}):null]},e.crm_contact_id||t)})}):null]}),(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-[minmax(0,1fr)_minmax(0,1fr)_minmax(130px,.6fr)]`,children:[(0,R.jsxs)(W,{children:[`Company email`,(0,R.jsx)(U,{value:d,autoComplete:`off`,placeholder:`engineer@508.dev`,onChange:e=>f(e.target.value)})]}),(0,R.jsxs)(W,{children:[`First name`,(0,R.jsx)(U,{value:p,autoComplete:`off`,placeholder:`First`,onChange:e=>m(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Middle name`,(0,R.jsx)(U,{value:h,autoComplete:`off`,placeholder:`Optional`,onChange:e=>g(e.target.value)})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-[minmax(0,1fr)_minmax(130px,.6fr)]`,children:[(0,R.jsxs)(W,{children:[`Last name`,(0,R.jsx)(U,{value:_,autoComplete:`off`,placeholder:`Last`,onChange:e=>v(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Country`,(0,R.jsx)(U,{value:y,autoComplete:`off`,placeholder:`Taiwan`,onChange:e=>b(e.target.value)})]})]}),(0,R.jsxs)(`details`,{className:`rounded-md border bg-background p-3`,children:[(0,R.jsx)(`summary`,{className:`cursor-pointer text-sm font-extrabold`,children:`Advanced options`}),(0,R.jsxs)(`div`,{className:`mt-3 grid gap-3 md:grid-cols-2`,children:[(0,R.jsxs)(W,{children:[`Gender`,(0,R.jsxs)(Vt,{value:x,onChange:e=>ee(e.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Default`}),tr.map(e=>(0,R.jsx)(`option`,{value:e,children:e},e))]})]}),(0,R.jsxs)(W,{children:[`Date of birth`,(0,R.jsx)(U,{value:te,type:`date`,autoComplete:`off`,onChange:e=>S(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Date of joining`,(0,R.jsx)(U,{value:C,type:`date`,autoComplete:`off`,onChange:e=>re(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Personal email`,(0,R.jsx)(U,{value:ie,type:`email`,autoComplete:`off`,placeholder:`Optional`,onChange:e=>ae(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Preferred contact email`,(0,R.jsxs)(Vt,{value:oe,onChange:e=>ce(e.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Default`}),nr.map(e=>(0,R.jsx)(`option`,{value:e,children:e},e))]})]})]})]}),(0,R.jsx)(`div`,{className:`flex flex-wrap items-center justify-between gap-3`,children:(0,R.jsxs)(B,{id:`setupEngineer`,type:`submit`,disabled:e||!d.trim()||!p.trim(),children:[(0,R.jsx)(se,{}),`Set up engineer`]})})]})})]})}function sr({person:e,loading:t,canWrite:n,onAssign:r,onStatusChange:i,onDraftEmail:a,onSendEmail:o,crmContactUrl:s,crmAttachmentUrl:c,canConfigure:u,onOpenConfiguration:d}){let f=e.name||e.email_508||e.email||`CRM contact`,[p,m]=(0,l.useState)($t(e.onboarder)),[h,g]=(0,l.useState)(!1),[_,v]=(0,l.useState)(null),[y,b]=(0,l.useState)(null),[x,ee]=(0,l.useState)({has_contributed:_n(Xt(e))===`onboarded`,discord_joined:e.discord_user_id?`yes`:`unknown`,agreement_signed:`unknown`});(0,l.useEffect)(()=>m($t(e.onboarder)),[e.onboarder]);let S=_n(Xt(e)),ne=e.profile_status||{},ae=[[`Discord`,ne.discord_linked],[`Resume`,ne.latest_resume],[`Skills`,Number(ne.skills_count||0)>0]].filter(([,e])=>!e),oe=s(e.crm_contact_id),se=c(e.latest_resume_id),ce=_?.onboarding_email_sent_at||e.onboarding_email_sent_at,w=_?.onboarding_email_sent_by||e.onboarding_email_sent_by,T=_?.onboarding_email_recipient||e.onboarding_email_recipient,le=!_||y!==null&&y.has_contributed===x.has_contributed&&y.discord_joined===x.discord_joined&&y.agreement_signed===x.agreement_signed,E=_&&!_.onboarding_email_sent_at?le?_.can_send?``:_.recipient_email?_.reply_to_email?`Send disabled: onboarding email SMTP is not configured.`:`Send disabled: your Reply-To email is missing.`:`Send disabled: candidate email is missing.`:`Send disabled: regenerate after changing draft options.`:``,ue=E.includes(`SMTP`),de=!!t[`onboarding-email-draft:${e.crm_contact_id}`],fe=!!t[`onboarding-email-send:${e.crm_contact_id}`],pe=_?.markdown_body||``;async function D(t=x){let n=await a(e.crm_contact_id,t);n&&(v(n),b({...t}),g(!0))}async function O(){if(!_||!y||!le)return;let t=await o(e.crm_contact_id,y,_.markdown_body);t&&v(t)}return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(Gt,{children:[(0,R.jsxs)(K,{children:[oe?(0,R.jsx)(`a`,{className:`font-extrabold text-primary`,href:oe,target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${f} in CRM`,children:f}):(0,R.jsx)(`strong`,{children:f}),(0,R.jsx)(`div`,{className:`text-sm text-muted-foreground`,children:e.email_508||e.email||``})]}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid max-w-56 gap-2`,children:[(0,R.jsx)(z,{variant:Qt(Xt(e)),children:e.onboarding_status_label||Zt(Xt(e))}),n?(0,R.jsxs)(Vt,{"aria-label":`Onboarding status for ${f}`,value:S,disabled:t[`onboarding-status:${e.crm_contact_id}`],onChange:t=>i(e.crm_contact_id,t.target.value),children:[S?null:(0,R.jsx)(`option`,{value:``,disabled:!0,children:`No status`}),mn.map(([e,t])=>(0,R.jsx)(`option`,{value:e,children:t},e))]}):null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`form`,{className:`grid max-w-64 grid-cols-[minmax(100px,1fr)_auto] items-center gap-2`,onSubmit:t=>{t.preventDefault(),r(e.crm_contact_id,p)},children:[(0,R.jsx)(U,{"aria-label":`Onboarder for ${f}`,value:p,placeholder:`508 username`,onChange:e=>m(e.target.value)}),(0,R.jsx)(B,{type:`submit`,size:`sm`,"aria-label":`Save onboarder for ${f}`,disabled:t[`onboarder:${e.crm_contact_id}`],children:`Save`})]})}),(0,R.jsx)(K,{children:qt(e.onboarding_updated_at)}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-2`,children:[ce?(0,R.jsxs)(z,{variant:`succeeded`,children:[`Sent `,qt(ce)]}):(0,R.jsx)(z,{variant:`neutral`,children:`Not sent`}),T?(0,R.jsx)(`span`,{className:`text-xs text-muted-foreground`,children:T}):null,w?(0,R.jsxs)(`span`,{className:`text-xs text-muted-foreground`,children:[`By `,w]}):null,n?(0,R.jsxs)(B,{type:`button`,size:`sm`,variant:h?`outline`:`secondary`,onClick:()=>{if(h){g(!1);return}g(!0),_||D()},disabled:de,children:[(0,R.jsx)(te,{}),_?`Edit draft`:`Draft email`]}):null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-1.5`,children:[se?(0,R.jsx)(`a`,{className:`inline-flex min-h-7 items-center rounded-md border bg-secondary px-2 text-xs font-extrabold`,href:se,target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${f} resume`,children:`Resume`}):null,an(e.linkedin)?(0,R.jsx)(`a`,{className:`inline-flex min-h-7 items-center rounded-md border bg-secondary px-2 text-xs font-extrabold`,href:an(e.linkedin),target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${f} LinkedIn`,children:`LinkedIn`}):null,on(e.github_username)?(0,R.jsx)(`a`,{className:`inline-flex min-h-7 items-center rounded-md border bg-secondary px-2 text-xs font-extrabold`,href:on(e.github_username),target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${f} GitHub`,children:e.github_username||`GitHub`}):null,!se&&!an(e.linkedin)&&!on(e.github_username)?`None`:null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-1.5`,children:[ae.map(([e])=>(0,R.jsxs)(z,{variant:`missing`,children:[`Missing `,e]},String(e))),ae.length===0?`None`:null]})})]}),h?(0,R.jsx)(Gt,{children:(0,R.jsx)(K,{colSpan:7,className:`bg-secondary/30`,children:(0,R.jsxs)(`div`,{className:`grid gap-3 rounded-md border bg-background p-4`,children:[(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-[auto_minmax(150px,220px)_minmax(150px,220px)_auto] md:items-end`,children:[(0,R.jsxs)(`label`,{className:`flex min-h-9 items-center gap-2 text-sm font-semibold`,children:[(0,R.jsx)(`input`,{type:`checkbox`,checked:x.has_contributed,onChange:e=>{ee({...x,has_contributed:e.target.checked})}}),`Contribution done`]}),(0,R.jsxs)(W,{children:[`Discord`,(0,R.jsxs)(Vt,{value:x.discord_joined,onChange:e=>ee({...x,discord_joined:e.target.value}),children:[(0,R.jsx)(`option`,{value:`unknown`,children:`Unknown`}),(0,R.jsx)(`option`,{value:`yes`,children:`Joined`}),(0,R.jsx)(`option`,{value:`no`,children:`Not joined`})]})]}),(0,R.jsxs)(W,{children:[`Agreement`,(0,R.jsxs)(Vt,{value:x.agreement_signed,onChange:e=>ee({...x,agreement_signed:e.target.value}),children:[(0,R.jsx)(`option`,{value:`unknown`,children:`Unknown`}),(0,R.jsx)(`option`,{value:`yes`,children:`Signed`}),(0,R.jsx)(`option`,{value:`no`,children:`Not signed`})]})]}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,onClick:()=>D(),disabled:de,children:[(0,R.jsx)(C,{}),`Regenerate`]})]}),_?(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(`div`,{className:`grid gap-2 text-sm md:grid-cols-3`,children:[(0,R.jsxs)(`span`,{children:[(0,R.jsx)(`strong`,{children:`To:`}),` `,_.recipient_email||`Missing`]}),(0,R.jsxs)(`span`,{children:[(0,R.jsx)(`strong`,{children:`Reply-To:`}),` `,_.reply_to_email||`Missing`]}),(0,R.jsxs)(`span`,{children:[(0,R.jsx)(`strong`,{children:`From:`}),` `,_.sender_display_name||`onboarding`]})]}),(0,R.jsxs)(W,{children:[`Subject`,(0,R.jsx)(U,{value:_.subject,readOnly:!0})]}),(0,R.jsxs)(W,{children:[`Draft`,(0,R.jsx)(`textarea`,{value:pe,className:`min-h-64 w-full rounded-md border border-input bg-background px-3 py-2 font-mono text-sm text-foreground shadow-xs transition-colors placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]`,onChange:e=>v({..._,markdown_body:e.target.value})})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-2`,children:[(0,R.jsxs)(B,{type:`button`,variant:`default`,onClick:O,disabled:fe||!_.can_send||!le||!pe.trim(),title:E||void 0,children:[(0,R.jsx)(re,{}),fe?`Sending`:`Send`]}),E?(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[E,ue&&u?(0,R.jsxs)(B,{type:`button`,size:`sm`,variant:`ghost`,className:`ml-2`,onClick:d,children:[(0,R.jsx)(ie,{}),`Configure`]}):null]}):null,_.marker_status===`error`?(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[`Marker not saved: `,_.marker_error||`unknown`]}):null]})]}):(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:de?`Drafting email`:`No draft loaded`})]})})}):null]})}function cr(e){return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(V,{className:`grid gap-3 p-4 md:grid-cols-4 md:items-end`,children:[(0,R.jsxs)(W,{children:[`Window`,(0,R.jsxs)(Vt,{id:`minutes`,value:e.minutes,onChange:t=>e.setMinutes(t.target.value),children:[(0,R.jsx)(`option`,{value:`15`,children:`15 minutes`}),(0,R.jsx)(`option`,{value:`60`,children:`1 hour`}),(0,R.jsx)(`option`,{value:`360`,children:`6 hours`}),(0,R.jsx)(`option`,{value:`1440`,children:`24 hours`})]})]}),(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{id:`status`,value:e.status,onChange:t=>e.setStatus(t.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Any status`}),(0,R.jsx)(`option`,{value:`queued`,children:`Queued`}),(0,R.jsx)(`option`,{value:`running`,children:`Running`}),(0,R.jsx)(`option`,{value:`succeeded`,children:`Succeeded`}),(0,R.jsx)(`option`,{value:`failed`,children:`Failed`}),(0,R.jsx)(`option`,{value:`dead`,children:`Dead`}),(0,R.jsx)(`option`,{value:`canceled`,children:`Canceled`})]})]}),(0,R.jsxs)(W,{children:[`Type`,(0,R.jsx)(U,{id:`jobType`,value:e.jobType,autoComplete:`off`,placeholder:`Any type`,onChange:t=>e.setJobType(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onSearch()})]}),(0,R.jsxs)(B,{id:`refreshJobs`,type:`button`,onClick:e.onSearch,disabled:e.loading.jobs,children:[(0,R.jsx)(C,{}),`Refresh background tasks`]})]}),(0,R.jsxs)(`section`,{className:`grid gap-3 md:grid-cols-4`,"aria-label":`Background task summary`,children:[(0,R.jsx)(On,{id:`metricTotal`,label:`Total`,value:e.jobs.length}),(0,R.jsx)(On,{id:`metricQueued`,label:`Queued`,value:e.jobCounts.queued||0}),(0,R.jsx)(On,{id:`metricRunning`,label:`Running`,value:e.jobCounts.running||0}),(0,R.jsx)(On,{id:`metricFailed`,label:`Failed`,value:(e.jobCounts.failed||0)+(e.jobCounts.dead||0)})]}),(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Recent background tasks`})}),(0,R.jsx)(kn,{hidden:e.jobs.length!==0,children:`No background tasks match these filters.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`jobsTable`,className:L(`min-w-[980px]`,e.jobs.length===0&&`hidden`),"aria-label":`Recent background tasks`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(Dn,{className:`w-[22%]`,label:`Task id`,scope:`jobs`,sort:e.sort,sortKey:`job_id`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[24%]`,label:`Task type`,scope:`jobs`,sort:e.sort,sortKey:`type`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[12%]`,label:`Status`,scope:`jobs`,sort:e.sort,sortKey:`status`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[12%]`,label:`Attempts`,scope:`jobs`,sort:e.sort,sortKey:`attempts`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[18%]`,label:`Updated`,scope:`jobs`,sort:e.sort,sortKey:`updated_at`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(G,{children:`Actions`})]})}),(0,R.jsx)(Wt,{id:`jobsBody`,children:e.jobs.map(t=>(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{className:`font-mono`,children:t.job_id}),(0,R.jsx)(K,{children:t.type}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:t.status||`neutral`,children:t.status})}),(0,R.jsxs)(K,{children:[t.attempts,`/`,t.max_attempts]}),(0,R.jsx)(K,{children:qt(t.updated_at)}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-2`,children:[(0,R.jsx)(B,{type:`button`,size:`sm`,variant:`outline`,"aria-label":`View details for ${t.type} task ${t.job_id}`,onClick:()=>e.onDetail(t.job_id),disabled:e.loading[`detail:${t.job_id}`],children:`Details`}),e.canWrite?(0,R.jsx)(B,{type:`button`,size:`sm`,"aria-label":`Rerun ${t.type} task ${t.job_id}`,onClick:()=>e.onRerun(t.job_id),disabled:e.loading[`rerun:${t.job_id}`],children:`Rerun`}):null]})})]},t.job_id))})]})})]}),e.jobDetail?(0,R.jsxs)(V,{id:`jobDetailPanel`,children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Task detail`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.jobDetail.job_id})]}),(0,R.jsxs)(Bt,{className:`grid gap-4`,children:[(0,R.jsx)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[[`Task type`,e.jobDetail.type],[`Status`,e.jobDetail.status],[`Attempts`,`${e.jobDetail.attempts}/${e.jobDetail.max_attempts}`],[`Updated`,qt(e.jobDetail.updated_at)],[`Created`,qt(e.jobDetail.created_at)],[`Run after`,qt(e.jobDetail.run_after)],[`Locked by`,e.jobDetail.locked_by||`None`],[`Last error`,e.jobDetail.last_error||`None`]].map(([e,t])=>(0,R.jsxs)(`div`,{className:`grid gap-1 rounded-md border bg-background p-3`,children:[(0,R.jsx)(`span`,{className:`text-[11px] font-extrabold uppercase text-muted-foreground`,children:e}),(0,R.jsx)(`strong`,{className:`break-words text-sm`,children:t})]},e))}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`h2`,{className:`mb-2 text-[15px] font-bold`,children:`Payload`}),(0,R.jsx)(`pre`,{className:`max-h-64 overflow-auto whitespace-pre-wrap break-words rounded-md border bg-background p-3 font-mono text-xs`,children:Yt(e.jobDetail.payload)||`No payload`})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`h2`,{className:`mb-2 text-[15px] font-bold`,children:`Result`}),(0,R.jsx)(`pre`,{className:`max-h-64 overflow-auto whitespace-pre-wrap break-words rounded-md border bg-background p-3 font-mono text-xs`,children:Yt(e.jobDetail.result)||`No result`})]})]})]}):null]})}function lr(e){return(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Recent audit`}),(0,R.jsxs)(B,{id:`refreshAudit`,type:`button`,variant:`outline`,onClick:e.onRefresh,disabled:e.loading.audit,children:[(0,R.jsx)(C,{}),`Refresh`]})]}),(0,R.jsx)(kn,{hidden:e.events.length!==0,children:`No audit events found.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`auditTable`,className:L(`min-w-[760px]`,e.events.length===0&&`hidden`),"aria-label":`Recent audit events`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(Dn,{className:`w-[24%]`,label:`Time`,scope:`audit`,sort:e.sort,sortKey:`occurred_at`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[28%]`,label:`Actor`,scope:`audit`,sort:e.sort,sortKey:`actor`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[28%]`,label:`Action`,scope:`audit`,sort:e.sort,sortKey:`action`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[20%]`,label:`Result`,scope:`audit`,sort:e.sort,sortKey:`result`,onSort:(t,n)=>e.onSort(n)})]})}),(0,R.jsx)(Wt,{id:`auditBody`,children:e.events.map(e=>(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:qt(e.occurred_at)}),(0,R.jsx)(K,{children:e.actor_display_name||e.actor_subject||e.actor_provider}),(0,R.jsx)(K,{children:e.action}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:e.result===`success`?`succeeded`:`failed`,children:e.result})})]},e.id||`${e.occurred_at||``}-${e.actor_subject||``}-${e.action||``}`))})]})})]})}function ur({report:e,loading:t,onRefresh:n}){let r=e?.summary||{},i=[[`Status`,e?.status_counts||{}],[`Intent`,e?.intent_counts||{}],[`Planner`,e?.planner_counts||{}]].flatMap(([e,t])=>Object.entries(t).map(([t,n])=>({label:e,value:t,count:n}))).sort((e,t)=>t.count-e.count||e.label.localeCompare(t.label)),a=Array.isArray(e?.recent_unsupported)?e.recent_unsupported:[];return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Agent requests`}),(0,R.jsxs)(B,{id:`refreshAgent`,type:`button`,variant:`outline`,onClick:n,disabled:t.agent,children:[(0,R.jsx)(C,{}),`Refresh`]})]}),(0,R.jsxs)(Bt,{className:`grid gap-3 md:grid-cols-5`,children:[(0,R.jsx)(On,{id:`agentMetricTotal`,label:`Total`,value:r.total||0}),(0,R.jsx)(On,{id:`agentMetricHandled`,label:`Handled`,value:r.handled||0}),(0,R.jsx)(On,{id:`agentMetricConfirmations`,label:`Confirmations`,value:r.requires_confirmation||0}),(0,R.jsx)(On,{id:`agentMetricClarifications`,label:`Clarifications`,value:r.needs_clarification||0}),(0,R.jsx)(On,{id:`agentMetricUnsupported`,label:`Not understood`,value:r.unsupported||0})]})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Request mix`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:`Recent agent.request audit events.`})]}),(0,R.jsx)(kn,{hidden:i.length!==0,children:`No agent request data found.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`agentBreakdownTable`,className:L(`min-w-[860px]`,i.length===0&&`hidden`),"aria-label":`Agent request breakdown`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{children:`Dimension`}),(0,R.jsx)(G,{children:`Value`}),(0,R.jsx)(G,{children:`Count`})]})}),(0,R.jsx)(Wt,{id:`agentBreakdownBody`,children:i.map(e=>(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:e.label}),(0,R.jsx)(K,{children:e.value}),(0,R.jsx)(K,{children:e.count})]},`${e.label}-${e.value}`))})]})})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Not understood`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:`Sanitized request text only.`})]}),(0,R.jsx)(kn,{hidden:a.length!==0,children:`No unsupported agent requests found.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`agentUnsupportedTable`,className:L(`min-w-[860px]`,a.length===0&&`hidden`),"aria-label":`Unsupported agent requests`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{children:`Time`}),(0,R.jsx)(G,{children:`Actor`}),(0,R.jsx)(G,{children:`Message`}),(0,R.jsx)(G,{children:`Result`})]})}),(0,R.jsx)(Wt,{id:`agentUnsupportedBody`,children:a.map(e=>(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:qt(e.occurred_at)}),(0,R.jsx)(K,{children:e.actor}),(0,R.jsx)(K,{children:e.message_sanitized}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:e.result===`success`?`succeeded`:`failed`,children:e.result||`unknown`})})]},`${e.occurred_at||``}-${e.actor||``}-${e.message_sanitized||``}`))})]})})]})]})}function dr({items:e,loading:t,canWrite:n,onRefresh:r,onSave:i,onClear:a,focusCategory:o,focusNonce:s}){let[c,u]=(0,l.useState)(`All`),[d,f]=(0,l.useState)(``),[p,m]=(0,l.useState)({}),h=(0,l.useMemo)(()=>{let t=new Set(e.map(e=>e.category)),n=sn.filter(e=>t.has(e.category)),r=Array.from(t).filter(e=>!cn.has(e)).sort().map(e=>({category:e,label:e,description:`Additional runtime settings.`}));return n.concat(r)},[e]),g=(0,l.useMemo)(()=>{let t=new Map;for(let n of e){if(c!==`All`&&n.category!==c)continue;let e=t.get(n.category)??[];e.push(n),t.set(n.category,e)}return Array.from(t.entries()).map(([e,t])=>{let n=cn.get(e),r=t.sort((e,t)=>Number(!un(e))-Number(!un(t))||e.label.localeCompare(t.label)),i=r.filter(un),a=r.filter(e=>!un(e));return{category:e,label:n?.label??e,description:n?.description??`Additional runtime settings.`,order:n?.index??sn.length,primaryItems:i.length?i:r,advancedItems:i.length?a:[],items:r}}).sort((e,t)=>e.order-t.order||e.label.localeCompare(t.label))},[e,c]),_=(0,l.useMemo)(()=>({configured:e.filter(e=>e.configured).length,envLocked:e.filter(e=>e.env_locked).length,missing:e.filter(e=>!e.configured).length}),[e]),v=g.reduce((e,t)=>e+t.items.length,0);(0,l.useEffect)(()=>{m(Object.fromEntries(e.map(e=>[e.key,e.is_secret?``:String(e.value??``)])))},[e]),(0,l.useEffect)(()=>{c!==`All`&&!e.some(e=>e.category===c)&&u(`All`)},[e,c]),(0,l.useEffect)(()=>{if(!o||!e.some(e=>e.category===o))return;u(o),f(o);let t=window.requestAnimationFrame?.(()=>{document.getElementById(ln(o))?.scrollIntoView({block:`start`,behavior:`smooth`})}),n=window.setTimeout(()=>f(``),4e3);return()=>{t!==void 0&&window.cancelAnimationFrame?.(t),window.clearTimeout(n)}},[o,s,e]);function y(e){return e.source===`env`?`ENV`:e.source===`database`?`DB`:`Default`}function b(e){let r=p[e.key]??``,i=!n||e.env_locked||t[`configuration:${e.key}`];return e.value_type===`bool`?(0,R.jsxs)(Vt,{"aria-label":`${e.label} value`,value:r,disabled:i,onChange:t=>m(n=>({...n,[e.key]:t.target.value})),children:[(0,R.jsx)(`option`,{value:``,children:`Default`}),(0,R.jsx)(`option`,{value:`true`,children:`True`}),(0,R.jsx)(`option`,{value:`false`,children:`False`})]}):(0,R.jsx)(U,{"aria-label":`${e.label} value`,value:r,type:e.is_secret?`password`:e.value_type===`int`?`number`:`text`,inputMode:e.value_type===`int`||e.value_type===`float`?`numeric`:`text`,placeholder:e.is_secret?`Set new value`:``,autoComplete:`off`,disabled:i,onChange:t=>m(n=>({...n,[e.key]:t.target.value}))})}function x(e,t,n){return(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`configurationTable-${n}`,className:`min-w-[980px]`,"aria-label":`${e} configuration settings`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{className:`w-[26%]`,children:`Setting`}),(0,R.jsx)(G,{className:`w-[12%]`,children:`Source`}),(0,R.jsx)(G,{className:`w-[18%]`,children:`Active`}),(0,R.jsx)(G,{className:`w-[29%]`,children:`Value`}),(0,R.jsx)(G,{className:`w-[15%]`,children:`Actions`})]})}),(0,R.jsx)(Wt,{id:`configurationBody-${n}`,children:t.map(e=>ee(e))})]})})}function ee(e){let r=t[`configuration:${e.key}`],o=n&&!e.env_locked&&!r,s=p[e.key]??``,c=!e.is_secret&&!s.trim();return(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-1`,children:[(0,R.jsx)(`strong`,{children:e.label}),(0,R.jsx)(`span`,{className:`font-mono text-xs text-muted-foreground`,children:e.key}),(0,R.jsx)(`span`,{className:`text-xs text-muted-foreground`,children:e.description}),e.restart_required?(0,R.jsx)(`div`,{children:(0,R.jsx)(z,{variant:`running`,children:`Restart`})}):null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-1.5`,children:[(0,R.jsx)(z,{variant:e.source===`env`?`running`:`neutral`,children:y(e)}),e.env_locked?(0,R.jsx)(`span`,{className:`text-xs text-muted-foreground`,children:`Environment locked`}):null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-1`,children:[(0,R.jsx)(z,{variant:e.configured?`succeeded`:`missing`,children:e.configured?`Configured`:`Missing`}),e.is_secret?(0,R.jsx)(`span`,{className:`font-mono text-xs text-muted-foreground`,children:e.masked_value||(e.configured?`Hidden`:`No secret`)}):(0,R.jsx)(`span`,{className:`break-words text-xs text-muted-foreground`,children:String(e.value??``)||`Default`}),e.is_secret&&e.secret_encryption_configured===!1?(0,R.jsx)(`span`,{className:`text-xs text-red-300`,children:`Encryption key missing`}):null]})}),(0,R.jsx)(K,{children:b(e)}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-2`,children:[(0,R.jsx)(B,{type:`button`,size:`sm`,onClick:()=>i(e.key,s),disabled:!o||e.is_secret&&!s.trim()||c||e.is_secret&&e.secret_encryption_configured===!1,children:`Save`}),(0,R.jsx)(B,{type:`button`,size:`sm`,variant:`outline`,onClick:()=>a(e.key),disabled:!o||e.source!==`database`,children:`Clear`})]})})]},e.key)}return(0,R.jsxs)(`div`,{className:`grid gap-4`,children:[(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Configuration`}),(0,R.jsxs)(B,{id:`refreshConfiguration`,type:`button`,variant:`outline`,onClick:r,disabled:t.configuration,children:[(0,R.jsx)(C,{}),`Refresh`]})]}),(0,R.jsxs)(Bt,{className:`grid gap-4`,children:[(0,R.jsx)(`section`,{className:`grid gap-3 sm:grid-cols-2 lg:grid-cols-4`,"aria-label":`Configuration summary`,children:[[`Total`,e.length],[`Configured`,_.configured],[`Missing`,_.missing],[`Env locked`,_.envLocked]].map(([e,t])=>(0,R.jsxs)(`div`,{className:`rounded-md border bg-background p-3`,children:[(0,R.jsx)(`span`,{className:`text-[11px] font-extrabold uppercase text-muted-foreground`,children:e}),(0,R.jsx)(`strong`,{className:`mt-1 block text-xl`,children:t})]},e))}),(0,R.jsxs)(`section`,{className:`flex flex-wrap gap-2`,"aria-label":`Configuration groups`,children:[(0,R.jsxs)(B,{type:`button`,size:`sm`,variant:c===`All`?`default`:`outline`,"aria-pressed":c===`All`,onClick:()=>u(`All`),children:[`All groups`,(0,R.jsx)(`span`,{className:`font-mono text-[11px]`,children:e.length})]}),h.map(t=>{let n=e.filter(e=>e.category===t.category).length;return(0,R.jsxs)(B,{type:`button`,size:`sm`,variant:c===t.category?`default`:`outline`,"aria-pressed":c===t.category,onClick:()=>u(t.category),children:[t.label,(0,R.jsx)(`span`,{className:`font-mono text-[11px]`,children:n})]},t.category)})]})]})]}),(0,R.jsx)(kn,{hidden:v!==0,children:`No configuration entries found.`}),g.map(e=>{let t=e.items.filter(e=>e.configured).length,n=e.items.length-t,r=e.items.some(e=>e.restart_required);return(0,R.jsxs)(V,{id:ln(e.category),className:L(`scroll-mt-4 transition-shadow`,d===e.category&&`ring-2 ring-primary ring-offset-2 ring-offset-background`),children:[(0,R.jsxs)(H,{className:`items-start`,children:[(0,R.jsxs)(`div`,{className:`grid gap-1`,children:[(0,R.jsx)(zt,{children:e.label}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.description})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-1.5`,children:[(0,R.jsxs)(z,{variant:`neutral`,children:[e.items.length,` settings`]}),(0,R.jsxs)(z,{variant:n?`missing`:`succeeded`,children:[t,` configured`]}),r?(0,R.jsx)(z,{variant:`running`,children:`Restart`}):null]})]}),x(e.label,e.primaryItems,e.category),e.advancedItems.length?(0,R.jsxs)(`details`,{className:`border-t bg-background/40`,children:[(0,R.jsxs)(`summary`,{className:`flex min-h-11 cursor-pointer items-center justify-between gap-3 px-4 py-3 text-sm font-extrabold`,children:[(0,R.jsx)(`span`,{children:`Advanced`}),(0,R.jsxs)(`span`,{className:`font-mono text-xs text-muted-foreground`,children:[e.advancedItems.length,` settings`]})]}),(0,R.jsx)(`div`,{className:`border-t`,children:x(`${e.label} advanced`,e.advancedItems,`${e.category}-advanced`)})]}):null]},e.category)})]})}var fr=document.getElementById(`root`);if(fr)(0,fe.createRoot)(fr).render((0,R.jsx)(l.StrictMode,{children:(0,R.jsx)(jn,{})}));else throw Error(`Missing #root container`); \ No newline at end of file +`).replace(kd,``)}function jd(e,t){return t=Ad(t),Ad(e)===t}function Md(e,t,n,r,i,o){switch(n){case`children`:typeof r==`string`?t===`body`||t===`textarea`&&r===``||U(e,r):(typeof r==`number`||typeof r==`bigint`)&&t!==`body`&&U(e,``+r);break;case`className`:jt(e,`class`,r);break;case`tabIndex`:jt(e,`tabindex`,r);break;case`dir`:case`role`:case`viewBox`:case`width`:case`height`:jt(e,n,r);break;case`style`:Ht(e,r,o);break;case`data`:if(t!==`object`){jt(e,`data`,r);break}case`src`:case`href`:if(r===``&&(t!==`a`||n!==`href`)){e.removeAttribute(n);break}if(r==null||typeof r==`function`||typeof r==`symbol`||typeof r==`boolean`){e.removeAttribute(n);break}r=G(``+r),e.setAttribute(n,r);break;case`action`:case`formAction`:if(typeof r==`function`){e.setAttribute(n,`javascript:throw new Error('A React form was unexpectedly submitted. If you called form.submit() manually, consider using form.requestSubmit() instead. If you\\'re trying to use event.stopPropagation() in a submit event handler, consider also calling event.preventDefault().')`);break}else typeof o==`function`&&(n===`formAction`?(t!==`input`&&Md(e,t,`name`,i.name,i,null),Md(e,t,`formEncType`,i.formEncType,i,null),Md(e,t,`formMethod`,i.formMethod,i,null),Md(e,t,`formTarget`,i.formTarget,i,null)):(Md(e,t,`encType`,i.encType,i,null),Md(e,t,`method`,i.method,i,null),Md(e,t,`target`,i.target,i,null)));if(r==null||typeof r==`symbol`||typeof r==`boolean`){e.removeAttribute(n);break}r=G(``+r),e.setAttribute(n,r);break;case`onClick`:r!=null&&(e.onclick=K);break;case`onScroll`:r!=null&&$(`scroll`,e);break;case`onScrollEnd`:r!=null&&$(`scrollend`,e);break;case`dangerouslySetInnerHTML`:if(r!=null){if(typeof r!=`object`||!(`__html`in r))throw Error(a(61));if(n=r.__html,n!=null){if(i.children!=null)throw Error(a(60));e.innerHTML=n}}break;case`multiple`:e.multiple=r&&typeof r!=`function`&&typeof r!=`symbol`;break;case`muted`:e.muted=r&&typeof r!=`function`&&typeof r!=`symbol`;break;case`suppressContentEditableWarning`:case`suppressHydrationWarning`:case`defaultValue`:case`defaultChecked`:case`innerHTML`:case`ref`:break;case`autoFocus`:break;case`xlinkHref`:if(r==null||typeof r==`function`||typeof r==`boolean`||typeof r==`symbol`){e.removeAttribute(`xlink:href`);break}n=G(``+r),e.setAttributeNS(`http://www.w3.org/1999/xlink`,`xlink:href`,n);break;case`contentEditable`:case`spellCheck`:case`draggable`:case`value`:case`autoReverse`:case`externalResourcesRequired`:case`focusable`:case`preserveAlpha`:r!=null&&typeof r!=`function`&&typeof r!=`symbol`?e.setAttribute(n,``+r):e.removeAttribute(n);break;case`inert`:case`allowFullScreen`:case`async`:case`autoPlay`:case`controls`:case`default`:case`defer`:case`disabled`:case`disablePictureInPicture`:case`disableRemotePlayback`:case`formNoValidate`:case`hidden`:case`loop`:case`noModule`:case`noValidate`:case`open`:case`playsInline`:case`readOnly`:case`required`:case`reversed`:case`scoped`:case`seamless`:case`itemScope`:r&&typeof r!=`function`&&typeof r!=`symbol`?e.setAttribute(n,``):e.removeAttribute(n);break;case`capture`:case`download`:!0===r?e.setAttribute(n,``):!1!==r&&r!=null&&typeof r!=`function`&&typeof r!=`symbol`?e.setAttribute(n,r):e.removeAttribute(n);break;case`cols`:case`rows`:case`size`:case`span`:r!=null&&typeof r!=`function`&&typeof r!=`symbol`&&!isNaN(r)&&1<=r?e.setAttribute(n,r):e.removeAttribute(n);break;case`rowSpan`:case`start`:r==null||typeof r==`function`||typeof r==`symbol`||isNaN(r)?e.removeAttribute(n):e.setAttribute(n,r);break;case`popover`:$(`beforetoggle`,e),$(`toggle`,e),At(e,`popover`,r);break;case`xlinkActuate`:Mt(e,`http://www.w3.org/1999/xlink`,`xlink:actuate`,r);break;case`xlinkArcrole`:Mt(e,`http://www.w3.org/1999/xlink`,`xlink:arcrole`,r);break;case`xlinkRole`:Mt(e,`http://www.w3.org/1999/xlink`,`xlink:role`,r);break;case`xlinkShow`:Mt(e,`http://www.w3.org/1999/xlink`,`xlink:show`,r);break;case`xlinkTitle`:Mt(e,`http://www.w3.org/1999/xlink`,`xlink:title`,r);break;case`xlinkType`:Mt(e,`http://www.w3.org/1999/xlink`,`xlink:type`,r);break;case`xmlBase`:Mt(e,`http://www.w3.org/XML/1998/namespace`,`xml:base`,r);break;case`xmlLang`:Mt(e,`http://www.w3.org/XML/1998/namespace`,`xml:lang`,r);break;case`xmlSpace`:Mt(e,`http://www.w3.org/XML/1998/namespace`,`xml:space`,r);break;case`is`:At(e,`is`,r);break;case`innerText`:case`textContent`:break;default:(!(2s)break;var u=c.transferSize,d=c.initiatorType;u&&Id(d)&&(c=c.responseEnd,o+=u*(c`u`?null:document;function xf(e,t,n){var r=bf;if(r&&typeof t==`string`&&t){var i=z(t);i=`link[rel="`+e+`"][href="`+i+`"]`,typeof n==`string`&&(i+=`[crossorigin="`+n+`"]`),hf.has(i)||(hf.add(i),e={rel:e,crossOrigin:n,href:t},r.querySelector(i)===null&&(t=r.createElement(`link`),Pd(t,`link`,e),P(t),r.head.appendChild(t)))}}function Sf(e){_f.D(e),xf(`dns-prefetch`,e,null)}function Cf(e,t){_f.C(e,t),xf(`preconnect`,e,t)}function wf(e,t,n){_f.L(e,t,n);var r=bf;if(r&&e&&t){var i=`link[rel="preload"][as="`+z(t)+`"]`;t===`image`&&n&&n.imageSrcSet?(i+=`[imagesrcset="`+z(n.imageSrcSet)+`"]`,typeof n.imageSizes==`string`&&(i+=`[imagesizes="`+z(n.imageSizes)+`"]`)):i+=`[href="`+z(e)+`"]`;var a=i;switch(t){case`style`:a=Af(e);break;case`script`:a=Pf(e)}mf.has(a)||(e=p({rel:`preload`,href:t===`image`&&n&&n.imageSrcSet?void 0:e,as:t},n),mf.set(a,e),r.querySelector(i)!==null||t===`style`&&r.querySelector(jf(a))||t===`script`&&r.querySelector(Ff(a))||(t=r.createElement(`link`),Pd(t,`link`,e),P(t),r.head.appendChild(t)))}}function Tf(e,t){_f.m(e,t);var n=bf;if(n&&e){var r=t&&typeof t.as==`string`?t.as:`script`,i=`link[rel="modulepreload"][as="`+z(r)+`"][href="`+z(e)+`"]`,a=i;switch(r){case`audioworklet`:case`paintworklet`:case`serviceworker`:case`sharedworker`:case`worker`:case`script`:a=Pf(e)}if(!mf.has(a)&&(e=p({rel:`modulepreload`,href:e},t),mf.set(a,e),n.querySelector(i)===null)){switch(r){case`audioworklet`:case`paintworklet`:case`serviceworker`:case`sharedworker`:case`worker`:case`script`:if(n.querySelector(Ff(a)))return}r=n.createElement(`link`),Pd(r,`link`,e),P(r),n.head.appendChild(r)}}}function Ef(e,t,n){_f.S(e,t,n);var r=bf;if(r&&e){var i=Ct(r).hoistableStyles,a=Af(e);t||=`default`;var o=i.get(a);if(!o){var s={loading:0,preload:null};if(o=r.querySelector(jf(a)))s.loading=5;else{e=p({rel:`stylesheet`,href:e,"data-precedence":t},n),(n=mf.get(a))&&Rf(e,n);var c=o=r.createElement(`link`);P(c),Pd(c,`link`,e),c._p=new Promise(function(e,t){c.onload=e,c.onerror=t}),c.addEventListener(`load`,function(){s.loading|=1}),c.addEventListener(`error`,function(){s.loading|=2}),s.loading|=4,Lf(o,t,r)}o={type:`stylesheet`,instance:o,count:1,state:s},i.set(a,o)}}}function Df(e,t){_f.X(e,t);var n=bf;if(n&&e){var r=Ct(n).hoistableScripts,i=Pf(e),a=r.get(i);a||(a=n.querySelector(Ff(i)),a||(e=p({src:e,async:!0},t),(t=mf.get(i))&&zf(e,t),a=n.createElement(`script`),P(a),Pd(a,`link`,e),n.head.appendChild(a)),a={type:`script`,instance:a,count:1,state:null},r.set(i,a))}}function Of(e,t){_f.M(e,t);var n=bf;if(n&&e){var r=Ct(n).hoistableScripts,i=Pf(e),a=r.get(i);a||(a=n.querySelector(Ff(i)),a||(e=p({src:e,async:!0,type:`module`},t),(t=mf.get(i))&&zf(e,t),a=n.createElement(`script`),P(a),Pd(a,`link`,e),n.head.appendChild(a)),a={type:`script`,instance:a,count:1,state:null},r.set(i,a))}}function kf(e,t,n,r){var i=(i=he.current)?gf(i):null;if(!i)throw Error(a(446));switch(e){case`meta`:case`title`:return null;case`style`:return typeof n.precedence==`string`&&typeof n.href==`string`?(t=Af(n.href),n=Ct(i).hoistableStyles,r=n.get(t),r||(r={type:`style`,instance:null,count:0,state:null},n.set(t,r)),r):{type:`void`,instance:null,count:0,state:null};case`link`:if(n.rel===`stylesheet`&&typeof n.href==`string`&&typeof n.precedence==`string`){e=Af(n.href);var o=Ct(i).hoistableStyles,s=o.get(e);if(s||(i=i.ownerDocument||i,s={type:`stylesheet`,instance:null,count:0,state:{loading:0,preload:null}},o.set(e,s),(o=i.querySelector(jf(e)))&&!o._p&&(s.instance=o,s.state.loading=5),mf.has(e)||(n={rel:`preload`,as:`style`,href:n.href,crossOrigin:n.crossOrigin,integrity:n.integrity,media:n.media,hrefLang:n.hrefLang,referrerPolicy:n.referrerPolicy},mf.set(e,n),o||Nf(i,e,n,s.state))),t&&r===null)throw Error(a(528,``));return s}if(t&&r!==null)throw Error(a(529,``));return null;case`script`:return t=n.async,n=n.src,typeof n==`string`&&t&&typeof t!=`function`&&typeof t!=`symbol`?(t=Pf(n),n=Ct(i).hoistableScripts,r=n.get(t),r||(r={type:`script`,instance:null,count:0,state:null},n.set(t,r)),r):{type:`void`,instance:null,count:0,state:null};default:throw Error(a(444,e))}}function Af(e){return`href="`+z(e)+`"`}function jf(e){return`link[rel="stylesheet"][`+e+`]`}function Mf(e){return p({},e,{"data-precedence":e.precedence,precedence:null})}function Nf(e,t,n,r){e.querySelector(`link[rel="preload"][as="style"][`+t+`]`)?r.loading=1:(t=e.createElement(`link`),r.preload=t,t.addEventListener(`load`,function(){return r.loading|=1}),t.addEventListener(`error`,function(){return r.loading|=2}),Pd(t,`link`,n),P(t),e.head.appendChild(t))}function Pf(e){return`[src="`+z(e)+`"]`}function Ff(e){return`script[async]`+e}function If(e,t,n){if(t.count++,t.instance===null)switch(t.type){case`style`:var r=e.querySelector(`style[data-href~="`+z(n.href)+`"]`);if(r)return t.instance=r,P(r),r;var i=p({},n,{"data-href":n.href,"data-precedence":n.precedence,href:null,precedence:null});return r=(e.ownerDocument||e).createElement(`style`),P(r),Pd(r,`style`,i),Lf(r,n.precedence,e),t.instance=r;case`stylesheet`:i=Af(n.href);var o=e.querySelector(jf(i));if(o)return t.state.loading|=4,t.instance=o,P(o),o;r=Mf(n),(i=mf.get(i))&&Rf(r,i),o=(e.ownerDocument||e).createElement(`link`),P(o);var s=o;return s._p=new Promise(function(e,t){s.onload=e,s.onerror=t}),Pd(o,`link`,r),t.state.loading|=4,Lf(o,n.precedence,e),t.instance=o;case`script`:return o=Pf(n.src),(i=e.querySelector(Ff(o)))?(t.instance=i,P(i),i):(r=n,(i=mf.get(o))&&(r=p({},n),zf(r,i)),e=e.ownerDocument||e,i=e.createElement(`script`),P(i),Pd(i,`link`,r),e.head.appendChild(i),t.instance=i);case`void`:return null;default:throw Error(a(443,t.type))}else t.type===`stylesheet`&&!(t.state.loading&4)&&(r=t.instance,t.state.loading|=4,Lf(r,n.precedence,e));return t.instance}function Lf(e,t,n){for(var r=n.querySelectorAll(`link[rel="stylesheet"][data-precedence],style[data-precedence]`),i=r.length?r[r.length-1]:null,a=i,o=0;o title`):null)}function Uf(e,t,n){if(n===1||t.itemProp!=null)return!1;switch(e){case`meta`:case`title`:return!0;case`style`:if(typeof t.precedence!=`string`||typeof t.href!=`string`||t.href===``)break;return!0;case`link`:if(typeof t.rel!=`string`||typeof t.href!=`string`||t.href===``||t.onLoad||t.onError)break;switch(t.rel){case`stylesheet`:return e=t.disabled,typeof t.precedence==`string`&&e==null;default:return!0}case`script`:if(t.async&&typeof t.async!=`function`&&typeof t.async!=`symbol`&&!t.onLoad&&!t.onError&&t.src&&typeof t.src==`string`)return!0}return!1}function Wf(e){return!(e.type===`stylesheet`&&!(e.state.loading&3))}function Gf(e,t,n,r){if(n.type===`stylesheet`&&(typeof r.media!=`string`||!1!==matchMedia(r.media).matches)&&!(n.state.loading&4)){if(n.instance===null){var i=Af(r.href),a=t.querySelector(jf(i));if(a){t=a._p,typeof t==`object`&&t&&typeof t.then==`function`&&(e.count++,e=Jf.bind(e),t.then(e,e)),n.state.loading|=4,n.instance=a,P(a);return}a=t.ownerDocument||t,r=Mf(r),(i=mf.get(i))&&Rf(r,i),a=a.createElement(`link`),P(a);var o=a;o._p=new Promise(function(e,t){o.onload=e,o.onerror=t}),Pd(a,`link`,r),n.instance=a}e.stylesheets===null&&(e.stylesheets=new Map),e.stylesheets.set(n,t),(t=n.state.preload)&&!(n.state.loading&3)&&(e.count++,n=Jf.bind(e),t.addEventListener(`load`,n),t.addEventListener(`error`,n))}}var Kf=0;function qf(e,t){return e.stylesheets&&e.count===0&&Xf(e,e.stylesheets),0Kf?50:800)+t);return e.unsuspend=n,function(){e.unsuspend=null,clearTimeout(r),clearTimeout(i)}}:null}function Jf(){if(this.count--,this.count===0&&(this.imgCount===0||!this.waitingForImages)){if(this.stylesheets)Xf(this,this.stylesheets);else if(this.unsuspend){var e=this.unsuspend;this.unsuspend=null,e()}}}var Yf=null;function Xf(e,t){e.stylesheets=null,e.unsuspend!==null&&(e.count++,Yf=new Map,t.forEach(Zf,e),Yf=null,Jf.call(e))}function Zf(e,t){if(!(t.state.loading&4)){var n=Yf.get(e);if(n)var r=n.get(null);else{n=new Map,Yf.set(e,n);for(var i=e.querySelectorAll(`link[data-precedence],style[data-precedence]`),a=0;a{function n(){if(!(typeof __REACT_DEVTOOLS_GLOBAL_HOOK__>`u`||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!=`function`))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(n)}catch(e){console.error(e)}}n(),t.exports=de()}))();function pe(e){var t,n,r=``;if(typeof e==`string`||typeof e==`number`)r+=e;else if(typeof e==`object`)if(Array.isArray(e)){var i=e.length;for(t=0;ttypeof e==`boolean`?`${e}`:e===0?`0`:e,k=D,me=(e,t)=>n=>{if(t?.variants==null)return k(e,n?.class,n?.className);let{variants:r,defaultVariants:i}=t,a=Object.keys(r).map(e=>{let t=n?.[e],a=i?.[e];if(t===null)return null;let o=O(t)||O(a);return r[e][o]}),o=n&&Object.entries(n).reduce((e,t)=>{let[n,r]=t;return r===void 0||(e[n]=r),e},{});return k(e,a,t?.compoundVariants?.reduce((e,t)=>{let{class:n,className:r,...a}=t;return Object.entries(a).every(e=>{let[t,n]=e;return Array.isArray(n)?n.includes({...i,...o}[t]):{...i,...o}[t]===n})?[...e,n,r]:e},[]),n?.class,n?.className)},A=(e,t)=>{let n=Array(e.length+t.length);for(let t=0;t({classGroupId:e,validator:t}),ge=(e=new Map,t=null,n)=>({nextPart:e,validators:t,classGroupId:n}),_e=`-`,ve=[],ye=`arbitrary..`,be=e=>{let t=Ce(e),{conflictingClassGroups:n,conflictingClassGroupModifiers:r}=e;return{getClassGroupId:e=>{if(e.startsWith(`[`)&&e.endsWith(`]`))return Se(e);let n=e.split(_e);return xe(n,+(n[0]===``&&n.length>1),t)},getConflictingClassGroupIds:(e,t)=>{if(t){let t=r[e],i=n[e];return t?i?A(i,t):t:i||ve}return n[e]||ve}}},xe=(e,t,n)=>{if(e.length-t===0)return n.classGroupId;let r=e[t],i=n.nextPart.get(r);if(i){let n=xe(e,t+1,i);if(n)return n}let a=n.validators;if(a===null)return;let o=t===0?e.join(_e):e.slice(t).join(_e),s=a.length;for(let e=0;ee.slice(1,-1).indexOf(`:`)===-1?void 0:(()=>{let t=e.slice(1,-1),n=t.indexOf(`:`),r=t.slice(0,n);return r?ye+r:void 0})(),Ce=e=>{let{theme:t,classGroups:n}=e;return we(n,t)},we=(e,t)=>{let n=ge();for(let r in e){let i=e[r];Te(i,n,r,t)}return n},Te=(e,t,n,r)=>{let i=e.length;for(let a=0;a{if(typeof e==`string`){De(e,t,n);return}if(typeof e==`function`){Oe(e,t,n,r);return}ke(e,t,n,r)},De=(e,t,n)=>{let r=e===``?t:Ae(t,e);r.classGroupId=n},Oe=(e,t,n,r)=>{if(je(e)){Te(e(r),t,n,r);return}t.validators===null&&(t.validators=[]),t.validators.push(he(n,e))},ke=(e,t,n,r)=>{let i=Object.entries(e),a=i.length;for(let e=0;e{let n=e,r=t.split(_e),i=r.length;for(let e=0;e`isThemeGetter`in e&&e.isThemeGetter===!0,Me=e=>{if(e<1)return{get:()=>void 0,set:()=>{}};let t=0,n=Object.create(null),r=Object.create(null),i=(i,a)=>{n[i]=a,t++,t>e&&(t=0,r=n,n=Object.create(null))};return{get(e){let t=n[e];if(t!==void 0)return t;if((t=r[e])!==void 0)return i(e,t),t},set(e,t){e in n?n[e]=t:i(e,t)}}},Ne=`!`,Pe=`:`,Fe=[],Ie=(e,t,n,r,i)=>({modifiers:e,hasImportantModifier:t,baseClassName:n,maybePostfixModifierPosition:r,isExternal:i}),Le=e=>{let{prefix:t,experimentalParseClassName:n}=e,r=e=>{let t=[],n=0,r=0,i=0,a,o=e.length;for(let s=0;si?a-i:void 0;return Ie(t,l,c,u)};if(t){let e=t+Pe,n=r;r=t=>t.startsWith(e)?n(t.slice(e.length)):Ie(Fe,!1,t,void 0,!0)}if(n){let e=r;r=t=>n({className:t,parseClassName:e})}return r},Re=e=>{let t=new Map;return e.orderSensitiveModifiers.forEach((e,n)=>{t.set(e,1e6+n)}),e=>{let n=[],r=[];for(let i=0;i0&&(r.sort(),n.push(...r),r=[]),n.push(a)):r.push(a)}return r.length>0&&(r.sort(),n.push(...r)),n}},ze=e=>({cache:Me(e.cacheSize),parseClassName:Le(e),sortModifiers:Re(e),postfixLookupClassGroupIds:Be(e),...be(e)}),Be=e=>{let t=Object.create(null),n=e.postfixLookupClassGroups;if(n)for(let e=0;e{let{parseClassName:n,getClassGroupId:r,getConflictingClassGroupIds:i,sortModifiers:a,postfixLookupClassGroupIds:o}=t,s=[],c=e.trim().split(Ve),l=``;for(let e=c.length-1;e>=0;--e){let t=c[e],{isExternal:u,modifiers:d,hasImportantModifier:f,baseClassName:p,maybePostfixModifierPosition:m}=n(t);if(u){l=t+(l.length>0?` `+l:l);continue}let h=!!m,g;if(h){g=r(p.substring(0,m));let e=g&&o[g]?r(p):void 0;e&&e!==g&&(g=e,h=!1)}else g=r(p);if(!g){if(!h){l=t+(l.length>0?` `+l:l);continue}if(g=r(p),!g){l=t+(l.length>0?` `+l:l);continue}h=!1}let _=d.length===0?``:d.length===1?d[0]:a(d).join(`:`),v=f?_+Ne:_,y=v+g;if(s.indexOf(y)>-1)continue;s.push(y);let b=i(g,h);for(let e=0;e0?` `+l:l)}return l},Ue=(...e)=>{let t=0,n,r,i=``;for(;t{if(typeof e==`string`)return e;let t,n=``;for(let r=0;r{let n,r,i,a,o=o=>(n=ze(t.reduce((e,t)=>t(e),e())),r=n.cache.get,i=n.cache.set,a=s,s(o)),s=e=>{let t=r(e);if(t)return t;let a=He(e,n);return i(e,a),a};return a=o,(...e)=>a(Ue(...e))},Ke=[],qe=e=>{let t=t=>t[e]||Ke;return t.isThemeGetter=!0,t},Je=/^\[(?:(\w[\w-]*):)?(.+)\]$/i,Ye=/^\((?:(\w[\w-]*):)?(.+)\)$/i,Xe=/^\d+(?:\.\d+)?\/\d+(?:\.\d+)?$/,Ze=/^(\d+(\.\d+)?)?(xs|sm|md|lg|xl)$/,Qe=/\d+(%|px|r?em|[sdl]?v([hwib]|min|max)|pt|pc|in|cm|mm|cap|ch|ex|r?lh|cq(w|h|i|b|min|max))|\b(calc|min|max|clamp)\(.+\)|^0$/,$e=/^(rgba?|hsla?|hwb|(ok)?(lab|lch)|color-mix)\(.+\)$/,et=/^(inset_)?-?((\d+)?\.?(\d+)[a-z]+|0)_-?((\d+)?\.?(\d+)[a-z]+|0)/,tt=/^(url|image|image-set|cross-fade|element|(repeating-)?(linear|radial|conic)-gradient)\(.+\)$/,nt=e=>Xe.test(e),j=e=>!!e&&!Number.isNaN(Number(e)),rt=e=>!!e&&Number.isInteger(Number(e)),it=e=>e.endsWith(`%`)&&j(e.slice(0,-1)),at=e=>Ze.test(e),ot=()=>!0,st=e=>Qe.test(e)&&!$e.test(e),ct=()=>!1,lt=e=>et.test(e),ut=e=>tt.test(e),dt=e=>!M(e)&&!N(e),ft=e=>e.startsWith(`@container`)&&(e[10]===`/`&&e[11]!==void 0||e[11]===`s`&&e[16]!==void 0&&e.startsWith(`-size/`,10)||e[11]===`n`&&e[18]!==void 0&&e.startsWith(`-normal/`,10)),pt=e=>Tt(e,kt,ct),M=e=>Je.test(e),mt=e=>Tt(e,At,st),ht=e=>Tt(e,jt,j),gt=e=>Tt(e,Nt,ot),_t=e=>Tt(e,Mt,ct),vt=e=>Tt(e,Dt,ct),yt=e=>Tt(e,Ot,ut),bt=e=>Tt(e,Pt,lt),N=e=>Ye.test(e),xt=e=>Et(e,At),St=e=>Et(e,Mt),Ct=e=>Et(e,Dt),P=e=>Et(e,kt),F=e=>Et(e,Ot),I=e=>Et(e,Pt,!0),wt=e=>Et(e,Nt,!0),Tt=(e,t,n)=>{let r=Je.exec(e);return r?r[1]?t(r[1]):n(r[2]):!1},Et=(e,t,n=!1)=>{let r=Ye.exec(e);return r?r[1]?t(r[1]):n:!1},Dt=e=>e===`position`||e===`percentage`,Ot=e=>e===`image`||e===`url`,kt=e=>e===`length`||e===`size`||e===`bg-size`,At=e=>e===`length`,jt=e=>e===`number`,Mt=e=>e===`family-name`,Nt=e=>e===`number`||e===`weight`,Pt=e=>e===`shadow`,Ft=Ge(()=>{let e=qe(`color`),t=qe(`font`),n=qe(`text`),r=qe(`font-weight`),i=qe(`tracking`),a=qe(`leading`),o=qe(`breakpoint`),s=qe(`container`),c=qe(`spacing`),l=qe(`radius`),u=qe(`shadow`),d=qe(`inset-shadow`),f=qe(`text-shadow`),p=qe(`drop-shadow`),m=qe(`blur`),h=qe(`perspective`),g=qe(`aspect`),_=qe(`ease`),v=qe(`animate`),y=()=>[`auto`,`avoid`,`all`,`avoid-page`,`page`,`left`,`right`,`column`],b=()=>[`center`,`top`,`bottom`,`left`,`right`,`top-left`,`left-top`,`top-right`,`right-top`,`bottom-right`,`right-bottom`,`bottom-left`,`left-bottom`],x=()=>[...b(),N,M],ee=()=>[`auto`,`hidden`,`clip`,`visible`,`scroll`],te=()=>[`auto`,`contain`,`none`],S=()=>[N,M,c],C=()=>[nt,`full`,`auto`,...S()],ne=()=>[rt,`none`,`subgrid`,N,M],re=()=>[`auto`,{span:[`full`,rt,N,M]},rt,N,M],ie=()=>[rt,`auto`,N,M],ae=()=>[`auto`,`min`,`max`,`fr`,N,M],oe=()=>[`start`,`end`,`center`,`between`,`around`,`evenly`,`stretch`,`baseline`,`center-safe`,`end-safe`],se=()=>[`start`,`end`,`center`,`stretch`,`center-safe`,`end-safe`],ce=()=>[`auto`,...S()],w=()=>[nt,`auto`,`full`,`dvw`,`dvh`,`lvw`,`lvh`,`svw`,`svh`,`min`,`max`,`fit`,...S()],T=()=>[nt,`screen`,`full`,`dvw`,`lvw`,`svw`,`min`,`max`,`fit`,...S()],le=()=>[nt,`screen`,`full`,`lh`,`dvh`,`lvh`,`svh`,`min`,`max`,`fit`,...S()],E=()=>[e,N,M],ue=()=>[...b(),Ct,vt,{position:[N,M]}],de=()=>[`no-repeat`,{repeat:[``,`x`,`y`,`space`,`round`]}],fe=()=>[`auto`,`cover`,`contain`,P,pt,{size:[N,M]}],pe=()=>[it,xt,mt],D=()=>[``,`none`,`full`,l,N,M],O=()=>[``,j,xt,mt],k=()=>[`solid`,`dashed`,`dotted`,`double`],me=()=>[`normal`,`multiply`,`screen`,`overlay`,`darken`,`lighten`,`color-dodge`,`color-burn`,`hard-light`,`soft-light`,`difference`,`exclusion`,`hue`,`saturation`,`color`,`luminosity`],A=()=>[j,it,Ct,vt],he=()=>[``,`none`,m,N,M],ge=()=>[`none`,j,N,M],_e=()=>[`none`,j,N,M],ve=()=>[j,N,M],ye=()=>[nt,`full`,...S()];return{cacheSize:500,theme:{animate:[`spin`,`ping`,`pulse`,`bounce`],aspect:[`video`],blur:[at],breakpoint:[at],color:[ot],container:[at],"drop-shadow":[at],ease:[`in`,`out`,`in-out`],font:[dt],"font-weight":[`thin`,`extralight`,`light`,`normal`,`medium`,`semibold`,`bold`,`extrabold`,`black`],"inset-shadow":[at],leading:[`none`,`tight`,`snug`,`normal`,`relaxed`,`loose`],perspective:[`dramatic`,`near`,`normal`,`midrange`,`distant`,`none`],radius:[at],shadow:[at],spacing:[`px`,j],text:[at],"text-shadow":[at],tracking:[`tighter`,`tight`,`normal`,`wide`,`wider`,`widest`]},classGroups:{aspect:[{aspect:[`auto`,`square`,nt,M,N,g]}],container:[`container`],"container-type":[{"@container":[``,`normal`,`size`,N,M]}],"container-named":[ft],columns:[{columns:[j,M,N,s]}],"break-after":[{"break-after":y()}],"break-before":[{"break-before":y()}],"break-inside":[{"break-inside":[`auto`,`avoid`,`avoid-page`,`avoid-column`]}],"box-decoration":[{"box-decoration":[`slice`,`clone`]}],box:[{box:[`border`,`content`]}],display:[`block`,`inline-block`,`inline`,`flex`,`inline-flex`,`table`,`inline-table`,`table-caption`,`table-cell`,`table-column`,`table-column-group`,`table-footer-group`,`table-header-group`,`table-row-group`,`table-row`,`flow-root`,`grid`,`inline-grid`,`contents`,`list-item`,`hidden`],sr:[`sr-only`,`not-sr-only`],float:[{float:[`right`,`left`,`none`,`start`,`end`]}],clear:[{clear:[`left`,`right`,`both`,`none`,`start`,`end`]}],isolation:[`isolate`,`isolation-auto`],"object-fit":[{object:[`contain`,`cover`,`fill`,`none`,`scale-down`]}],"object-position":[{object:x()}],overflow:[{overflow:ee()}],"overflow-x":[{"overflow-x":ee()}],"overflow-y":[{"overflow-y":ee()}],overscroll:[{overscroll:te()}],"overscroll-x":[{"overscroll-x":te()}],"overscroll-y":[{"overscroll-y":te()}],position:[`static`,`fixed`,`absolute`,`relative`,`sticky`],inset:[{inset:C()}],"inset-x":[{"inset-x":C()}],"inset-y":[{"inset-y":C()}],start:[{"inset-s":C(),start:C()}],end:[{"inset-e":C(),end:C()}],"inset-bs":[{"inset-bs":C()}],"inset-be":[{"inset-be":C()}],top:[{top:C()}],right:[{right:C()}],bottom:[{bottom:C()}],left:[{left:C()}],visibility:[`visible`,`invisible`,`collapse`],z:[{z:[rt,`auto`,N,M]}],basis:[{basis:[nt,`full`,`auto`,s,...S()]}],"flex-direction":[{flex:[`row`,`row-reverse`,`col`,`col-reverse`]}],"flex-wrap":[{flex:[`nowrap`,`wrap`,`wrap-reverse`]}],flex:[{flex:[j,nt,`auto`,`initial`,`none`,M]}],grow:[{grow:[``,j,N,M]}],shrink:[{shrink:[``,j,N,M]}],order:[{order:[rt,`first`,`last`,`none`,N,M]}],"grid-cols":[{"grid-cols":ne()}],"col-start-end":[{col:re()}],"col-start":[{"col-start":ie()}],"col-end":[{"col-end":ie()}],"grid-rows":[{"grid-rows":ne()}],"row-start-end":[{row:re()}],"row-start":[{"row-start":ie()}],"row-end":[{"row-end":ie()}],"grid-flow":[{"grid-flow":[`row`,`col`,`dense`,`row-dense`,`col-dense`]}],"auto-cols":[{"auto-cols":ae()}],"auto-rows":[{"auto-rows":ae()}],gap:[{gap:S()}],"gap-x":[{"gap-x":S()}],"gap-y":[{"gap-y":S()}],"justify-content":[{justify:[...oe(),`normal`]}],"justify-items":[{"justify-items":[...se(),`normal`]}],"justify-self":[{"justify-self":[`auto`,...se()]}],"align-content":[{content:[`normal`,...oe()]}],"align-items":[{items:[...se(),{baseline:[``,`last`]}]}],"align-self":[{self:[`auto`,...se(),{baseline:[``,`last`]}]}],"place-content":[{"place-content":oe()}],"place-items":[{"place-items":[...se(),`baseline`]}],"place-self":[{"place-self":[`auto`,...se()]}],p:[{p:S()}],px:[{px:S()}],py:[{py:S()}],ps:[{ps:S()}],pe:[{pe:S()}],pbs:[{pbs:S()}],pbe:[{pbe:S()}],pt:[{pt:S()}],pr:[{pr:S()}],pb:[{pb:S()}],pl:[{pl:S()}],m:[{m:ce()}],mx:[{mx:ce()}],my:[{my:ce()}],ms:[{ms:ce()}],me:[{me:ce()}],mbs:[{mbs:ce()}],mbe:[{mbe:ce()}],mt:[{mt:ce()}],mr:[{mr:ce()}],mb:[{mb:ce()}],ml:[{ml:ce()}],"space-x":[{"space-x":S()}],"space-x-reverse":[`space-x-reverse`],"space-y":[{"space-y":S()}],"space-y-reverse":[`space-y-reverse`],size:[{size:w()}],"inline-size":[{inline:[`auto`,...T()]}],"min-inline-size":[{"min-inline":[`auto`,...T()]}],"max-inline-size":[{"max-inline":[`none`,...T()]}],"block-size":[{block:[`auto`,...le()]}],"min-block-size":[{"min-block":[`auto`,...le()]}],"max-block-size":[{"max-block":[`none`,...le()]}],w:[{w:[s,`screen`,...w()]}],"min-w":[{"min-w":[s,`screen`,`none`,...w()]}],"max-w":[{"max-w":[s,`screen`,`none`,`prose`,{screen:[o]},...w()]}],h:[{h:[`screen`,`lh`,...w()]}],"min-h":[{"min-h":[`screen`,`lh`,`none`,...w()]}],"max-h":[{"max-h":[`screen`,`lh`,...w()]}],"font-size":[{text:[`base`,n,xt,mt]}],"font-smoothing":[`antialiased`,`subpixel-antialiased`],"font-style":[`italic`,`not-italic`],"font-weight":[{font:[r,wt,gt]}],"font-stretch":[{"font-stretch":[`ultra-condensed`,`extra-condensed`,`condensed`,`semi-condensed`,`normal`,`semi-expanded`,`expanded`,`extra-expanded`,`ultra-expanded`,it,M]}],"font-family":[{font:[St,_t,t]}],"font-features":[{"font-features":[M]}],"fvn-normal":[`normal-nums`],"fvn-ordinal":[`ordinal`],"fvn-slashed-zero":[`slashed-zero`],"fvn-figure":[`lining-nums`,`oldstyle-nums`],"fvn-spacing":[`proportional-nums`,`tabular-nums`],"fvn-fraction":[`diagonal-fractions`,`stacked-fractions`],tracking:[{tracking:[i,N,M]}],"line-clamp":[{"line-clamp":[j,`none`,N,ht]}],leading:[{leading:[a,...S()]}],"list-image":[{"list-image":[`none`,N,M]}],"list-style-position":[{list:[`inside`,`outside`]}],"list-style-type":[{list:[`disc`,`decimal`,`none`,N,M]}],"text-alignment":[{text:[`left`,`center`,`right`,`justify`,`start`,`end`]}],"placeholder-color":[{placeholder:E()}],"text-color":[{text:E()}],"text-decoration":[`underline`,`overline`,`line-through`,`no-underline`],"text-decoration-style":[{decoration:[...k(),`wavy`]}],"text-decoration-thickness":[{decoration:[j,`from-font`,`auto`,N,mt]}],"text-decoration-color":[{decoration:E()}],"underline-offset":[{"underline-offset":[j,`auto`,N,M]}],"text-transform":[`uppercase`,`lowercase`,`capitalize`,`normal-case`],"text-overflow":[`truncate`,`text-ellipsis`,`text-clip`],"text-wrap":[{text:[`wrap`,`nowrap`,`balance`,`pretty`]}],indent:[{indent:S()}],"tab-size":[{tab:[rt,N,M]}],"vertical-align":[{align:[`baseline`,`top`,`middle`,`bottom`,`text-top`,`text-bottom`,`sub`,`super`,N,M]}],whitespace:[{whitespace:[`normal`,`nowrap`,`pre`,`pre-line`,`pre-wrap`,`break-spaces`]}],break:[{break:[`normal`,`words`,`all`,`keep`]}],wrap:[{wrap:[`break-word`,`anywhere`,`normal`]}],hyphens:[{hyphens:[`none`,`manual`,`auto`]}],content:[{content:[`none`,N,M]}],"bg-attachment":[{bg:[`fixed`,`local`,`scroll`]}],"bg-clip":[{"bg-clip":[`border`,`padding`,`content`,`text`]}],"bg-origin":[{"bg-origin":[`border`,`padding`,`content`]}],"bg-position":[{bg:ue()}],"bg-repeat":[{bg:de()}],"bg-size":[{bg:fe()}],"bg-image":[{bg:[`none`,{linear:[{to:[`t`,`tr`,`r`,`br`,`b`,`bl`,`l`,`tl`]},rt,N,M],radial:[``,N,M],conic:[rt,N,M]},F,yt]}],"bg-color":[{bg:E()}],"gradient-from-pos":[{from:pe()}],"gradient-via-pos":[{via:pe()}],"gradient-to-pos":[{to:pe()}],"gradient-from":[{from:E()}],"gradient-via":[{via:E()}],"gradient-to":[{to:E()}],rounded:[{rounded:D()}],"rounded-s":[{"rounded-s":D()}],"rounded-e":[{"rounded-e":D()}],"rounded-t":[{"rounded-t":D()}],"rounded-r":[{"rounded-r":D()}],"rounded-b":[{"rounded-b":D()}],"rounded-l":[{"rounded-l":D()}],"rounded-ss":[{"rounded-ss":D()}],"rounded-se":[{"rounded-se":D()}],"rounded-ee":[{"rounded-ee":D()}],"rounded-es":[{"rounded-es":D()}],"rounded-tl":[{"rounded-tl":D()}],"rounded-tr":[{"rounded-tr":D()}],"rounded-br":[{"rounded-br":D()}],"rounded-bl":[{"rounded-bl":D()}],"border-w":[{border:O()}],"border-w-x":[{"border-x":O()}],"border-w-y":[{"border-y":O()}],"border-w-s":[{"border-s":O()}],"border-w-e":[{"border-e":O()}],"border-w-bs":[{"border-bs":O()}],"border-w-be":[{"border-be":O()}],"border-w-t":[{"border-t":O()}],"border-w-r":[{"border-r":O()}],"border-w-b":[{"border-b":O()}],"border-w-l":[{"border-l":O()}],"divide-x":[{"divide-x":O()}],"divide-x-reverse":[`divide-x-reverse`],"divide-y":[{"divide-y":O()}],"divide-y-reverse":[`divide-y-reverse`],"border-style":[{border:[...k(),`hidden`,`none`]}],"divide-style":[{divide:[...k(),`hidden`,`none`]}],"border-color":[{border:E()}],"border-color-x":[{"border-x":E()}],"border-color-y":[{"border-y":E()}],"border-color-s":[{"border-s":E()}],"border-color-e":[{"border-e":E()}],"border-color-bs":[{"border-bs":E()}],"border-color-be":[{"border-be":E()}],"border-color-t":[{"border-t":E()}],"border-color-r":[{"border-r":E()}],"border-color-b":[{"border-b":E()}],"border-color-l":[{"border-l":E()}],"divide-color":[{divide:E()}],"outline-style":[{outline:[...k(),`none`,`hidden`]}],"outline-offset":[{"outline-offset":[j,N,M]}],"outline-w":[{outline:[``,j,xt,mt]}],"outline-color":[{outline:E()}],shadow:[{shadow:[``,`none`,u,I,bt]}],"shadow-color":[{shadow:E()}],"inset-shadow":[{"inset-shadow":[`none`,d,I,bt]}],"inset-shadow-color":[{"inset-shadow":E()}],"ring-w":[{ring:O()}],"ring-w-inset":[`ring-inset`],"ring-color":[{ring:E()}],"ring-offset-w":[{"ring-offset":[j,mt]}],"ring-offset-color":[{"ring-offset":E()}],"inset-ring-w":[{"inset-ring":O()}],"inset-ring-color":[{"inset-ring":E()}],"text-shadow":[{"text-shadow":[`none`,f,I,bt]}],"text-shadow-color":[{"text-shadow":E()}],opacity:[{opacity:[j,N,M]}],"mix-blend":[{"mix-blend":[...me(),`plus-darker`,`plus-lighter`]}],"bg-blend":[{"bg-blend":me()}],"mask-clip":[{"mask-clip":[`border`,`padding`,`content`,`fill`,`stroke`,`view`]},`mask-no-clip`],"mask-composite":[{mask:[`add`,`subtract`,`intersect`,`exclude`]}],"mask-image-linear-pos":[{"mask-linear":[j]}],"mask-image-linear-from-pos":[{"mask-linear-from":A()}],"mask-image-linear-to-pos":[{"mask-linear-to":A()}],"mask-image-linear-from-color":[{"mask-linear-from":E()}],"mask-image-linear-to-color":[{"mask-linear-to":E()}],"mask-image-t-from-pos":[{"mask-t-from":A()}],"mask-image-t-to-pos":[{"mask-t-to":A()}],"mask-image-t-from-color":[{"mask-t-from":E()}],"mask-image-t-to-color":[{"mask-t-to":E()}],"mask-image-r-from-pos":[{"mask-r-from":A()}],"mask-image-r-to-pos":[{"mask-r-to":A()}],"mask-image-r-from-color":[{"mask-r-from":E()}],"mask-image-r-to-color":[{"mask-r-to":E()}],"mask-image-b-from-pos":[{"mask-b-from":A()}],"mask-image-b-to-pos":[{"mask-b-to":A()}],"mask-image-b-from-color":[{"mask-b-from":E()}],"mask-image-b-to-color":[{"mask-b-to":E()}],"mask-image-l-from-pos":[{"mask-l-from":A()}],"mask-image-l-to-pos":[{"mask-l-to":A()}],"mask-image-l-from-color":[{"mask-l-from":E()}],"mask-image-l-to-color":[{"mask-l-to":E()}],"mask-image-x-from-pos":[{"mask-x-from":A()}],"mask-image-x-to-pos":[{"mask-x-to":A()}],"mask-image-x-from-color":[{"mask-x-from":E()}],"mask-image-x-to-color":[{"mask-x-to":E()}],"mask-image-y-from-pos":[{"mask-y-from":A()}],"mask-image-y-to-pos":[{"mask-y-to":A()}],"mask-image-y-from-color":[{"mask-y-from":E()}],"mask-image-y-to-color":[{"mask-y-to":E()}],"mask-image-radial":[{"mask-radial":[N,M]}],"mask-image-radial-from-pos":[{"mask-radial-from":A()}],"mask-image-radial-to-pos":[{"mask-radial-to":A()}],"mask-image-radial-from-color":[{"mask-radial-from":E()}],"mask-image-radial-to-color":[{"mask-radial-to":E()}],"mask-image-radial-shape":[{"mask-radial":[`circle`,`ellipse`]}],"mask-image-radial-size":[{"mask-radial":[{closest:[`side`,`corner`],farthest:[`side`,`corner`]}]}],"mask-image-radial-pos":[{"mask-radial-at":b()}],"mask-image-conic-pos":[{"mask-conic":[j]}],"mask-image-conic-from-pos":[{"mask-conic-from":A()}],"mask-image-conic-to-pos":[{"mask-conic-to":A()}],"mask-image-conic-from-color":[{"mask-conic-from":E()}],"mask-image-conic-to-color":[{"mask-conic-to":E()}],"mask-mode":[{mask:[`alpha`,`luminance`,`match`]}],"mask-origin":[{"mask-origin":[`border`,`padding`,`content`,`fill`,`stroke`,`view`]}],"mask-position":[{mask:ue()}],"mask-repeat":[{mask:de()}],"mask-size":[{mask:fe()}],"mask-type":[{"mask-type":[`alpha`,`luminance`]}],"mask-image":[{mask:[`none`,N,M]}],filter:[{filter:[``,`none`,N,M]}],blur:[{blur:he()}],brightness:[{brightness:[j,N,M]}],contrast:[{contrast:[j,N,M]}],"drop-shadow":[{"drop-shadow":[``,`none`,p,I,bt]}],"drop-shadow-color":[{"drop-shadow":E()}],grayscale:[{grayscale:[``,j,N,M]}],"hue-rotate":[{"hue-rotate":[j,N,M]}],invert:[{invert:[``,j,N,M]}],saturate:[{saturate:[j,N,M]}],sepia:[{sepia:[``,j,N,M]}],"backdrop-filter":[{"backdrop-filter":[``,`none`,N,M]}],"backdrop-blur":[{"backdrop-blur":he()}],"backdrop-brightness":[{"backdrop-brightness":[j,N,M]}],"backdrop-contrast":[{"backdrop-contrast":[j,N,M]}],"backdrop-grayscale":[{"backdrop-grayscale":[``,j,N,M]}],"backdrop-hue-rotate":[{"backdrop-hue-rotate":[j,N,M]}],"backdrop-invert":[{"backdrop-invert":[``,j,N,M]}],"backdrop-opacity":[{"backdrop-opacity":[j,N,M]}],"backdrop-saturate":[{"backdrop-saturate":[j,N,M]}],"backdrop-sepia":[{"backdrop-sepia":[``,j,N,M]}],"border-collapse":[{border:[`collapse`,`separate`]}],"border-spacing":[{"border-spacing":S()}],"border-spacing-x":[{"border-spacing-x":S()}],"border-spacing-y":[{"border-spacing-y":S()}],"table-layout":[{table:[`auto`,`fixed`]}],caption:[{caption:[`top`,`bottom`]}],transition:[{transition:[``,`all`,`colors`,`opacity`,`shadow`,`transform`,`none`,N,M]}],"transition-behavior":[{transition:[`normal`,`discrete`]}],duration:[{duration:[j,`initial`,N,M]}],ease:[{ease:[`linear`,`initial`,_,N,M]}],delay:[{delay:[j,N,M]}],animate:[{animate:[`none`,v,N,M]}],backface:[{backface:[`hidden`,`visible`]}],perspective:[{perspective:[h,N,M]}],"perspective-origin":[{"perspective-origin":x()}],rotate:[{rotate:ge()}],"rotate-x":[{"rotate-x":ge()}],"rotate-y":[{"rotate-y":ge()}],"rotate-z":[{"rotate-z":ge()}],scale:[{scale:_e()}],"scale-x":[{"scale-x":_e()}],"scale-y":[{"scale-y":_e()}],"scale-z":[{"scale-z":_e()}],"scale-3d":[`scale-3d`],skew:[{skew:ve()}],"skew-x":[{"skew-x":ve()}],"skew-y":[{"skew-y":ve()}],transform:[{transform:[N,M,``,`none`,`gpu`,`cpu`]}],"transform-origin":[{origin:x()}],"transform-style":[{transform:[`3d`,`flat`]}],translate:[{translate:ye()}],"translate-x":[{"translate-x":ye()}],"translate-y":[{"translate-y":ye()}],"translate-z":[{"translate-z":ye()}],"translate-none":[`translate-none`],zoom:[{zoom:[rt,N,M]}],accent:[{accent:E()}],appearance:[{appearance:[`none`,`auto`]}],"caret-color":[{caret:E()}],"color-scheme":[{scheme:[`normal`,`dark`,`light`,`light-dark`,`only-dark`,`only-light`]}],cursor:[{cursor:[`auto`,`default`,`pointer`,`wait`,`text`,`move`,`help`,`not-allowed`,`none`,`context-menu`,`progress`,`cell`,`crosshair`,`vertical-text`,`alias`,`copy`,`no-drop`,`grab`,`grabbing`,`all-scroll`,`col-resize`,`row-resize`,`n-resize`,`e-resize`,`s-resize`,`w-resize`,`ne-resize`,`nw-resize`,`se-resize`,`sw-resize`,`ew-resize`,`ns-resize`,`nesw-resize`,`nwse-resize`,`zoom-in`,`zoom-out`,N,M]}],"field-sizing":[{"field-sizing":[`fixed`,`content`]}],"pointer-events":[{"pointer-events":[`auto`,`none`]}],resize:[{resize:[`none`,``,`y`,`x`]}],"scroll-behavior":[{scroll:[`auto`,`smooth`]}],"scrollbar-thumb-color":[{"scrollbar-thumb":E()}],"scrollbar-track-color":[{"scrollbar-track":E()}],"scrollbar-gutter":[{"scrollbar-gutter":[`auto`,`stable`,`both`]}],"scrollbar-w":[{scrollbar:[`auto`,`thin`,`none`]}],"scroll-m":[{"scroll-m":S()}],"scroll-mx":[{"scroll-mx":S()}],"scroll-my":[{"scroll-my":S()}],"scroll-ms":[{"scroll-ms":S()}],"scroll-me":[{"scroll-me":S()}],"scroll-mbs":[{"scroll-mbs":S()}],"scroll-mbe":[{"scroll-mbe":S()}],"scroll-mt":[{"scroll-mt":S()}],"scroll-mr":[{"scroll-mr":S()}],"scroll-mb":[{"scroll-mb":S()}],"scroll-ml":[{"scroll-ml":S()}],"scroll-p":[{"scroll-p":S()}],"scroll-px":[{"scroll-px":S()}],"scroll-py":[{"scroll-py":S()}],"scroll-ps":[{"scroll-ps":S()}],"scroll-pe":[{"scroll-pe":S()}],"scroll-pbs":[{"scroll-pbs":S()}],"scroll-pbe":[{"scroll-pbe":S()}],"scroll-pt":[{"scroll-pt":S()}],"scroll-pr":[{"scroll-pr":S()}],"scroll-pb":[{"scroll-pb":S()}],"scroll-pl":[{"scroll-pl":S()}],"snap-align":[{snap:[`start`,`end`,`center`,`align-none`]}],"snap-stop":[{snap:[`normal`,`always`]}],"snap-type":[{snap:[`none`,`x`,`y`,`both`]}],"snap-strictness":[{snap:[`mandatory`,`proximity`]}],touch:[{touch:[`auto`,`none`,`manipulation`]}],"touch-x":[{"touch-pan":[`x`,`left`,`right`]}],"touch-y":[{"touch-pan":[`y`,`up`,`down`]}],"touch-pz":[`touch-pinch-zoom`],select:[{select:[`none`,`text`,`all`,`auto`]}],"will-change":[{"will-change":[`auto`,`scroll`,`contents`,`transform`,N,M]}],fill:[{fill:[`none`,...E()]}],"stroke-w":[{stroke:[j,xt,mt,ht]}],stroke:[{stroke:[`none`,...E()]}],"forced-color-adjust":[{"forced-color-adjust":[`auto`,`none`]}]},conflictingClassGroups:{"container-named":[`container-type`],overflow:[`overflow-x`,`overflow-y`],overscroll:[`overscroll-x`,`overscroll-y`],inset:[`inset-x`,`inset-y`,`inset-bs`,`inset-be`,`start`,`end`,`top`,`right`,`bottom`,`left`],"inset-x":[`right`,`left`],"inset-y":[`top`,`bottom`],flex:[`basis`,`grow`,`shrink`],gap:[`gap-x`,`gap-y`],p:[`px`,`py`,`ps`,`pe`,`pbs`,`pbe`,`pt`,`pr`,`pb`,`pl`],px:[`pr`,`pl`],py:[`pt`,`pb`],m:[`mx`,`my`,`ms`,`me`,`mbs`,`mbe`,`mt`,`mr`,`mb`,`ml`],mx:[`mr`,`ml`],my:[`mt`,`mb`],size:[`w`,`h`],"font-size":[`leading`],"fvn-normal":[`fvn-ordinal`,`fvn-slashed-zero`,`fvn-figure`,`fvn-spacing`,`fvn-fraction`],"fvn-ordinal":[`fvn-normal`],"fvn-slashed-zero":[`fvn-normal`],"fvn-figure":[`fvn-normal`],"fvn-spacing":[`fvn-normal`],"fvn-fraction":[`fvn-normal`],"line-clamp":[`display`,`overflow`],rounded:[`rounded-s`,`rounded-e`,`rounded-t`,`rounded-r`,`rounded-b`,`rounded-l`,`rounded-ss`,`rounded-se`,`rounded-ee`,`rounded-es`,`rounded-tl`,`rounded-tr`,`rounded-br`,`rounded-bl`],"rounded-s":[`rounded-ss`,`rounded-es`],"rounded-e":[`rounded-se`,`rounded-ee`],"rounded-t":[`rounded-tl`,`rounded-tr`],"rounded-r":[`rounded-tr`,`rounded-br`],"rounded-b":[`rounded-br`,`rounded-bl`],"rounded-l":[`rounded-tl`,`rounded-bl`],"border-spacing":[`border-spacing-x`,`border-spacing-y`],"border-w":[`border-w-x`,`border-w-y`,`border-w-s`,`border-w-e`,`border-w-bs`,`border-w-be`,`border-w-t`,`border-w-r`,`border-w-b`,`border-w-l`],"border-w-x":[`border-w-r`,`border-w-l`],"border-w-y":[`border-w-t`,`border-w-b`],"border-color":[`border-color-x`,`border-color-y`,`border-color-s`,`border-color-e`,`border-color-bs`,`border-color-be`,`border-color-t`,`border-color-r`,`border-color-b`,`border-color-l`],"border-color-x":[`border-color-r`,`border-color-l`],"border-color-y":[`border-color-t`,`border-color-b`],translate:[`translate-x`,`translate-y`,`translate-none`],"translate-none":[`translate`,`translate-x`,`translate-y`,`translate-z`],"scroll-m":[`scroll-mx`,`scroll-my`,`scroll-ms`,`scroll-me`,`scroll-mbs`,`scroll-mbe`,`scroll-mt`,`scroll-mr`,`scroll-mb`,`scroll-ml`],"scroll-mx":[`scroll-mr`,`scroll-ml`],"scroll-my":[`scroll-mt`,`scroll-mb`],"scroll-p":[`scroll-px`,`scroll-py`,`scroll-ps`,`scroll-pe`,`scroll-pbs`,`scroll-pbe`,`scroll-pt`,`scroll-pr`,`scroll-pb`,`scroll-pl`],"scroll-px":[`scroll-pr`,`scroll-pl`],"scroll-py":[`scroll-pt`,`scroll-pb`],touch:[`touch-x`,`touch-y`,`touch-pz`],"touch-x":[`touch`],"touch-y":[`touch`],"touch-pz":[`touch`]},conflictingClassGroupModifiers:{"font-size":[`leading`]},postfixLookupClassGroups:[`container-type`],orderSensitiveModifiers:[`*`,`**`,`after`,`backdrop`,`before`,`details-content`,`file`,`first-letter`,`first-line`,`marker`,`placeholder`,`selection`]}});function L(...e){return Ft(D(e))}var It=e((e=>{var t=Symbol.for(`react.transitional.element`),n=Symbol.for(`react.fragment`);function r(e,n,r){var i=null;if(r!==void 0&&(i=``+r),n.key!==void 0&&(i=``+n.key),`key`in n)for(var a in r={},n)a!==`key`&&(r[a]=n[a]);else r=n;return n=r.ref,{$$typeof:t,type:e,key:i,ref:n===void 0?null:n,props:r}}e.Fragment=n,e.jsx=r,e.jsxs=r})),R=e(((e,t)=>{t.exports=It()}))(),Lt=me(`inline-flex min-h-[22px] items-center rounded-full border px-2 py-0.5 text-[11px] font-extrabold uppercase leading-tight`,{variants:{variant:{neutral:`border-border bg-secondary text-muted-foreground`,succeeded:`border-emerald-400/35 bg-emerald-500/15 text-emerald-300`,failed:`border-red-400/40 bg-red-500/15 text-red-300`,dead:`border-red-400/40 bg-red-500/15 text-red-300`,missing:`border-red-400/40 bg-red-500/15 text-red-300`,running:`border-amber-400/40 bg-amber-500/15 text-amber-300`,queued:`border-teal-400/40 bg-teal-500/15 text-teal-200`,canceled:`border-border bg-secondary text-muted-foreground`}},defaultVariants:{variant:`neutral`}});function z({className:e,variant:t,...n}){return(0,R.jsx)(`span`,{"data-slot":`badge`,className:L(Lt({variant:t,className:e})),...n})}var Rt=me(`inline-flex min-h-9 shrink-0 items-center justify-center gap-2 whitespace-nowrap rounded-md border text-sm font-semibold transition-colors focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 [&_svg]:shrink-0`,{variants:{variant:{default:`border-primary bg-primary text-primary-foreground hover:bg-primary/90`,secondary:`border-border bg-secondary text-secondary-foreground hover:bg-secondary/80`,outline:`border-border bg-background hover:bg-accent hover:text-accent-foreground`,ghost:`border-transparent hover:bg-accent hover:text-accent-foreground`,destructive:`border-destructive bg-destructive text-white hover:bg-destructive/90`},size:{default:`h-9 px-4 py-2`,sm:`h-8 rounded-md px-3 text-xs`,icon:`size-9`}},defaultVariants:{variant:`secondary`,size:`default`}});function B({className:e,variant:t,size:n,type:r=`button`,...i}){return(0,R.jsx)(`button`,{"data-slot":`button`,type:r,className:L(Rt({variant:t,size:n,className:e})),...i})}function V({className:e,...t}){return(0,R.jsx)(`div`,{"data-slot":`card`,className:L(`rounded-lg border bg-card text-card-foreground shadow-[0_18px_44px_rgb(0_0_0/0.22)]`,e),...t})}function H({className:e,...t}){return(0,R.jsx)(`div`,{"data-slot":`card-header`,className:L(`flex items-center justify-between gap-3 border-b px-4 py-3`,e),...t})}function zt({className:e,...t}){return(0,R.jsx)(`h2`,{"data-slot":`card-title`,className:L(`text-[15px] font-bold`,e),...t})}function Bt({className:e,...t}){return(0,R.jsx)(`div`,{"data-slot":`card-content`,className:L(`p-4`,e),...t})}function U({className:e,type:t,...n}){return(0,R.jsx)(`input`,{"data-slot":`input`,type:t,className:L(`flex min-h-9 w-full rounded-md border border-input bg-background px-3 py-2 text-sm text-foreground shadow-xs transition-colors placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50`,e),...n})}function W({className:e,...t}){return(0,R.jsx)(`label`,{"data-slot":`label`,className:L(`grid gap-1.5 text-xs font-bold text-muted-foreground`,e),...t})}function Vt({className:e,...t}){return(0,R.jsx)(`select`,{"data-slot":`select`,className:L(`flex min-h-9 w-full rounded-md border border-input bg-background px-3 py-2 text-sm text-foreground shadow-xs transition-colors focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50`,e),...t})}function Ht({className:e,...t}){return(0,R.jsx)(`table`,{"data-slot":`table`,className:L(`w-full border-collapse text-sm`,e),...t})}function Ut({className:e,...t}){return(0,R.jsx)(`thead`,{"data-slot":`table-header`,className:e,...t})}function Wt({className:e,...t}){return(0,R.jsx)(`tbody`,{"data-slot":`table-body`,className:e,...t})}function Gt({className:e,...t}){return(0,R.jsx)(`tr`,{"data-slot":`table-row`,className:L(`border-b transition-colors hover:bg-muted/45`,e),...t})}function G({className:e,...t}){return(0,R.jsx)(`th`,{"data-slot":`table-head`,className:L(`bg-secondary px-3 py-3 text-left align-middle text-xs font-extrabold text-muted-foreground`,e),...t})}function K({className:e,...t}){return(0,R.jsx)(`td`,{"data-slot":`table-cell`,className:L(`px-3 py-3 align-middle text-sm`,e),...t})}var Kt={pending:`Needs review`,selected:`Assigned to onboarder`,reachingout:`Reaching out`,awaitingcontribution:`Awaiting contribution`,onboarded:`Onboarded`,waitlist:`Waitlist`,rejected:`Rejected`};function qt(e){if(!e)return``;let t=new Date(e);return Number.isNaN(t.getTime())?e:t.toLocaleString(void 0,{year:`numeric`,month:`short`,day:`numeric`,hour:`2-digit`,minute:`2-digit`})}function Jt(e,t=new Date){if(!e)return null;let n=new Date(e);if(Number.isNaN(n.getTime()))return null;let r=t.getTime()-n.getTime();return r<0?0:Math.floor(r/864e5)}function Yt(e){return e==null?``:JSON.stringify(e,null,2)}function Xt(e){return e.onboarding_state||e.onboardingState||e.cOnboardingState||``}function Zt(e){let t=String(e||``).trim();if(!t)return`No status`;let n=t.toLowerCase();return Kt[n]?Kt[n]:t.replace(/[-_]+/g,` `).replace(/\s+/g,` `).trim().replace(/\b\w/g,e=>e.toUpperCase())}function Qt(e){let t=String(e||``).trim().toLowerCase();return!t||t===`pending`?`neutral`:t===`selected`?`queued`:t===`rejected`?`failed`:t===`onboarded`?`succeeded`:t===`waitlist`?`running`:`queued`}function $t(e){let t=String(e||``).trim();return!t||t.toLowerCase()===`none`?``:t}function en(e){let t=String(e||``).trim();return t?/^https?:\/\//i.test(t)?t:`https://${t.replace(/^\/+/,``)}`:``}function tn(e){try{return new URL(en(e))}catch{return null}}function nn(e,t){let n=e.toLowerCase();return n===t||n.endsWith(`.${t}`)}function rn(e){return e.split(`/`).filter(Boolean).map(e=>encodeURIComponent(e)).join(`/`)}function an(e){let t=String(e||``).trim();if(!t)return``;let n=tn(t);if(n&&nn(n.hostname,`linkedin.com`))return n.href;if(/^https?:\/\//i.test(t))return``;let r=t.replace(/^@/,``).replace(/^\/+|\/+$/g,``).replace(/^in\//i,``);return r?`https://www.linkedin.com/in/${rn(r)}`:``}function on(e){let t=String(e||``).trim().replace(/^@/,``);if(!t)return``;let n=tn(t);if(n&&nn(n.hostname,`github.com`))return n.href;if(/^https?:\/\//i.test(t))return``;let r=t.replace(/^\/+|\/+$/g,``);return r?`https://github.com/${rn(r)}`:``}var sn=[{category:`CRM`,label:`CRM`,description:`EspoCRM connection settings used by the API, worker, and Discord bot.`},{category:`Projects`,label:`Projects`,description:`ERPNext credentials and project workflow settings.`},{category:`Onboarding`,label:`Onboarding`,description:`Editable onboarding integrations such as DocuSeal, Outline, and onboarding email SMTP.`},{category:`Newsletter`,label:`Newsletter`,description:`Brevo, Keila, and recurring 508 members audience sync settings.`},{category:`AI`,label:`AI Providers`,description:`Provider credentials, base URLs, and model defaults.`},{category:`Agent`,label:`Agent Runtime`,description:`Planner, fallback, and tiered model routing for agent workflows.`},{category:`Observability`,label:`Observability`,description:`Telemetry and request tracing integrations.`},{category:`Intake`,label:`Intake`,description:`Resume and mailbox intake limits and parser defaults.`},{category:`Operations`,label:`Operations`,description:`Queue, sync, GitHub, and notification behavior.`},{category:`Legacy`,label:`Legacy`,description:`Older integrations retained for compatibility.`}],cn=new Map(sn.map((e,t)=>[e.category,{...e,index:t}]));function ln(e){return`configurationGroup-${e.replace(/[^a-zA-Z0-9_-]+/g,`-`)}`}function un(e){return e.key.startsWith(`ONBOARDING_EMAIL_`)||e.is_secret||e.value_type===`url`||e.key.endsWith(`_MODEL`)||e.key.endsWith(`_API_USER`)||e.key.endsWith(`_BASE_URL`)}var dn={people:`/dashboard/people`,gigs:`/dashboard/gigs`,projects:`/dashboard/projects`,onboarding:`/dashboard/onboarding`,jobs:`/dashboard/jobs`,agent:`/dashboard/agent`,audit:`/dashboard/audit`,configuration:`/dashboard/configuration`},fn={people:`people:read`,gigs:`gigs:read`,projects:`projects:read`,onboarding:`onboarding:read`,jobs:`jobs:read`,agent:`audit:read`,audit:`audit:read`,configuration:`configuration:read`},pn={discord:{label:`Discord`,options:[[`linked`,`Linked`],[`missing`,`Missing`]]},email_508:{label:`508 email`,options:[[`present`,`Present`],[`missing`,`Missing`]]},resume:{label:`Resume`,options:[[`present`,`Present`],[`missing`,`Missing`]]},skills:{label:`Skills`,options:[[`present`,`Parsed`],[`missing`,`Not parsed`]]},sync_status:{label:`Sync status`,options:[[`active`,`Active`],[`conflict`,`Conflict`],[`missing_in_crm`,`Missing in CRM`]]}},mn=[[`pending`,`Needs review`],[`selected`,`Assigned to onboarder`],[`reachingout`,`Reaching out`],[`awaitingcontribution`,`Awaiting contribution`],[`onboarded`,`Onboarded`],[`waitlist`,`Waitlist`],[`rejected`,`Rejected`]],hn=mn.slice(0,4),gn=new Set([`onboarded`,`waitlist`,`rejected`]);function _n(e){return String(e||``).trim().toLowerCase().replace(/[-_\s]+/g,``)}var vn=class extends Error{status;statusText;payload;url;method;constructor(e,t,n,r,i,a){super(e),this.name=`ApiRequestError`,this.status=t,this.statusText=n,this.payload=r,this.url=i,this.method=a}};function yn(e,t){let n=e.detail;if(typeof n==`string`&&n.trim())return n;let r=e.error;return typeof r==`string`?r===`person_not_found`?`No CRM person, ERPNext user, or ERPNext supplier matched "${typeof e.person==`string`&&e.person.trim()?e.person:`that person`}". Try an email address or an exact name from CRM/ERPNext.`:r===`candidate_not_found`?`The selected person record is no longer available. Search again and choose one of the current matches.`:r===`invalid_crm_profile`?`Paste a valid CRM Contact profile URL or Contact id.`:r===`crm_profile_not_found`?`That CRM Contact profile was not found.`:r===`crm_profile_mismatch`?`CRM returned a different Contact than the profile requested. Check the profile URL and try again.`:r===`crm_profile_lookup_failed`?`CRM profile lookup failed. Try again after CRM is reachable.`:r===`ambiguous_person`?`Multiple people matched. Choose the matching person record.`:r||t:t}function bn(e,t){return typeof e==`string`&&e.trim()?e:e instanceof Error&&e.message.trim()?e.message:t}function xn(){return window.location.pathname.split(`/`).filter(Boolean)[1]||``}function Sn(){let e=xn();return Object.hasOwn(dn,e)?e:`people`}function Cn(e=`gigs`){let[,t,n]=window.location.pathname.split(`/`).filter(Boolean);if(t!==e||!n)return``;try{return decodeURIComponent(n)}catch{return``}}async function q(e,t={}){let n=String(t.method||`GET`).toUpperCase(),r=new Headers(t.headers);r.set(`Accept`,`application/json`);let i;try{i=await fetch(e,{credentials:`same-origin`,...t,headers:r})}catch(t){throw new vn(bn(t,`Network request failed`),0,`Network request failed`,null,e,n)}if(i.status===401){let t=`${window.location.pathname}${window.location.search}`||`/dashboard`;throw window.location.assign(`/auth/login?next=${encodeURIComponent(t)}`),new vn(`Session expired`,i.status,i.statusText,null,e,n)}if(!i.ok){let t=i.statusText,r=null;try{r=await i.json(),r&&typeof r==`object`&&(t=yn(r,String(t||`Request failed`)))}catch{t=i.statusText}throw new vn(typeof t==`string`?t:JSON.stringify(t),i.status,i.statusText,r,e,n)}return i.json()}function wn(e,t,n){if(e===`gigs`){let e=t;if(n===`title`)return e.title||``;if(n===`status`)return e.status||``;if(n===`applications`)return Number(e.application_count||0);if(n===`activity`)return zn(e)}if(e===`projects`){let e=t;if(n===`display_name`)return e.display_name||``;if(n===`customer`)return e.customer||``;if(n===`status`)return e.source_status||``;if(n===`roster_count`)return Number(e.roster_count||0);if(n===`modified`)return e.source_modified_at||e.last_synced_at||``}if(e===`onboarding`){let e=t,r=e.profile_status||{};if(n===`name`)return e.name||e.email_508||e.email||``;if(n===`onboarding_state`){let t=Xt(e);return t.toLowerCase()===`pending`?`zzz-${t}`:t}if(n===`onboarder`)return e.onboarder||``;if(n===`updated`)return e.onboarding_updated_at||``;if(n===`profile_gaps`)return[!r.discord_linked,!r.latest_resume,Number(r.skills_count||0)<=0].filter(Boolean).length}if(e===`people`){let e=t,r=e.profile_status||{};if(n===`name`)return e.name||e.email_508||e.email||``;if(n===`status`)return[r.crm_active,r.is_member,r.discord_linked,r.email_508,r.latest_resume].filter(Boolean).length;if(n===`discord`)return e.discord_username||e.discord_user_id||``;if(n===`resume`)return e.latest_resume_name||e.latest_resume_id||``}if(e===`audit`){let e=t;if(n===`actor`)return e.actor_display_name||e.actor_subject||e.actor_provider||``}return t[n]??``}function Tn(e,t,n){let r=n.direction===`asc`?1:-1;return[...t].sort((t,i)=>{let a=wn(e,t,n.key),o=wn(e,i,n.key);return typeof a==`number`&&typeof o==`number`?(a-o)*r:String(a).localeCompare(String(o),void 0,{numeric:!0})*r})}function En({label:e,scope:t,sort:n,sortKey:r,onSort:i}){let a=n.key===r,o=n.direction===`asc`?`↑`:`↓`;return(0,R.jsx)(`button`,{type:`button`,"data-sort-scope":t,"data-sort-key":r,className:`text-left font-[inherit] text-inherit hover:text-foreground`,onClick:()=>i(t,r),children:a?`${e} ${o}`:e})}function Dn({className:e,label:t,scope:n,sort:r,sortKey:i,onSort:a}){return(0,R.jsx)(G,{className:e,"aria-sort":r.key===i?r.direction===`asc`?`ascending`:`descending`:`none`,children:(0,R.jsx)(En,{label:t,scope:n,sort:r,sortKey:i,onSort:a})})}function On({label:e,value:t,id:n}){return(0,R.jsxs)(V,{className:`p-4`,children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:e}),(0,R.jsx)(`strong`,{id:n,className:`block text-2xl`,children:t})]})}function kn({children:e,hidden:t}){return t?null:(0,R.jsx)(`div`,{className:`px-4 py-7 text-center text-sm text-muted-foreground`,children:e})}function An({value:e,query:t}){let n=t.trim().toLowerCase();if(!n)return(0,R.jsx)(R.Fragment,{children:e});let r=e.toLowerCase(),i=[],a=0,o=r.indexOf(n);for(;o>=0;){o>a&&i.push(e.slice(a,o));let t=o+n.length;i.push((0,R.jsx)(`mark`,{className:`rounded-sm bg-amber-200 px-0.5 text-inherit dark:bg-amber-500/35`,children:e.slice(o,t)},`${o}-${t}`)),a=t,o=r.indexOf(n,a)}return avoid 0);function bt(e){return s.includes(e)}function N(e){return s.includes(`${e}:dry_run`)}function xt(e){return bt(e)||N(e)}function St(e){return bt(fn[e])}function Ct(){return Object.keys(dn).find(e=>St(e))||`people`}function P(e,t){o({message:e,tone:t})}function F(e,t){P(bn(e,t),`error`)}function I(e,t){Ee(n=>({...n,[e]:t}))}function wt(e,t=!1){let n=e;St(n)||(P(`${n[0].toUpperCase()}${n.slice(1)} requires SSO validation`,`error`),n=Ct()),n!==`gigs`&&le(``),n!==`projects`&&ue(``),n===`gigs`&&t&&le(``),n===`projects`&&t&&ue(``),i(n),t?window.history.pushState({view:n},``,dn[n]):(!Object.hasOwn(dn,xn())||n!==e)&&window.history.replaceState({view:n},``,dn[n])}yt.current=wt;function Tt(e){return!u||!e?``:`${u}/#Contact/view/${encodeURIComponent(e)}`}function Et(e){return!u||!e?``:`${u}/api/v1/Attachment/file/${encodeURIComponent(e)}`}function Dt(e,t){Me(n=>{let r=n[e];return{...n,[e]:{key:t,direction:r.key===t&&r.direction===`asc`?`desc`:`asc`}}})}function Ot(e){le(e),w(h.find(t=>t.id===e)||null),i(`gigs`),window.history.pushState({view:`gigs`,gigId:e},``,`/dashboard/gigs/${encodeURIComponent(e)}`)}function kt(){le(``),w(null),window.history.replaceState({view:`gigs`},``,dn.gigs)}function At(e){ue(e),i(`projects`),window.history.pushState({view:`projects`,projectId:e},``,`/dashboard/projects/${encodeURIComponent(e)}`)}function jt(){ue(``),window.history.replaceState({view:`projects`},``,dn.projects)}async function Mt(){let e=await q(`/dashboard/api/me`);n(e);let t=Array.isArray(e.permissions)?e.permissions:[];return c(t),d((e.crm_base_url||``).replace(/\/+$/,``)),t}function Nt(){let e=new URLSearchParams({minutes:Ne,limit:`100`});return Fe&&e.set(`status`,Fe),Le.trim()&&e.set(`type`,Le.trim()),`/dashboard/api/jobs?${e.toString()}`}function Pt(){let e=new URLSearchParams({limit:String(Ge)});return ze&&e.set(`status`,ze),Ve.trim()&&e.set(`query`,Ve.trim()),Ue&&e.set(`include_historical`,`true`),`/dashboard/api/gigs?${e.toString()}`}function Ft(){let e=new URLSearchParams({limit:`100`,status:Ye});return qe.trim()&&e.set(`query`,qe.trim()),`/dashboard/api/projects?${e.toString()}`}async function It(){I(`jobs`,!0),P(`Loading background tasks`);try{let e=await q(Nt());p(e),P(`Loaded ${e.length} background task${e.length===1?``:`s`}`,`ok`)}catch(e){F(e,`Unable to load background tasks`)}finally{I(`jobs`,!1)}}async function Lt(){I(`gigs`,!0);try{let e=await q(Pt());y(e),P(`Loaded ${e.length} gig${e.length===1?``:`s`}`,`ok`),Jt()}catch(e){F(e,`Unable to load gigs`)}finally{I(`gigs`,!1)}}async function z(){I(`projects`,!0);try{let e=await q(Ft());S(e.projects||[]),ne(e.summary||{}),P(`Loaded ${(e.projects||[]).length} project${(e.projects||[]).length===1?``:`s`}`,`ok`)}catch(e){F(e,`Unable to load projects`)}finally{I(`projects`,!1)}}async function Rt(){I(`syncProjects`,!0),P(`Queueing project sync`);try{let e=await q(`/dashboard/api/sync/projects`,{method:`POST`});e.dry_run?P(`Dry run only: would queue ${e.would_enqueue?.job_type||`project sync`}`,`warning`):P(`Queued project sync ${e.job_id}`,`ok`)}catch(e){F(e,`Unable to queue project sync`)}finally{I(`syncProjects`,!1)}}async function V(e){let t=e.trim();if(t.length<2)return[];try{return(await q(`/dashboard/api/erpnext/customers?${new URLSearchParams({query:t}).toString()}`)).customers||[]}catch(e){return P(e instanceof Error?e.message:`Unable to search customers`,`error`),[]}}async function H(e){let t=e.trim();if(t.length<2)return[];try{return(await q(`/dashboard/api/erpnext/contacts?${new URLSearchParams({query:t}).toString()}`)).contacts||[]}catch(e){return P(e instanceof Error?e.message:`Unable to search contacts`,`error`),[]}}async function zt(e){let t=e.trim();if(t.length<2)return[];try{return(await q(`/dashboard/api/erpnext/account-managers?${new URLSearchParams({query:t}).toString()}`)).users||[]}catch(e){return P(e instanceof Error?e.message:`Unable to search account managers`,`error`),[]}}async function Bt(){try{let e=(await q(`/dashboard/api/erpnext/cost-centers`)).cost_centers||[];return e.length?e:[{name:`Projects - 5`,cost_center_name:`Projects`}]}catch(e){return P(e instanceof Error?e.message:`Unable to load cost centers`,`error`),[{name:`Projects - 5`,cost_center_name:`Projects`}]}}async function U(e){I(`createProject`,!0);try{let t=await q(`/dashboard/api/projects/create`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify(e)});return t.project.id?(S(e=>e.some(e=>e.id===t.project.id)?e.map(e=>e.id===t.project.id?t.project:e):[t.project,...e]),P(t.setup_warnings?.length?t.setup_warning_message||`Created ERP project setup; account manager setup needs follow-up`:`Created ERP project setup`,t.setup_warnings?.length?`warning`:`ok`),At(t.project.id)):(P([t.cache_refresh_message||`Created ERP project in ERPNext; local sync is pending`,t.setup_warnings?.length?t.setup_warning_message||`Account manager setup needs follow-up`:``].filter(Boolean).join(` `),t.setup_warnings?.length?`warning`:`ok`),z()),!0}catch(e){return P(e instanceof Error?e.message:`Unable to create project`,`error`),!1}finally{I(`createProject`,!1)}}async function W(e,t){I(`project:${e}:status`,!0);try{let n=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/status`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:t})});S(t=>t.map(t=>t.id===e?n.project:t)),P(`Updated project status`,`ok`)}catch(e){F(e,`Unable to update project`)}finally{I(`project:${e}:status`,!1)}}async function Vt(e,t){if(e.length===0)return!1;I(`projectsBulkUpdate`,!0);try{let n=await q(`/dashboard/api/projects/bulk`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({project_ids:e,...t})}),r=n.projects||[];S(e=>e.map(e=>r.find(t=>t.id===e.id)||e));let i=n.failures||[];return P(i.length?`Updated ${r.length}; ${i.length} failed`:`Updated ${r.length} project${r.length===1?``:`s`}`,i.length?`error`:`ok`),i.length===0}catch(e){return F(e,`Unable to bulk update projects`),!1}finally{I(`projectsBulkUpdate`,!1)}}async function Ht(e,t,n,r){let i=t.trim(),a=n.trim();if(!i||!a)return!1;I(`project:${e}:user`,!0);try{let t=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/users`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({user:i,candidate_id:a,...r||{}})});return S(n=>n.map(n=>n.id===e?t.project:n)),P(t.activity_cost_error?`Added project user; rate failed`:t.activity_cost?`Added project user and rate`:`Added project user`,t.activity_cost_error?`error`:`ok`),!0}catch(e){return F(e,`Unable to add project user`),!1}finally{I(`project:${e}:user`,!1)}}async function Ut(e,t){let n=t.trim();if(!n)return!1;I(`project:${e}:user`,!0);try{let t=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/users/remove`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({user:n})});return S(n=>n.map(n=>n.id===e?t.project:n)),P(`Removed project user`,`ok`),!0}catch(e){return P(e instanceof Error?e.message:`Unable to remove project user`,`error`),!1}finally{I(`project:${e}:user`,!1)}}async function Wt(e,t,n){let r=t.trim();if(!r)return!1;I(`project:${e}:historical`,!0);try{let t=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/historical-members`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({person:r,candidate_id:n})});return S(n=>n.map(n=>n.id===e?t.project:n)),Ae(null),P(`Added historical project member`,`ok`),!0}catch(t){if(t instanceof vn&&t.status===409){let n=t.payload?.candidates||[];if(n.length>0)return Ae({projectId:e,person:r,candidates:n}),P(`Choose the matching person record`,`error`),!1}return F(t,`Unable to add historical member`),!1}finally{I(`project:${e}:historical`,!1)}}async function Gt(e,t){let n=t.trim();if(!n)return!1;I(`project:${e}:historical`,!0);try{let t=await q(`/dashboard/api/projects/${encodeURIComponent(e)}/historical-members/remove`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({source_user_id:n})});return S(n=>n.map(n=>n.id===e?t.project:n)),P(`Removed historical project member`,`ok`),!0}catch(e){return P(e instanceof Error?e.message:`Unable to remove historical member`,`error`),!1}finally{I(`project:${e}:historical`,!1)}}async function G(e,t,n){I(`project:${e}:wiki`,!0);try{await q(`/dashboard/api/projects/${encodeURIComponent(e)}/wiki-match`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:t,row_key:n})}),P(t===`no_row`?`Marked as no wiki row`:`Confirmed wiki match`,`ok`),await K()}catch(e){F(e,`Unable to save wiki match`)}finally{I(`project:${e}:wiki`,!1)}}async function K(){I(`wikiMatches`,!0);try{oe(await q(`/dashboard/api/projects/wiki-matches`)),P(`Loaded wiki match preview`,`ok`)}catch(e){F(e,`Unable to load wiki matches`)}finally{I(`wikiMatches`,!1)}}async function Kt(e){I(`gig:${e}:detail`,!0);try{w(await q(`/dashboard/api/gigs/${encodeURIComponent(e)}`))}catch(e){w(null),F(e,`Unable to load gig`)}finally{I(`gig:${e}:detail`,!1)}}async function qt(){await Lt(),T&&await Kt(T)}async function Jt(){if(bt(`gigs:read`)){I(`notifications`,!0);try{let e=await q(`/dashboard/api/notifications?limit=20`);Qe(e.stale_days||7),fe(e.notifications||[])}catch(e){F(e,`Unable to load notifications`)}finally{I(`notifications`,!1)}}}async function Yt(e,t){I(`gig:${e}:status`,!0);try{let n=(await q(`/dashboard/api/gigs/${encodeURIComponent(e)}/status`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:t})})).discord_title_sync?.status;P(n===`error`?`Updated gig status; Discord title sync failed`:`Updated gig status`,n===`error`?`error`:`ok`),await Lt(),T===e&&await Kt(e)}catch(e){F(e,`Unable to update gig`)}finally{I(`gig:${e}:status`,!1)}}async function Xt(e,t,n){I(`application:${t}:status`,!0);try{await q(`/dashboard/api/gigs/${encodeURIComponent(e)}/applications/${encodeURIComponent(t)}/status`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:n})}),P(`Updated candidate status`,`ok`),await Lt(),T===e&&await Kt(e)}catch(e){F(e,`Unable to update candidate`)}finally{I(`application:${t}:status`,!1)}}async function Qt(e,t){let n=t.trim();if(!n)return P(`Paste a CRM Contact profile first`,`warning`),!1;I(`gig:${e}:addCandidate`,!0);try{return await q(`/dashboard/api/gigs/${encodeURIComponent(e)}/applications`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({crm_profile:n})}),P(`Added candidate`,`ok`),await Lt(),T===e&&await Kt(e),!0}catch(e){return F(e,`Unable to add candidate`),!1}finally{I(`gig:${e}:addCandidate`,!1)}}function $t(){let e=new URLSearchParams({limit:`25`});$e.trim()&&e.set(`query`,$e.trim()),tt&&e.set(`is_member`,tt);for(let[t,n]of Object.entries(j))n&&e.set(t,n);return`/dashboard/api/people?${e.toString()}`}async function en(){I(`people`,!0);try{k(await q($t()))}catch(e){F(e,`Unable to load people`)}finally{I(`people`,!1)}}function tn(){let e=new URLSearchParams({limit:`25`});ct.trim()&&e.set(`query`,ct.trim()),ut&&e.set(`onboarding_state`,ut),ft.trim()&&e.set(`onboarder`,ft.trim());for(let[t,n]of Object.entries(M))n&&e.set(t,n);return`/dashboard/api/onboarding?${e.toString()}`}async function nn(){I(`onboarding`,!0);try{A(await q(tn()))}catch(e){F(e,`Unable to load onboarding`)}finally{I(`onboarding`,!1)}}async function rn(e,t){if(!e)return P(`Missing CRM contact`,`error`),null;let n=`onboarding-email-draft:${e}`;I(n,!0);try{let n=await q(`/dashboard/api/onboarding/${encodeURIComponent(e)}/email/draft`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify(t)});return P(`Drafted onboarding email`,`ok`),n}catch(e){return F(e,`Unable to draft onboarding email`),null}finally{I(n,!1)}}async function an(e,t,n){if(!e)return P(`Missing CRM contact`,`error`),null;let r=`onboarding-email-send:${e}`;I(r,!0);try{let r=await q(`/dashboard/api/onboarding/${encodeURIComponent(e)}/email/send`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({...t,markdown_body:n})});return A(t=>t.map(t=>t.crm_contact_id===e?{...t,onboarding_email_sent_at:r.onboarding_email_sent_at||t.onboarding_email_sent_at,onboarding_email_sent_by:r.onboarding_email_sent_by||t.onboarding_email_sent_by,onboarding_email_recipient:r.onboarding_email_recipient||r.recipient_email||t.onboarding_email_recipient}:t)),P(`Sent onboarding email`,`ok`),r}catch(e){return F(e,`Unable to send onboarding email`),null}finally{I(r,!1)}}async function on(){I(`audit`,!0);try{ge(await q(`/dashboard/api/audit-events?limit=25`))}catch(e){F(e,`Unable to load audit events`)}finally{I(`audit`,!1)}}async function sn(){I(`agent`,!0);try{ve(await q(`/dashboard/api/agent?limit=100`))}catch(e){F(e,`Unable to load agent report`)}finally{I(`agent`,!1)}}async function cn(){I(`configuration`,!0);try{be((await q(`/dashboard/api/configuration`)).items)}catch(e){F(e,`Unable to load configuration`)}finally{I(`configuration`,!1)}}async function ln(e,t){I(`configuration:${e}`,!0);try{be((await q(`/dashboard/api/configuration/${encodeURIComponent(e)}`,{method:`PUT`,headers:{"Content-Type":`application/json`},body:JSON.stringify({value:t})})).items),P(`Saved ${e}`,`ok`)}catch(t){F(t,`Unable to save ${e}`)}finally{I(`configuration:${e}`,!1)}}async function un(e){I(`configuration:${e}`,!0);try{be((await q(`/dashboard/api/configuration/${encodeURIComponent(e)}`,{method:`PUT`,headers:{"Content-Type":`application/json`},body:JSON.stringify({clear:!0})})).items),P(`Cleared ${e}`,`ok`)}catch(t){F(t,`Unable to clear ${e}`)}finally{I(`configuration:${e}`,!1)}}async function mn(e){I(`detail:${e}`,!0),P(`Loading ${e}`);try{we(await q(`/dashboard/api/jobs/${encodeURIComponent(e)}`)),P(`Loaded ${e}`,`ok`)}catch(e){F(e,`Unable to load task detail`)}finally{I(`detail:${e}`,!1)}}async function hn(e){I(`rerun:${e}`,!0),P(`Rerunning ${e}`);try{let t=await q(`/dashboard/api/jobs/${encodeURIComponent(e)}/rerun`,{method:`POST`});t.dry_run?P(`Dry run only: would rerun ${t.would_enqueue?.job_type||e}`,`warning`):(P(`Queued rerun ${t.job_id}`,`ok`),await It())}catch(e){F(e,`Unable to rerun task`)}finally{I(`rerun:${e}`,!1)}}async function yn(){I(`syncPeople`,!0),P(`Queueing people sync`);try{let e=await q(`/dashboard/api/sync/people`,{method:`POST`});e.dry_run?P(`Dry run only: would queue ${e.would_enqueue?.job_type||`people sync`}`,`warning`):P(`Queued people sync ${e.job_id}`,`ok`)}catch(e){F(e,`Unable to queue people sync`)}finally{I(`syncPeople`,!1)}}async function wn(){I(`syncNewsletters`,!0),P(`Queueing newsletter sync`);try{let e=await q(`/dashboard/api/sync/newsletters`,{method:`POST`});e.dry_run?P(`Dry run only: would queue ${e.would_enqueue?.job_type||`newsletter sync`}`,`warning`):P(`Queued newsletter sync ${e.job_id}`,`ok`)}catch(e){F(e,`Unable to queue newsletter sync`)}finally{I(`syncNewsletters`,!1)}}async function En(e,t){let n=String(e||``).trim(),r=t.trim();if(!n){P(`Missing CRM contact id`,`error`);return}if(!r){P(`Enter a 508 username`,`error`);return}I(`onboarder:${n}`,!0),P(`Assigning ${r}`);try{let e=await q(`/dashboard/api/onboarding/${encodeURIComponent(n)}/onboarder`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({onboarder:r})});A(t=>t.map(t=>t.crm_contact_id===e.contact_id?{...t,onboarder:e.onboarder,onboarding_state:e.state_updated&&e.onboarding_state?e.onboarding_state:t.onboarding_state,onboarding_status_label:e.onboarding_status_label||(e.state_updated?void 0:t.onboarding_status_label)}:t)),P(`Assigned ${e.onboarder}`,`ok`)}catch(e){F(e,`Unable to assign onboarder`)}finally{I(`onboarder:${n}`,!1)}}async function Dn(e,t){let n=String(e||``).trim(),r=t.trim();if(!n){P(`Missing CRM contact id`,`error`);return}if(!r){P(`Choose an onboarding status`,`error`);return}I(`onboarding-status:${n}`,!0),P(`Updating onboarding status`);try{let e=await q(`/dashboard/api/onboarding/${encodeURIComponent(n)}/status`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({status:r})}),t=_n(e.onboarding_state),i=e.onboarding_status_label||Zt(t);A(n=>n.map(n=>n.crm_contact_id===e.contact_id?{...n,onboarding_state:t,onboarding_status_label:i}:n).filter(n=>n.crm_contact_id!==e.contact_id||!gn.has(t))),P(`Status set to ${i}`,`ok`)}catch(e){F(e,`Unable to update onboarding status`)}finally{I(`onboarding-status:${n}`,!1)}}async function On(e){let t=e.email.trim().toLowerCase(),n=e.first_name.trim();if(!t?.endsWith(`@508.dev`))return P(`Enter the engineer's @508.dev email`,`error`),null;if(!n)return P(`Enter the engineer name`,`error`),null;I(`engineerSetup`,!0);try{let r=await q(`/dashboard/api/onboarding/engineers`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({...e,email:t,first_name:n})});return P(`Set up ${r.employee_name||r.user||t}`,`ok`),r}catch(e){if(e instanceof vn&&e.status===409){let t=e.payload&&typeof e.payload==`object`?e.payload:null,n=(Array.isArray(t?.matches)?t.matches:[]).map(e=>e?.label||e?.email).filter(Boolean).slice(0,2).join(`, `);P(n?`Similar account exists: ${n}`:`Similar account exists; confirm before creating`,`error`)}else P(e instanceof Error?e.message:`Unable to set up engineer`,`error`);return null}finally{I(`engineerSetup`,!1)}}async function kn(){I(`logout`,!0);try{let e=await q(`/auth/logout`,{method:`POST`});window.location.assign(e.end_session_url||`/dashboard`)}catch(e){F(e,`Unable to log out`),I(`logout`,!1)}}(0,l.useEffect)(()=>{Mt().then(e=>{let t=Sn(),n=e.includes(fn[t])?t:Object.keys(dn).find(t=>e.includes(fn[t]))||`people`;le(n===`gigs`?Cn():``),ue(n===`projects`?Cn(`projects`):``),i(n),(!Object.hasOwn(dn,xn())||n!==t)&&window.history.replaceState({view:n},``,dn[n])}).catch(e=>{F(e,`Dashboard failed to load`)})},[]),(0,l.useEffect)(()=>{let e=()=>{le(Cn()),ue(Cn(`projects`)),yt.current(Sn(),!1)};return window.addEventListener(`popstate`,e),()=>window.removeEventListener(`popstate`,e)},[]),(0,l.useEffect)(()=>{if(!a.message)return;let e=window.setTimeout(()=>o({message:``}),4500);return()=>window.clearTimeout(e)},[a.message]),(0,l.useEffect)(()=>{},[]),(0,l.useEffect)(()=>{s.length!==0&&(bt(`gigs:read`)&&Jt(),r===`people`&&en(),r===`gigs`&&Lt(),r===`projects`&&z(),r===`onboarding`&&nn(),r===`jobs`&&It(),r===`agent`&&sn(),r===`audit`&&on(),r===`configuration`&&cn())},[r]),(0,l.useEffect)(()=>{s.length!==0&&(bt(`gigs:read`)&&Jt(),r===`people`&&en(),r===`gigs`&&Lt(),r===`projects`&&z(),r===`onboarding`&&nn(),r===`jobs`&&It(),r===`agent`&&sn(),r===`audit`&&on(),r===`configuration`&&cn())},[s]),(0,l.useEffect)(()=>{r===`jobs`&&s.length>0&&It()},[Ne,Fe]),(0,l.useEffect)(()=>{r===`gigs`&&s.length>0&&Lt()},[ze,Ue,Ge]),(0,l.useEffect)(()=>{r===`projects`&&s.length>0&&z()},[Ye]),(0,l.useEffect)(()=>{r===`gigs`&&T&&s.length>0&&Kt(T)},[r,T,s]),(0,l.useEffect)(()=>{r===`people`&&s.length>0&&en()},[tt]),(0,l.useEffect)(()=>{r===`people`&&s.length>0&&en()},[j]),(0,l.useEffect)(()=>{r===`onboarding`&&s.length>0&&nn()},[ut]),(0,l.useEffect)(()=>{r===`onboarding`&&s.length>0&&nn()},[M]);let An=(0,l.useMemo)(()=>Tn(`jobs`,f,je.jobs),[f,je.jobs]),jn=(0,l.useMemo)(()=>Tn(`people`,O,je.people),[O,je.people]),Fn=(0,l.useMemo)(()=>Tn(`onboarding`,me,je.onboarding),[me,je.onboarding]),In=(0,l.useMemo)(()=>Tn(`gigs`,h,je.gigs),[h,je.gigs]),Ln=(0,l.useMemo)(()=>Tn(`projects`,te,je.projects),[te,je.projects]),Rn=(0,l.useMemo)(()=>se?.id===T?se:In.find(e=>e.id===T)||null,[se,T,In]),zn=(0,l.useMemo)(()=>Ln.find(e=>e.id===E)||null,[E,Ln]),Bn=(0,l.useMemo)(()=>Tn(`audit`,he,je.audit),[he,je.audit]),Vn=(0,l.useMemo)(()=>f.reduce((e,t)=>(e[t.status]=(e[t.status]||0)+1,e),{}),[f]),Hn=Object.keys(pn).filter(e=>!j[e]),Un=Object.keys(pn).filter(e=>e!==`sync_status`&&e!==`email_508`&&!M[e]);function Wn(e){if(e.type===`stale_recruiting_gig`){let t=e.engagement_id||(e.id.startsWith(`stale-recruiting:`)?e.id.slice(17):``);t?Ot(t):(Be(`recruiting`),wt(`gigs`,!0))}D(!1)}(0,l.useEffect)(()=>{!Hn.includes(it)&&Hn[0]&&at(Hn[0])},[Hn,it]),(0,l.useEffect)(()=>{let e=pn[it]?.options;e?.[0]&&!e.some(([e])=>e===ot)&&st(e[0][0])},[it,ot]),(0,l.useEffect)(()=>{!Un.includes(ht)&&Un[0]&>(Un[0])},[Un,ht]),(0,l.useEffect)(()=>{let e=pn[ht]?.options;e?.[0]&&!e.some(([e])=>e===_t)&&vt(e[0][0])},[ht,_t]);let Kn=[t?.email,t?.crm_contact_id?`CRM ${t.crm_contact_id}`:``,t?.actor_provider].filter(Boolean).join(` | `);return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsx)(`header`,{className:`sticky top-0 z-20 border-b bg-background/90 backdrop-blur`,children:(0,R.jsxs)(`div`,{className:`mx-auto flex max-w-7xl flex-col gap-4 px-5 py-4 md:flex-row md:items-center md:justify-between`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`h1`,{className:`text-xl font-bold`,children:`508 Operations Dashboard`}),(0,R.jsx)(`p`,{className:`text-sm text-muted-foreground`,children:`Operations view for authenticated 508 operators.`})]}),(0,R.jsxs)(`div`,{className:`flex min-w-0 items-center gap-3`,children:[bt(`gigs:read`)?(0,R.jsx)(`div`,{className:`relative`,children:(0,R.jsxs)(B,{id:`notifications`,type:`button`,variant:`outline`,size:`icon`,"aria-label":`Notifications`,"aria-expanded":pe,onClick:()=>D(e=>!e),children:[(0,R.jsx)(g,{}),de.length>0?(0,R.jsx)(`span`,{className:`absolute -right-1 -top-1 grid min-h-5 min-w-5 place-items-center rounded-full bg-red-500 px-1 text-[11px] font-bold text-white`,children:de.length}):null]})}):null,(0,R.jsxs)(`div`,{className:`grid min-w-0 gap-0.5 text-right text-sm text-muted-foreground`,children:[(0,R.jsx)(`strong`,{id:`userName`,className:`truncate text-foreground`,children:t?.display_name||t?.email||t?.subject||`Loading user`}),(0,R.jsx)(`span`,{id:`userMeta`,className:`truncate`,children:Kn||`Checking session`})]}),(0,R.jsxs)(B,{id:`logout`,type:`button`,variant:`outline`,onClick:kn,disabled:Te.logout,children:[(0,R.jsx)(ee,{}),`Log out`]})]})]})}),(0,R.jsx)(Mn,{open:pe,notifications:de,loading:Te.notifications,onClose:()=>D(!1),onRefresh:Jt,onOpenNotification:Wn}),(0,R.jsx)(Pn,{toast:a}),null,(0,R.jsx)(Nn,{choice:ke,loading:!!(ke&&Te[`project:${ke.projectId}:historical`]),crmContactUrl:Tt,onClose:()=>Ae(null),onChoose:e=>{ke&&Wt(ke.projectId,ke.person,e)}}),(0,R.jsxs)(`main`,{className:`mx-auto grid max-w-7xl grid-cols-1 gap-5 px-5 py-5 md:grid-cols-[190px_minmax(0,1fr)]`,children:[(0,R.jsx)(`nav`,{className:`grid content-start gap-1 md:sticky md:top-24`,"aria-label":`Dashboard sections`,children:[[`people`,`People`,ce],[`gigs`,`Gigs`,_],[`projects`,`Projects`,x],[`onboarding`,`Onboarding`,v],[`jobs`,`Background tasks`,m],[`agent`,`Agent`,ae],[`audit`,`Audit`,b],[`configuration`,`Configuration`,ie]].filter(([e])=>St(e)).map(([e,t,n])=>(0,R.jsxs)(`a`,{className:L(`flex min-h-10 items-center gap-2 rounded-md border border-transparent px-3 text-sm font-extrabold text-muted-foreground hover:border-border hover:bg-secondary hover:text-foreground`,r===e&&`border-primary bg-accent text-accent-foreground`),"data-view-link":e,"data-permission":fn[e],href:dn[e],"aria-current":r===e?`page`:void 0,onClick:t=>{t.preventDefault(),wt(e,!0)},children:[(0,R.jsx)(n,{className:`size-4`}),t]},e))}),(0,R.jsxs)(`div`,{className:`grid min-w-0 gap-5`,children:[r===`people`?(0,R.jsx)($n,{crmBaseUrl:u,people:jn,sort:je.people,canSync:xt(`people:sync`),canSyncNewsletters:xt(`people:sync`),loading:Te,peopleQuery:$e,peopleMember:tt,peopleFilters:j,peopleFilterKind:it,peopleFilterValue:ot,peopleFilterKeys:Hn,onSearch:en,onSync:yn,onSyncNewsletters:wn,onSort:e=>Dt(`people`,e),setPeopleQuery:et,setPeopleMember:nt,setPeopleFilterKind:at,setPeopleFilterValue:st,addFilter:()=>{rt(e=>({...e,[it]:ot}))},removeFilter:e=>{rt(t=>{let n={...t};return delete n[e],n})},crmContactUrl:Tt,crmAttachmentUrl:Et}):null,r===`gigs`?(0,R.jsx)(Jn,{gigs:In,selectedGig:Rn,selectedGigId:T,sort:je.gigs,loading:Te,status:ze,query:Ve,includeHistorical:Ue,limit:Ge,staleDays:Ze,canWrite:bt(`gigs:write`),canIncludeHistorical:bt(`people:read`),crmContactUrl:Tt,crmAttachmentUrl:Et,setStatus:Be,setQuery:He,setIncludeHistorical:We,setLimit:Ke,onRefresh:qt,onSort:e=>Dt(`gigs`,e),onOpenGig:Ot,onCloseGig:kt,onUpdateStatus:Yt,onAddApplication:Qt,onUpdateApplicationStatus:Xt}):null,r===`projects`?(0,R.jsx)(Gn,{projects:Ln,selectedProject:zn,selectedProjectId:E,summary:C,wikiMatches:re,sort:je.projects,loading:Te,query:qe,status:Ye,canSync:xt(`projects:sync`),canWrite:bt(`projects:write`),crmContactUrl:Tt,setQuery:Je,setStatus:Xe,onSearch:z,onSync:Rt,onSearchCustomers:V,onSearchContacts:H,onSearchAccountManagers:zt,onLoadCostCenters:Bt,onCreateProject:U,onUpdateStatus:W,onBulkUpdate:Vt,onAddUser:Ht,onRemoveUser:Ut,onAddHistoricalMember:Wt,onRemoveHistoricalMember:Gt,onUpdateWikiMatch:G,onWikiMatches:K,onOpenProject:At,onCloseProject:jt,onSort:e=>Dt(`projects`,e)}):null,r===`onboarding`?(0,R.jsx)(er,{people:Fn,sort:je.onboarding,loading:Te,onboardingQuery:ct,onboardingState:ut,onboarderFilter:ft,onboardingFilters:M,onboardingFilterKind:ht,onboardingFilterValue:_t,onboardingFilterKeys:Un,onSearch:nn,onSort:e=>Dt(`onboarding`,e),onAssign:En,onStatusChange:Dn,onDraftEmail:rn,onSendEmail:an,onSetupEngineer:On,setOnboardingQuery:lt,setOnboardingState:dt,setOnboarderFilter:pt,setOnboardingFilterKind:gt,setOnboardingFilterValue:vt,addFilter:()=>{mt(e=>({...e,[ht]:_t}))},removeFilter:e=>{mt(t=>{let n={...t};return delete n[e],n})},crmContactUrl:Tt,crmAttachmentUrl:Et,canWrite:bt(`onboarding:write`),canConfigure:bt(`configuration:write`),onOpenConfiguration:()=>{Se({category:`Onboarding`,nonce:Date.now()}),wt(`configuration`,!0)}}):null,r===`jobs`?(0,R.jsx)(cr,{jobs:An,jobDetail:Ce,sort:je.jobs,loading:Te,minutes:Ne,status:Fe,jobType:Le,jobCounts:Vn,canWrite:xt(`jobs:write`),setMinutes:Pe,setStatus:Ie,setJobType:Re,onSearch:It,onSort:e=>Dt(`jobs`,e),onDetail:mn,onRerun:hn}):null,r===`audit`?(0,R.jsx)(lr,{events:Bn,sort:je.audit,loading:Te,onRefresh:on,onSort:e=>Dt(`audit`,e)}):null,r===`agent`?(0,R.jsx)(ur,{report:_e,loading:Te,onRefresh:sn}):null,r===`configuration`?(0,R.jsx)(dr,{items:ye,loading:Te,canWrite:bt(`configuration:write`),focusCategory:xe?.category,focusNonce:xe?.nonce,onRefresh:cn,onSave:ln,onClear:un}):null]})]})]})}function Mn({open:e,notifications:t,loading:n,onClose:r,onRefresh:i,onOpenNotification:a}){return e?(0,R.jsxs)(`div`,{className:`fixed inset-0 z-40`,"aria-labelledby":`notificationsTitle`,"aria-modal":`true`,role:`dialog`,children:[(0,R.jsx)(`button`,{type:`button`,className:`absolute inset-0 cursor-default bg-black/45`,"aria-label":`Close notifications`,onClick:r}),(0,R.jsxs)(`aside`,{className:`absolute right-0 top-0 grid h-full w-full max-w-md grid-rows-[auto_minmax(0,1fr)] border-l bg-background shadow-2xl`,children:[(0,R.jsxs)(`div`,{className:`flex items-center justify-between gap-3 border-b p-4`,children:[(0,R.jsxs)(`div`,{className:`grid gap-0.5`,children:[(0,R.jsx)(`strong`,{id:`notificationsTitle`,className:`text-base`,children:`Notifications`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:t.length===0?`No active notifications`:`${t.length} active`})]}),(0,R.jsxs)(`div`,{className:`flex items-center gap-2`,children:[(0,R.jsxs)(B,{type:`button`,variant:`outline`,size:`sm`,onClick:i,disabled:n,children:[(0,R.jsx)(C,{}),`Refresh`]}),(0,R.jsx)(B,{type:`button`,variant:`ghost`,size:`icon`,"aria-label":`Close`,onClick:r,children:(0,R.jsx)(w,{})})]})]}),(0,R.jsx)(`div`,{className:`min-h-0 overflow-auto p-4`,children:t.length===0?(0,R.jsx)(`div`,{className:`rounded-md border border-dashed p-6 text-sm text-muted-foreground`,children:`No active notifications.`}):(0,R.jsx)(`div`,{className:`grid gap-3`,children:t.map(e=>(0,R.jsxs)(`button`,{type:`button`,className:`grid gap-2 rounded-md border p-3 text-left hover:bg-secondary`,onClick:()=>a(e),children:[(0,R.jsx)(`span`,{className:`text-sm font-bold`,children:e.title}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.message})]},e.id))})})]})]}):null}function Nn({choice:e,loading:t,crmContactUrl:n,onClose:r,onChoose:i}){return e?(0,R.jsxs)(`div`,{className:`fixed inset-0 z-50 grid place-items-center p-4`,"aria-labelledby":`historicalPersonChoiceTitle`,"aria-modal":`true`,role:`dialog`,children:[(0,R.jsx)(`button`,{type:`button`,className:`absolute inset-0 cursor-default bg-black/45`,"aria-label":`Close person selection`,onClick:r}),(0,R.jsxs)(`div`,{className:`relative grid w-full max-w-2xl gap-4 rounded-md border bg-background p-5 shadow-2xl`,children:[(0,R.jsxs)(`div`,{className:`flex items-start justify-between gap-3`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`strong`,{id:`historicalPersonChoiceTitle`,className:`block text-base`,children:`Choose person record`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.person})]}),(0,R.jsx)(B,{type:`button`,variant:`ghost`,size:`icon`,"aria-label":`Close person selection`,onClick:r,children:(0,R.jsx)(w,{})})]}),(0,R.jsx)(`div`,{className:`grid gap-2`,children:e.candidates.map(e=>(0,R.jsxs)(`div`,{className:`grid gap-3 rounded-md border p-3 md:grid-cols-[minmax(0,1fr)_auto] md:items-center`,children:[(0,R.jsxs)(`div`,{className:`min-w-0`,children:[(0,R.jsx)(`strong`,{className:`block truncate`,children:e.label||e.full_name||e.email||`Person`}),(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-x-3 gap-y-1 text-sm text-muted-foreground`,children:[e.email?(0,R.jsx)(`span`,{children:e.email}):null,e.sources?.length?(0,R.jsx)(`span`,{children:e.sources.join(`, `)}):null,e.erpnext_user_id?(0,R.jsxs)(`span`,{children:[`ERP `,e.erpnext_user_id]}):null,e.supplier_erpnext_id?(0,R.jsxs)(`span`,{children:[`Supplier `,e.supplier_erpnext_id]}):null,e.crm_contact_id&&n(e.crm_contact_id)?(0,R.jsx)(`a`,{className:`font-semibold text-primary underline-offset-4 hover:underline`,href:n(e.crm_contact_id),target:`_blank`,rel:`noreferrer`,children:`CRM`}):null]})]}),(0,R.jsx)(B,{type:`button`,disabled:t,onClick:()=>i(e.candidate_id),children:`Select`})]},e.candidate_id))})]})]}):null}function Pn({toast:e}){return e.message?(0,R.jsx)(`div`,{id:`toast`,role:`status`,className:L(`fixed bottom-5 right-5 z-50 max-w-sm rounded-md border bg-background px-4 py-3 text-sm font-semibold shadow-lg`,e.tone===`ok`&&`border-emerald-500/40 text-emerald-300`,e.tone===`warning`&&`border-amber-500/40 text-amber-200`,e.tone===`error`&&`border-red-500/40 text-red-300`),children:e.message}):null}function Fn({filters:e,onRemove:t,suffix:n=`filter`}){return(0,R.jsx)(`fieldset`,{className:`m-0 flex min-h-7 flex-wrap gap-2 border-0 p-0`,"aria-label":`Active filters`,children:Object.entries(e).map(([e,r])=>{let i=pn[e],a=i.options.find(([e])=>e===r),o=`${i.label}: ${a?a[1]:r}`;return(0,R.jsxs)(B,{type:`button`,variant:`outline`,size:`sm`,className:`rounded-full`,"aria-label":`Remove ${o} ${n}`,onClick:()=>t(e),children:[o,` x`]},e)})})}var In=[`recruiting`,`filled`,`unknown`,`lost`,`outdated`],Ln=[`suggested`,`interested`,`reviewing`,`contacted`,`accepted`,`unavailable`,`rejected`,`withdrawn`];function Rn(e){return String(e||``).replace(/[-_]+/g,` `).replace(/\s+/g,` `).trim().replace(/\b\w/g,e=>e.toUpperCase())}function zn(e){let t=[e.last_activity_at,e.last_status_changed_at,e.posted_at,e.created_at].map(e=>e?new Date(e).getTime():NaN).filter(e=>!Number.isNaN(e));return t.length>0?new Date(Math.max(...t)).toISOString():``}function Bn(e,t){if(e.status!==`recruiting`)return null;let n=Jt(zn(e));return n===null||ne.projects.map(e=>e.id),[e.projects]),m=(0,l.useMemo)(()=>new Set(p),[p]),g=n.filter(e=>m.has(e)),_=e.projects.length>0&&g.length===e.projects.length;(0,l.useEffect)(()=>{r(e=>e.filter(e=>m.has(e)))},[m]);function v(e,t){r(n=>t?Array.from(new Set([...n,e])):n.filter(t=>t!==e))}async function b(){let t={};i&&(t.status=i),o&&(t.project_type=o),await e.onBulkUpdate(g,t)&&(r([]),a(``),s(``),u(!1))}let x=(0,R.jsxs)(V,{className:`grid gap-3 p-4 md:grid-cols-[minmax(0,1fr)_180px_auto_auto_auto] md:items-end`,children:[(0,R.jsxs)(W,{children:[`Search projects`,(0,R.jsx)(U,{id:`projectQuery`,value:e.query,autoComplete:`off`,placeholder:`Project, customer, ERP id`,onChange:t=>e.setQuery(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onSearch()})]}),(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{id:`projectStatus`,value:e.status,onChange:t=>e.setStatus(t.target.value),children:[(0,R.jsx)(`option`,{value:`Open`,children:`Open`}),(0,R.jsx)(`option`,{value:``,children:`Any status`})]})]}),(0,R.jsxs)(B,{id:`refreshProjects`,type:`button`,onClick:e.onSearch,disabled:e.loading.projects,children:[(0,R.jsx)(C,{}),`Refresh`]}),e.canSync?(0,R.jsxs)(B,{id:`syncProjects`,type:`button`,variant:`outline`,onClick:e.onSync,disabled:e.loading.syncProjects,children:[(0,R.jsx)(C,{}),`Sync ERP`]}):null,(0,R.jsxs)(B,{id:`wikiProjectMatches`,type:`button`,variant:`outline`,onClick:e.onWikiMatches,disabled:e.loading.wikiMatches,children:[(0,R.jsx)(ne,{}),`Wiki match`]})]});return e.selectedProjectId&&!e.selectedProject&&e.loading.projects?(0,R.jsxs)(R.Fragment,{children:[x,(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Project detail`})}),(0,R.jsx)(Bt,{className:`text-sm text-muted-foreground`,children:`Loading project.`})]})]}):e.selectedProjectId&&!e.selectedProject?(0,R.jsxs)(R.Fragment,{children:[x,(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Project detail`})}),(0,R.jsxs)(Bt,{className:`grid gap-3`,children:[(0,R.jsx)(`p`,{className:`text-sm text-muted-foreground`,children:`This project is not in the current result set. Clear filters or refresh the project list.`}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,onClick:e.onCloseProject,children:[(0,R.jsx)(h,{}),`Back to projects`]})]})]})]}):e.selectedProject?(0,R.jsxs)(R.Fragment,{children:[x,(0,R.jsx)(qn,{project:e.selectedProject,loading:e.loading,canWrite:e.canWrite,crmContactUrl:e.crmContactUrl,onBack:e.onCloseProject,onUpdateStatus:e.onUpdateStatus,onAddUser:e.onAddUser,onRemoveUser:e.onRemoveUser,onAddHistoricalMember:e.onAddHistoricalMember,onRemoveHistoricalMember:e.onRemoveHistoricalMember})]}):(0,R.jsxs)(R.Fragment,{children:[x,(0,R.jsxs)(`section`,{className:`grid gap-3 md:grid-cols-2`,"aria-label":`Project summary`,children:[(0,R.jsx)(On,{id:`projectMetricOpen`,label:`Open`,value:e.summary.open_project_count||0}),(0,R.jsx)(On,{id:`projectMetricTotal`,label:`Projects`,value:e.summary.project_count||0})]}),e.canWrite?(0,R.jsxs)(V,{className:`flex flex-wrap items-center justify-between gap-3 p-4`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Selected`}),(0,R.jsxs)(`strong`,{className:`block`,children:[g.length,` project(s)`]})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-2`,children:[(0,R.jsxs)(B,{type:`button`,onClick:()=>f(!0),children:[(0,R.jsx)(S,{}),`New project`]}),(0,R.jsx)(B,{type:`button`,variant:`outline`,disabled:g.length===0,onClick:()=>u(!0),children:`Bulk edit`})]})]}):null,d?(0,R.jsx)(Kn,{loading:e.loading.createProject,onClose:()=>f(!1),onSearchCustomers:e.onSearchCustomers,onSearchContacts:e.onSearchContacts,onSearchAccountManagers:e.onSearchAccountManagers,onLoadCostCenters:e.onLoadCostCenters,onCreateProject:e.onCreateProject}):null,c?(0,R.jsxs)(`div`,{className:`fixed inset-0 z-50 grid place-items-center p-4`,"aria-labelledby":`bulkProjectEditTitle`,"aria-modal":`true`,role:`dialog`,children:[(0,R.jsx)(`button`,{type:`button`,className:`absolute inset-0 cursor-default bg-black/45`,"aria-label":`Close bulk project edit`,onClick:()=>u(!1)}),(0,R.jsxs)(`div`,{className:`relative grid w-full max-w-lg gap-4 rounded-md border bg-background p-5 shadow-2xl`,children:[(0,R.jsxs)(`div`,{className:`flex items-start justify-between gap-3`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`strong`,{id:`bulkProjectEditTitle`,className:`block text-base`,children:`Bulk edit projects`}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[g.length,` selected`]})]}),(0,R.jsx)(B,{type:`button`,variant:`ghost`,size:`icon`,"aria-label":`Close bulk project edit`,onClick:()=>u(!1),children:(0,R.jsx)(w,{})})]}),(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsx)(`strong`,{className:`text-sm`,children:`Changes`}),(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{value:i,onChange:e=>a(e.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`No change`}),(0,R.jsx)(`option`,{value:`Open`,children:`Open`}),(0,R.jsx)(`option`,{value:`Completed`,children:`Completed`}),(0,R.jsx)(`option`,{value:`Cancelled`,children:`Cancelled`})]})]}),(0,R.jsxs)(W,{children:[`ERP Type`,(0,R.jsxs)(Vt,{value:o,onChange:e=>s(e.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`No change`}),(0,R.jsx)(`option`,{value:`Internal`,children:`Internal`}),(0,R.jsx)(`option`,{value:`External`,children:`External`})]})]})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-2`,children:[(0,R.jsx)(B,{type:`button`,variant:`outline`,onClick:()=>u(!1),children:`Cancel`}),(0,R.jsx)(B,{type:`button`,disabled:e.loading.projectsBulkUpdate||g.length===0||!i&&!o,onClick:()=>void b(),children:`Apply changes`})]})]})]}):null,(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`ERP projects`}),(0,R.jsx)(`span`,{id:`projectsStatus`,className:`text-sm text-muted-foreground`,children:e.loading.projects?`Loading`:`${e.projects.length} shown | synced ${qt(e.summary.last_synced_at)}`})]}),(0,R.jsx)(kn,{hidden:e.projects.length!==0,children:`No projects match this view. Sync ERP projects if the cache is empty.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`projectsTable`,className:L(`min-w-[1100px]`,e.projects.length===0&&`hidden`),"aria-label":`ERP projects`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[e.canWrite?(0,R.jsx)(G,{className:`w-[48px]`,children:(0,R.jsx)(`input`,{type:`checkbox`,"aria-label":`Select all visible projects`,checked:_,onChange:e=>{r(e.target.checked?p:[])}})}):null,(0,R.jsx)(Dn,{className:`w-[24%]`,label:`Project`,scope:`projects`,sort:e.sort,sortKey:`display_name`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[16%]`,label:`Customer`,scope:`projects`,sort:e.sort,sortKey:`customer`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[10%]`,label:`Status`,scope:`projects`,sort:e.sort,sortKey:`status`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(G,{className:`w-[16%]`,children:`Timeline`}),(0,R.jsx)(Dn,{className:`w-[10%]`,label:`Roster`,scope:`projects`,sort:e.sort,sortKey:`roster_count`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[14%]`,label:`Modified`,scope:`projects`,sort:e.sort,sortKey:`modified`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(G,{children:`ERP`})]})}),(0,R.jsx)(Wt,{id:`projectsBody`,children:e.projects.map(t=>{let n=t.roster_members||[];return(0,R.jsxs)(Gt,{children:[e.canWrite?(0,R.jsx)(K,{children:(0,R.jsx)(`input`,{type:`checkbox`,"aria-label":`Select ${t.display_name}`,checked:g.includes(t.id),onChange:e=>v(t.id,e.target.checked)})}):null,(0,R.jsxs)(K,{children:[(0,R.jsx)(`button`,{type:`button`,className:`text-left font-bold text-primary underline-offset-4 hover:underline`,onClick:()=>e.onOpenProject(t.id),children:t.display_name}),(0,R.jsxs)(`div`,{className:`mt-1 flex flex-wrap items-center gap-1.5`,children:[t.project_type?(0,R.jsx)(z,{variant:`neutral`,children:t.project_type}):null,t.linked_engagement_count?(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[t.linked_engagement_count,` linked gig`]}):null]})]}),(0,R.jsx)(K,{children:t.customer_erpnext_url?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-semibold text-primary underline-offset-4 hover:underline`,href:t.customer_erpnext_url,target:`_blank`,rel:`noreferrer`,children:[t.customer,(0,R.jsx)(y,{className:`size-3.5`})]}):t.customer||`None`}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:Vn(t.source_status),children:t.source_status||`Unknown`})}),(0,R.jsx)(K,{children:[t.actual_start_date,t.actual_end_date].filter(Boolean).map(e=>Hn(e)).join(` to `)||`Not set`}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-1`,children:[(0,R.jsx)(`strong`,{children:n.length}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[n.map(Un).slice(0,4).join(`, `)||`No ERP roster`,n.length>4?` +${n.length-4}`:``]})]})}),(0,R.jsx)(K,{children:qt(t.source_modified_at)}),(0,R.jsx)(K,{className:`text-xs`,children:t.erpnext_project_url?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-mono font-semibold text-primary underline-offset-4 hover:underline`,href:t.erpnext_project_url,target:`_blank`,rel:`noreferrer`,children:[t.erpnext_project_id,(0,R.jsx)(y,{className:`size-3.5`})]}):(0,R.jsx)(`span`,{className:`font-mono`,children:`Unlinked`})})]},t.id)})})]})})]}),e.wikiMatches?(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Wiki match preview`}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[e.wikiMatches.document?.title||`Client & Project Info`,` |`,` `,qt(e.wikiMatches.document?.updatedAt)]})]}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`wikiMatchesTable`,className:`min-w-[920px]`,"aria-label":`Wiki matches`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{children:`ERP project`}),(0,R.jsx)(G,{children:`Best wiki row`}),(0,R.jsx)(G,{children:`Confidence`}),(0,R.jsx)(G,{children:`Section`}),(0,R.jsx)(G,{children:`Decision`})]})}),(0,R.jsx)(Wt,{children:t.map((t,n)=>{let r=t.project,i=t.best_match?.row||{},a=t.manual_match?.match_status||``,o=r?.id||i.row_key||[i.section,i.Client].filter(Boolean).join(`:`)||`wiki-match-${n}`;return(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:r?.display_name||`Unknown`}),(0,R.jsxs)(K,{children:[(0,R.jsx)(`strong`,{children:i.Client||`No match`}),(0,R.jsx)(`div`,{className:`text-sm text-muted-foreground`,children:[i.DRI,i.Members].filter(Boolean).join(` | `)})]}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:t.best_match?.confidence===`high`?`succeeded`:t.best_match?.confidence===`medium`?`running`:`neutral`,children:t.best_match?`${t.best_match.confidence} ${t.best_match.score}`:`none`})}),(0,R.jsx)(K,{children:i.section||``}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-2`,children:[a?(0,R.jsx)(z,{variant:a===`confirmed`?`succeeded`:`neutral`,children:a===`no_row`?`No wiki row`:`Confirmed`}):null,e.canWrite&&r?.id?(0,R.jsxs)(R.Fragment,{children:[i.row_key?(0,R.jsx)(B,{type:`button`,variant:`outline`,size:`sm`,disabled:e.loading[`project:${r.id}:wiki`],onClick:()=>void e.onUpdateWikiMatch(r.id,`confirmed`,i.row_key),children:`Confirm`}):null,(0,R.jsx)(B,{type:`button`,variant:`outline`,size:`sm`,disabled:e.loading[`project:${r.id}:wiki`],onClick:()=>void e.onUpdateWikiMatch(r.id,`no_row`),children:`No row`})]}):null]})})]},o)})})]})})]}):null]})}function Kn(e){let[t,n]=(0,l.useState)(``),[r,i]=(0,l.useState)(`new`),[a,o]=(0,l.useState)(``),[s,c]=(0,l.useState)(``),[u,d]=(0,l.useState)(``),[f,p]=(0,l.useState)([]),[m,h]=(0,l.useState)(``),[g,_]=(0,l.useState)(``),[v,y]=(0,l.useState)([]),[b,x]=(0,l.useState)(`USD`),[ee,te]=(0,l.useState)(``),[S,C]=(0,l.useState)(``),[ne,re]=(0,l.useState)(``),[ie,ae]=(0,l.useState)(``),[oe,se]=(0,l.useState)(``),[ce,T]=(0,l.useState)(``),[le,E]=(0,l.useState)(`United States`),[ue,de]=(0,l.useState)(``),[fe,pe]=(0,l.useState)(`new`),[D,O]=(0,l.useState)(``),[k,me]=(0,l.useState)(``),[A,he]=(0,l.useState)([]),[ge,_e]=(0,l.useState)(``),[ve,ye]=(0,l.useState)(``),[be,xe]=(0,l.useState)(``),[Se,Ce]=(0,l.useState)(``),[we,Te]=(0,l.useState)(``),[Ee,De]=(0,l.useState)(!1),[Oe,ke]=(0,l.useState)([{name:`Projects - 5`,cost_center_name:`Projects`}]),[Ae,je]=(0,l.useState)(`Projects - 5`),[Me,Ne]=(0,l.useState)(``),[Pe,Fe]=(0,l.useState)(!1),Ie=(0,l.useRef)(e.onSearchCustomers),Le=(0,l.useRef)(e.onSearchContacts),Re=(0,l.useRef)(e.onSearchAccountManagers),ze=(0,l.useRef)(e.onLoadCostCenters),Be=(0,l.useRef)(0),Ve=(0,l.useRef)(0),He=(0,l.useRef)(0),Ue=(0,l.useRef)(0),We=t.trim()?`Engineering for ${t.trim()}`.slice(0,140):``,Ge=[ne,ie,oe,ce,ue].some(e=>e.trim()),Ke=[ge,ve,be,Se,we].some(e=>e.trim()),qe=t.trim()&&(r===`new`?a.trim():u.trim())&&!e.loading;(0,l.useEffect)(()=>{Ie.current=e.onSearchCustomers},[e.onSearchCustomers]),(0,l.useEffect)(()=>{Le.current=e.onSearchContacts},[e.onSearchContacts]),(0,l.useEffect)(()=>{Re.current=e.onSearchAccountManagers},[e.onSearchAccountManagers]),(0,l.useEffect)(()=>{ze.current=e.onLoadCostCenters},[e.onLoadCostCenters]),(0,l.useEffect)(()=>{let e=!0,t=Be.current+1;return Be.current=t,ze.current().then(n=>{!e||Be.current!==t||(ke(n),je(e=>n.some(t=>t.name===e)?e:`Projects - 5`))}),()=>{e=!1}},[]),(0,l.useEffect)(()=>{if(r!==`existing`){Ve.current+=1,p([]);return}let e=!0,t=Ve.current+1;Ve.current=t;let n=window.setTimeout(()=>{Ie.current(s).then(n=>{!e||Ve.current!==t||p(n)})},250);return()=>{e=!1,window.clearTimeout(n)}},[r,s]),(0,l.useEffect)(()=>{if(r!==`new`){He.current+=1,y([]);return}let e=!0,t=He.current+1;He.current=t;let n=window.setTimeout(()=>{Re.current(m).then(n=>{!e||He.current!==t||y(n)})},250);return()=>{e=!1,window.clearTimeout(n)}},[r,m]),(0,l.useEffect)(()=>{if(r!==`new`||fe!==`existing`){Ue.current+=1,he([]);return}let e=!0,t=Ue.current+1;Ue.current=t;let n=window.setTimeout(()=>{Le.current(D).then(n=>{!e||Ue.current!==t||he(n)})},250);return()=>{e=!1,window.clearTimeout(n)}},[r,fe,D]);async function Je(){qe&&await e.onCreateProject({project_name:t.trim(),customer_mode:r,customer_name:r===`new`?a.trim():void 0,customer:r===`existing`?u.trim():void 0,account_manager:r===`new`&&g.trim()||void 0,default_billing_currency:r===`new`?b.trim()||`USD`:void 0,default_cost_center:Ae.trim()||`Projects - 5`,activity_type:Pe&&Me.trim()||void 0,customer_details:r===`new`&&ee.trim()||void 0,customer_website:r===`new`&&S.trim()||void 0,address_line1:r===`new`&&ne.trim()||void 0,address_line2:r===`new`&&ie.trim()||void 0,address_city:r===`new`&&oe.trim()||void 0,address_state:r===`new`&&ce.trim()||void 0,address_country:r===`new`&&ne.trim()?le.trim()||`United States`:void 0,address_postal_code:r===`new`&&ue.trim()||void 0,contact:r===`new`&&fe===`existing`&&k.trim()||void 0,contact_first_name:r===`new`&&fe===`new`&&ge.trim()||void 0,contact_last_name:r===`new`&&fe===`new`&&ve.trim()||void 0,contact_email:r===`new`&&fe===`new`&&be.trim()||void 0,contact_phone:r===`new`&&fe===`new`&&Se.trim()||void 0,contact_mobile:r===`new`&&fe===`new`&&we.trim()||void 0})&&e.onClose()}return(0,R.jsxs)(`div`,{className:`fixed inset-0 z-50 grid place-items-center p-4`,"aria-labelledby":`createProjectTitle`,"aria-modal":`true`,role:`dialog`,children:[(0,R.jsx)(`button`,{type:`button`,className:`absolute inset-0 cursor-default bg-black/45`,"aria-label":`Close project creation`,onClick:e.onClose}),(0,R.jsxs)(`div`,{className:`relative grid max-h-[90vh] w-full max-w-2xl gap-4 overflow-y-auto rounded-md border bg-background p-5 shadow-2xl`,children:[(0,R.jsxs)(`div`,{className:`flex items-start justify-between gap-3`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`strong`,{id:`createProjectTitle`,className:`block text-base`,children:`New ERP project`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:`Creates a project and links a new or existing customer.`})]}),(0,R.jsx)(B,{type:`button`,variant:`ghost`,size:`icon`,"aria-label":`Close project creation`,onClick:e.onClose,children:(0,R.jsx)(w,{})})]}),(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsxs)(W,{children:[`Project name *`,(0,R.jsx)(U,{value:t,autoComplete:`off`,maxLength:140,placeholder:`Acme Portal`,onChange:e=>n(e.target.value)})]}),(0,R.jsxs)(`div`,{className:`grid gap-2`,children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Customer`}),(0,R.jsx)(`div`,{className:`grid grid-cols-2 gap-2`,children:[`new`,`existing`].map(e=>(0,R.jsx)(B,{type:`button`,variant:r===e?`default`:`outline`,onClick:()=>i(e),children:e===`new`?`New customer`:`Existing customer`},e))})]}),r===`new`?(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Customer name *`,(0,R.jsx)(U,{value:a,autoComplete:`off`,maxLength:140,placeholder:`Acme`,onChange:e=>o(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Account manager`,(0,R.jsx)(U,{value:m,autoComplete:`off`,placeholder:`Search @508.dev user`,onChange:e=>{h(e.target.value),_(``)}})]}),m.trim().length>=2?(0,R.jsx)(`div`,{className:`grid max-h-40 gap-2 overflow-y-auto rounded-md border p-2 md:col-span-2`,children:v.length?v.map(e=>{let t=e.email||e.name||``;return(0,R.jsxs)(`label`,{className:`flex cursor-pointer items-start gap-2 rounded-sm px-2 py-1.5 hover:bg-secondary`,children:[(0,R.jsx)(`input`,{type:`radio`,name:`erpAccountManager`,value:t,checked:g===t,onChange:()=>{_(t),h(t)}}),(0,R.jsxs)(`span`,{className:`grid gap-0.5 text-sm`,children:[(0,R.jsx)(`strong`,{children:e.full_name||t}),(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:t})]})]},t)}):(0,R.jsx)(`span`,{className:`px-2 py-3 text-sm text-muted-foreground`,children:`No enabled @508.dev users found.`})}):null]}):(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsxs)(W,{children:[`Find customer *`,(0,R.jsx)(U,{value:s,autoComplete:`off`,placeholder:`Search customer`,onChange:e=>c(e.target.value)})]}),(0,R.jsx)(`div`,{className:`grid max-h-48 gap-2 overflow-y-auto rounded-md border p-2`,children:f.length?f.map(e=>{let t=e.name||e.customer_name||``;return(0,R.jsxs)(`label`,{className:`flex cursor-pointer items-start gap-2 rounded-sm px-2 py-1.5 hover:bg-secondary`,children:[(0,R.jsx)(`input`,{type:`radio`,name:`erpCustomer`,value:t,checked:u===t,onChange:()=>d(t)}),(0,R.jsxs)(`span`,{className:`grid gap-0.5 text-sm`,children:[(0,R.jsx)(`strong`,{children:e.customer_name||t}),(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:[t,e.default_currency].filter(Boolean).join(` | `)})]})]},t)}):(0,R.jsx)(`span`,{className:`px-2 py-3 text-sm text-muted-foreground`,children:`Search at least two characters.`})})]}),r===`new`?(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Customer details`,(0,R.jsx)(`textarea`,{value:ee,className:`min-h-20 w-full rounded-md border border-input bg-background px-3 py-2 text-sm text-foreground shadow-xs transition-colors placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]`,maxLength:2e3,placeholder:`More information`,onChange:e=>te(e.target.value)})]}),(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Website`,(0,R.jsx)(U,{value:S,autoComplete:`url`,placeholder:`https://example.com`,onChange:e=>C(e.target.value)})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsxs)(`div`,{className:`flex items-center justify-between gap-3`,children:[(0,R.jsx)(`strong`,{className:`text-sm text-foreground`,children:`Contact`}),(0,R.jsx)(`div`,{className:`grid grid-cols-2 gap-2`,children:[`new`,`existing`].map(e=>(0,R.jsx)(B,{type:`button`,size:`sm`,variant:fe===e?`default`:`outline`,onClick:()=>pe(e),children:e===`new`?`New`:`Existing`},e))})]}),fe===`new`?(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[(0,R.jsxs)(W,{children:[`First name `,Ke?`*`:``,(0,R.jsx)(U,{value:ge,autoComplete:`given-name`,onChange:e=>_e(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Last name`,(0,R.jsx)(U,{value:ve,autoComplete:`family-name`,onChange:e=>ye(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Email`,(0,R.jsx)(U,{value:be,type:`email`,autoComplete:`email`,onChange:e=>xe(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Phone`,(0,R.jsx)(U,{value:Se,type:`tel`,autoComplete:`tel`,onChange:e=>Ce(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Mobile`,(0,R.jsx)(U,{value:we,type:`tel`,autoComplete:`tel`,onChange:e=>Te(e.target.value)})]})]}):(0,R.jsxs)(`div`,{className:`grid gap-3`,children:[(0,R.jsxs)(W,{children:[`Find contact`,(0,R.jsx)(U,{value:D,autoComplete:`off`,placeholder:`Search name or email`,onChange:e=>O(e.target.value)})]}),(0,R.jsx)(`div`,{className:`grid max-h-48 gap-2 overflow-y-auto rounded-md border p-2`,children:A.length?A.map(e=>{let t=e.name||``,n=e.full_name||t,r=[{key:`company`,value:e.company_name},{key:`email`,value:e.email_id},{key:`phone`,value:e.phone},{key:`mobile`,value:e.mobile_no}].filter(e=>!!e.value);return(0,R.jsxs)(`label`,{className:`flex cursor-pointer items-start gap-2 rounded-sm px-2 py-1.5 hover:bg-secondary`,children:[(0,R.jsx)(`input`,{type:`radio`,name:`erpContact`,value:t,checked:k===t,onChange:()=>me(t)}),(0,R.jsxs)(`span`,{className:`grid gap-0.5 text-sm`,children:[(0,R.jsx)(`strong`,{children:(0,R.jsx)(An,{value:n,query:D})}),r.length?(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:r.map((e,t)=>(0,R.jsxs)(`span`,{children:[t>0?` | `:``,(0,R.jsx)(An,{value:e.value,query:D})]},e.key))}):null]})]},t)}):(0,R.jsx)(`span`,{className:`px-2 py-3 text-sm text-muted-foreground`,children:`Search at least two characters.`})})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[(0,R.jsx)(`strong`,{className:`text-sm text-foreground md:col-span-2`,children:`Address`}),(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Address line 1 `,Ge?`*`:``,(0,R.jsx)(U,{value:ne,autoComplete:`address-line1`,onChange:e=>re(e.target.value)})]}),(0,R.jsxs)(W,{className:`md:col-span-2`,children:[`Address line 2`,(0,R.jsx)(U,{value:ie,autoComplete:`address-line2`,onChange:e=>ae(e.target.value)})]}),(0,R.jsxs)(W,{children:[`City`,(0,R.jsx)(U,{value:oe,autoComplete:`address-level2`,onChange:e=>se(e.target.value)})]}),(0,R.jsxs)(W,{children:[`State`,(0,R.jsx)(U,{value:ce,autoComplete:`address-level1`,onChange:e=>T(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Postal code`,(0,R.jsx)(U,{value:ue,autoComplete:`postal-code`,onChange:e=>de(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Country`,(0,R.jsx)(U,{value:le,autoComplete:`country-name`,onChange:e=>E(e.target.value)})]})]})]}):null,(0,R.jsxs)(`div`,{className:`grid gap-3 rounded-md border p-3`,children:[(0,R.jsx)(B,{type:`button`,variant:`outline`,onClick:()=>De(e=>!e),children:Ee?`Hide advanced`:`Show advanced`}),Ee?(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[r===`new`?(0,R.jsxs)(W,{children:[`Billing currency`,(0,R.jsx)(U,{value:b,autoComplete:`off`,maxLength:3,onChange:e=>x(e.target.value.toUpperCase())})]}):null,(0,R.jsxs)(W,{children:[`Cost center`,(0,R.jsx)(Vt,{value:Ae,onChange:e=>je(e.target.value),children:Oe.map(e=>{let t=e.name||``;return(0,R.jsx)(`option`,{value:t,children:[t,e.company].filter(Boolean).join(` | `)},t)})})]}),(0,R.jsxs)(W,{children:[`Activity type`,(0,R.jsx)(U,{value:Pe?Me:We,autoComplete:`off`,maxLength:140,placeholder:We||`Engineering for project`,onChange:e=>{Fe(!0),Ne(e.target.value)}})]})]}):null]})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-2`,children:[(0,R.jsx)(B,{type:`button`,variant:`outline`,onClick:e.onClose,children:`Cancel`}),(0,R.jsx)(B,{type:`button`,disabled:!qe,onClick:()=>void Je(),children:`Create project`})]})]})]})}function qn(e){let t=e.project,n=t.roster_members||[],[r,i]=(0,l.useState)(``),[a,o]=(0,l.useState)([]),[s,c]=(0,l.useState)(``),[u,d]=(0,l.useState)(``),[f,p]=(0,l.useState)(``),[m,g]=(0,l.useState)(``),_=[t.actual_start_date||t.expected_start_date,t.actual_end_date||t.expected_end_date].filter(Boolean).map(e=>Hn(e)).join(` to `)||`Not set`,v=typeof t.percent_complete==`number`?`${Math.round(t.percent_complete)}%`:`Not set`,b=a.find(e=>e.candidate_id===s),x=r.trim().includes(`@`)?r.trim().length>=5:r.trim().length>=3,ee=!!(u.trim()||f.trim()||m.trim()),te=Wn(f),S=Wn(m),C=!!((f.trim()||m.trim())&&!u.trim()),ne=!!(u.trim()&&(!f.trim()||!m.trim())),re=!!(f.trim()&&te===void 0)||!!(m.trim()&&S===void 0),ie=C||ne||re||te!==void 0&&te<0||S!==void 0&&S<0,ae=ee&&!ie?{activity_type:u.trim(),billing_rate:te,costing_rate:S}:void 0;(0,l.useEffect)(()=>{if(!e.canWrite)return;let t=r.trim();if(s&&b&&t===(b.email||b.label||``))return;if(s&&c(``),!(t.includes(`@`)?t.length>=5:t.length>=3)){o([]);return}let n=new AbortController,i=window.setTimeout(()=>{q(`/dashboard/api/project-member-candidates?query=${encodeURIComponent(t)}`,{signal:n.signal}).then(e=>o(e)).catch(e=>{e instanceof DOMException&&e.name===`AbortError`||o([])})},500);return()=>{n.abort(),window.clearTimeout(i)}},[r,e.canWrite,b,s]);function se(e){c(e.candidate_id),i(e.email||e.label||e.full_name||r)}return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-[auto_minmax(0,1fr)_auto] md:items-start`,children:[(0,R.jsxs)(B,{type:`button`,variant:`outline`,onClick:e.onBack,children:[(0,R.jsx)(h,{}),`Projects`]}),(0,R.jsxs)(`div`,{className:`min-w-0`,children:[(0,R.jsx)(zt,{children:t.display_name}),(0,R.jsxs)(`div`,{className:`mt-2 flex flex-wrap items-center gap-2 text-sm text-muted-foreground`,children:[(0,R.jsx)(z,{variant:Vn(t.source_status),children:t.source_status||`Unknown`}),t.erpnext_project_id?(0,R.jsx)(`span`,{className:`font-mono`,children:t.erpnext_project_id}):null,t.last_synced_at?(0,R.jsxs)(`span`,{children:[`Synced `,qt(t.last_synced_at)]}):null]})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-start gap-2 md:justify-end`,children:[e.canWrite?(0,R.jsxs)(Vt,{className:`w-[160px]`,"aria-label":`Status for ${t.display_name}`,value:t.source_status||``,disabled:e.loading[`project:${t.id}:status`],onChange:n=>e.onUpdateStatus(t.id,n.target.value),children:[(0,R.jsx)(`option`,{value:`Open`,children:`Open`}),(0,R.jsx)(`option`,{value:`Completed`,children:`Completed`}),(0,R.jsx)(`option`,{value:`Cancelled`,children:`Cancelled`})]}):null,t.erpnext_project_url?(0,R.jsxs)(`a`,{className:`inline-flex min-h-9 items-center justify-center gap-2 rounded-md border bg-secondary px-3 text-sm font-semibold`,href:t.erpnext_project_url,target:`_blank`,rel:`noreferrer`,children:[(0,R.jsx)(y,{className:`size-4`}),`ERP project`]}):null,t.customer_erpnext_url?(0,R.jsxs)(`a`,{className:`inline-flex min-h-9 items-center justify-center gap-2 rounded-md border bg-secondary px-3 text-sm font-semibold`,href:t.customer_erpnext_url,target:`_blank`,rel:`noreferrer`,children:[(0,R.jsx)(y,{className:`size-4`}),`ERP customer`]}):null]})]})}),(0,R.jsxs)(Bt,{className:`grid gap-4 md:grid-cols-2 lg:grid-cols-4`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Customer`}),(0,R.jsx)(`strong`,{className:`block`,children:t.customer||`None`})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Timeline`}),(0,R.jsx)(`strong`,{className:`block`,children:_})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Progress`}),(0,R.jsx)(`strong`,{className:`block`,children:v})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Linked Gigs`}),(0,R.jsx)(`strong`,{className:`block`,children:t.linked_engagement_count||0})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`ERP Type`}),(0,R.jsx)(`div`,{className:`mt-1`,children:t.project_type?(0,R.jsx)(z,{variant:`neutral`,children:t.project_type}):(0,R.jsx)(`strong`,{className:`block`,children:`Not set`})})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`ERP Modified`}),(0,R.jsx)(`strong`,{className:`block`,children:qt(t.source_modified_at)})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Cache ID`}),(0,R.jsx)(`strong`,{className:`block break-all font-mono text-xs`,children:t.id})]})]})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Project roster`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:n.length?`${n.length} synced ERP user${n.length===1?``:`s`}`:`No ERP roster`})]}),e.canWrite?(0,R.jsxs)(Bt,{className:`grid gap-3 border-b md:grid-cols-[minmax(260px,1fr)_minmax(180px,.7fr)_minmax(130px,.45fr)_minmax(130px,.45fr)_auto_auto] md:items-end`,children:[(0,R.jsxs)(`div`,{className:`relative`,children:[(0,R.jsxs)(W,{children:[`Person search`,(0,R.jsx)(U,{value:r,autoComplete:`off`,placeholder:`Search @508.dev person`,onChange:e=>i(e.target.value),onKeyDown:e=>{e.key===`Enter`&&(e.preventDefault(),a.length===1&&se(a[0]))}})]}),x&&!s?(0,R.jsx)(`div`,{className:`absolute z-20 mt-1 max-h-64 w-full overflow-auto rounded-md border bg-background shadow-lg`,children:a.length?a.map(e=>(0,R.jsxs)(`button`,{type:`button`,className:`grid w-full gap-0.5 px-3 py-2 text-left hover:bg-secondary focus:bg-secondary focus:outline-none`,onClick:()=>se(e),children:[(0,R.jsx)(`span`,{className:`truncate text-sm font-bold`,children:e.label||e.full_name||e.email||`Person`}),(0,R.jsx)(`span`,{className:`truncate text-xs text-muted-foreground`,children:[e.email,e.sources?.join(`, `)].filter(Boolean).join(` | `)})]},e.candidate_id)):(0,R.jsx)(`div`,{className:`px-3 py-2 text-sm text-muted-foreground`,children:`No verified @508.dev results`})}):null]}),(0,R.jsxs)(W,{children:[`Activity Type`,(0,R.jsx)(U,{value:u,autoComplete:`off`,placeholder:`Optional rate step`,onChange:e=>d(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Billing rate`,(0,R.jsx)(U,{value:f,inputMode:`decimal`,autoComplete:`off`,placeholder:`USD/hr`,onChange:e=>p(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Costing rate`,(0,R.jsx)(U,{value:m,inputMode:`decimal`,autoComplete:`off`,placeholder:`USD/hr`,onChange:e=>g(e.target.value)})]}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,disabled:e.loading[`project:${t.id}:user`]||!s||!b?.email||ie,onClick:()=>void e.onAddUser(t.id,b?.email||r,s,ae).then(e=>{e&&(i(``),o([]),c(``),d(``),p(``),g(``))}),children:[(0,R.jsx)(ce,{}),`Add ERP user`]}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,disabled:e.loading[`project:${t.id}:historical`]||!r.trim(),onClick:()=>void e.onAddHistoricalMember(t.id,r).then(e=>{e&&i(``)}),children:[(0,R.jsx)(ce,{}),`Add historical`]})]}):null,(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{className:`min-w-[860px]`,"aria-label":`Project roster`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{children:`Name`}),(0,R.jsx)(G,{children:`Email`}),(0,R.jsx)(G,{children:`ERP user`}),(0,R.jsx)(G,{children:`Links`}),(0,R.jsx)(G,{children:`Source`}),(0,R.jsx)(G,{children:`Last seen`}),e.canWrite?(0,R.jsx)(G,{children:`Actions`}):null]})}),(0,R.jsx)(Wt,{children:n.length?n.map(n=>{let r=Un(n),i=n.source_user_id||n.email||``,a=n.roster_kind===`historical`||n.source===`manual`;return(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:(0,R.jsx)(`strong`,{children:n.full_name||n.email||n.source_user_id})}),(0,R.jsx)(K,{children:n.email||`None`}),(0,R.jsx)(K,{className:`font-mono text-xs`,children:n.erpnext_user_url?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-semibold text-primary underline-offset-4 hover:underline`,href:n.erpnext_user_url,target:`_blank`,rel:`noreferrer`,children:[n.source_user_id||`ERP user`,(0,R.jsx)(y,{className:`size-3.5`})]}):n.source_user_id||`Unknown`}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-2`,children:[n.supplier_erpnext_url?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-semibold text-primary underline-offset-4 hover:underline`,href:n.supplier_erpnext_url,target:`_blank`,rel:`noreferrer`,children:[`Supplier`,(0,R.jsx)(y,{className:`size-3.5`})]}):null,n.crm_contact_id&&e.crmContactUrl(n.crm_contact_id)?(0,R.jsxs)(`a`,{className:`inline-flex items-center gap-1 font-semibold text-primary underline-offset-4 hover:underline`,href:e.crmContactUrl(n.crm_contact_id),target:`_blank`,rel:`noreferrer`,children:[`CRM`,(0,R.jsx)(y,{className:`size-3.5`})]}):null,!n.supplier_erpnext_url&&!(n.crm_contact_id&&e.crmContactUrl(n.crm_contact_id))?(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:`None`}):null]})}),(0,R.jsx)(K,{children:n.roster_kind||n.source||`ERP`}),(0,R.jsx)(K,{children:qt(n.last_seen_at)}),e.canWrite?(0,R.jsx)(K,{children:(0,R.jsxs)(B,{type:`button`,variant:`outline`,size:`sm`,disabled:!i||e.loading[`project:${t.id}:${a?`historical`:`user`}`],onClick:()=>{window.confirm(`Remove ${r} from this project roster?`)&&(a?e.onRemoveHistoricalMember(t.id,i):e.onRemoveUser(t.id,i))},children:[(0,R.jsx)(oe,{}),`Remove`]})}):null]},`${n.source||``}:${n.source_user_id||n.email}`)}):(0,R.jsx)(Gt,{children:(0,R.jsx)(K,{colSpan:e.canWrite?7:6,className:`text-sm text-muted-foreground`,children:`No roster rows have been synced for this project.`})})})]})})]})]})}function Jn(e){let t=e.gigs.reduce((t,n)=>(t.total+=1,t.applications+=Number(n.application_count||0),t.interested+=Number(n.interested_count||0),Bn(n,e.staleDays)!==null&&(t.stale+=1),t),{total:0,applications:0,interested:0,stale:0}),n=(0,R.jsxs)(V,{className:`grid gap-3 p-4 md:grid-cols-[minmax(140px,.75fr)_minmax(220px,1.25fr)_auto_auto_auto] md:items-end`,children:[(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{id:`gigStatus`,value:e.status,onChange:t=>e.setStatus(t.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Any status`}),In.map(e=>(0,R.jsx)(`option`,{value:e,children:Rn(e)},e))]})]}),(0,R.jsxs)(W,{children:[`Search gigs`,(0,R.jsx)(U,{id:`gigQuery`,value:e.query,autoComplete:`off`,placeholder:`Title, gig text, #tag, @poster`,onChange:t=>e.setQuery(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onRefresh()})]}),e.canIncludeHistorical?(0,R.jsxs)(`label`,{className:`flex min-h-9 items-center gap-2 text-xs font-bold text-muted-foreground`,children:[(0,R.jsx)(`input`,{type:`checkbox`,checked:e.includeHistorical,onChange:t=>e.setIncludeHistorical(t.target.checked)}),`Include historical`]}):null,(0,R.jsxs)(B,{id:`searchGigs`,type:`button`,onClick:e.onRefresh,disabled:e.loading.gigs,children:[(0,R.jsx)(ne,{}),`Search`]}),(0,R.jsxs)(B,{id:`refreshGigs`,type:`button`,variant:`outline`,onClick:e.onRefresh,disabled:e.loading.gigs,children:[(0,R.jsx)(C,{}),`Refresh`]}),e.gigs.length>=e.limit?(0,R.jsx)(B,{type:`button`,variant:`outline`,onClick:()=>e.setLimit(Math.min(e.limit+100,500)),disabled:e.loading.gigs||e.limit>=500,children:`Load more`}):null]}),r=e.selectedGigId?e.loading[`gig:${e.selectedGigId}:detail`]:!1;return e.selectedGigId&&!e.selectedGig&&(e.loading.gigs||r)?(0,R.jsxs)(R.Fragment,{children:[n,(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Gig detail`})}),(0,R.jsx)(Bt,{className:`text-sm text-muted-foreground`,children:`Loading gig.`})]})]}):e.selectedGigId&&!e.selectedGig?(0,R.jsxs)(R.Fragment,{children:[n,(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Gig detail`})}),(0,R.jsxs)(Bt,{className:`grid gap-3`,children:[(0,R.jsx)(`p`,{className:`text-sm text-muted-foreground`,children:`This gig is not in the current result set. Clear filters or refresh the gig list.`}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,onClick:e.onCloseGig,children:[(0,R.jsx)(h,{}),`Back to gigs`]})]})]})]}):e.selectedGig?(0,R.jsxs)(R.Fragment,{children:[n,(0,R.jsx)(Zn,{gig:e.selectedGig,loading:e.loading,canWrite:e.canWrite,crmContactUrl:e.crmContactUrl,crmAttachmentUrl:e.crmAttachmentUrl,staleDays:e.staleDays,onBack:e.onCloseGig,onUpdateStatus:e.onUpdateStatus,onAddApplication:e.onAddApplication,onUpdateApplicationStatus:e.onUpdateApplicationStatus})]}):(0,R.jsxs)(R.Fragment,{children:[n,(0,R.jsxs)(`section`,{className:`grid gap-3 md:grid-cols-4`,"aria-label":`Gig summary`,children:[(0,R.jsx)(On,{id:`gigMetricTotal`,label:`Gigs`,value:t.total}),(0,R.jsx)(On,{id:`gigMetricCandidates`,label:`Candidates`,value:t.applications}),(0,R.jsx)(On,{id:`gigMetricInterested`,label:`Interested`,value:t.interested}),(0,R.jsx)(On,{id:`gigMetricStale`,label:`Stale recruiting`,value:t.stale})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Discord gigs`}),(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center justify-end gap-2`,children:[(0,R.jsxs)(B,{type:`button`,variant:`ghost`,size:`sm`,onClick:()=>e.onSort(`activity`),"aria-label":`Sort gigs by activity`,children:[`Activity`,` `,e.sort.key===`activity`?e.sort.direction===`asc`?`↑`:`↓`:``]}),(0,R.jsxs)(B,{type:`button`,variant:`ghost`,size:`sm`,onClick:()=>e.onSort(`title`),"aria-label":`Sort gigs by title`,children:[`Title `,e.sort.key===`title`?e.sort.direction===`asc`?`↑`:`↓`:``]}),(0,R.jsx)(`span`,{id:`gigsStatus`,className:`text-sm text-muted-foreground`,children:e.loading.gigs?`Loading`:`${e.gigs.length} shown`})]})]}),(0,R.jsx)(kn,{hidden:e.gigs.length!==0,children:`No gigs match this view.`}),(0,R.jsx)(`div`,{id:`gigsBody`,className:L(`grid gap-3 p-4`,e.gigs.length===0&&`hidden`),children:e.gigs.map(t=>(0,R.jsx)(Yn,{gig:t,loading:e.loading,canWrite:e.canWrite,staleDays:e.staleDays,onOpenGig:e.onOpenGig,onUpdateStatus:e.onUpdateStatus},t.id))})]})]})}function Yn({gig:e,loading:t,canWrite:n,onOpenGig:r,onUpdateStatus:i,staleDays:a}){let o=Array.isArray(e.applications)?e.applications:[],s=e.status===`recruiting`,c=e.discord_guild_id&&e.discord_thread_id?`https://discord.com/channels/${encodeURIComponent(e.discord_guild_id)}/${encodeURIComponent(e.discord_thread_id)}`:``,l=Bn(e,a);return(0,R.jsxs)(`article`,{className:L(`grid gap-4 rounded-md border bg-background p-4 lg:grid-cols-[minmax(0,1fr)_220px_180px] lg:items-start`,!s&&`border-l-4 border-l-muted-foreground/60 bg-secondary/45`),children:[(0,R.jsxs)(`div`,{className:`min-w-0`,children:[(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-2`,children:[(0,R.jsx)(`a`,{className:`text-base font-extrabold text-primary`,href:`/dashboard/gigs/${encodeURIComponent(e.id)}`,onClick:t=>{t.preventDefault(),r(e.id)},children:e.title||`Untitled gig`}),(0,R.jsx)(z,{variant:e.status===`filled`?`succeeded`:e.status===`lost`?`failed`:s?`queued`:`neutral`,children:e.status_label||Rn(e.status)}),s?null:(0,R.jsx)(z,{variant:`neutral`,children:`Not recruiting`}),l===null?null:(0,R.jsxs)(z,{variant:`running`,children:[l,`d stale`]})]}),(0,R.jsxs)(`div`,{className:`mt-2 flex flex-wrap gap-1.5`,children:[e.posting_type?(0,R.jsx)(z,{variant:`neutral`,children:Rn(e.posting_type)}):null,e.discord_channel_name?(0,R.jsxs)(z,{variant:`neutral`,children:[`#`,e.discord_channel_name]}):null,(e.required_skills||[]).slice(0,5).map(e=>(0,R.jsx)(z,{variant:`queued`,children:e},e)),(e.preferred_skills||[]).slice(0,3).map(e=>(0,R.jsx)(z,{variant:`neutral`,children:e},e))]}),(0,R.jsxs)(`div`,{className:`mt-3 flex flex-wrap gap-x-4 gap-y-1 text-sm text-muted-foreground`,children:[(0,R.jsxs)(`span`,{children:[`Activity `,qt(zn(e))||`unknown`]}),(0,R.jsxs)(`span`,{children:[`Posted `,qt(e.posted_at)||`unknown`]}),c?(0,R.jsx)(`a`,{className:`font-extrabold text-primary`,href:c,target:`_blank`,rel:`noreferrer`,children:`Open Discord thread`}):null]})]}),(0,R.jsxs)(`div`,{className:`grid grid-cols-2 gap-2 text-sm lg:grid-cols-1`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`block text-xs font-bold text-muted-foreground`,children:`People`}),(0,R.jsx)(`strong`,{children:e.application_count||o.length}),(0,R.jsxs)(`span`,{className:`ml-2 text-muted-foreground`,children:[Number(e.interested_count||0),` interested`]})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`block text-xs font-bold text-muted-foreground`,children:`Top candidates`}),(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:o.slice(0,3).map(e=>Xn(e)).join(`, `)||`None yet`})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-2`,children:[n?(0,R.jsx)(Vt,{"aria-label":`Status for ${e.title||`gig`}`,value:e.status,disabled:t[`gig:${e.id}:status`],onChange:t=>i(e.id,t.target.value),children:In.map(e=>(0,R.jsx)(`option`,{value:e,children:Rn(e)},e))}):null,(0,R.jsx)(B,{type:`button`,onClick:()=>r(e.id),children:`Manage people`})]})]})}function Xn(e){return e.name||e.email_508||e.discord_username||(typeof e.evaluation?.discord_username==`string`?e.evaluation.discord_username:``)||`Candidate`}function Zn({gig:e,loading:t,canWrite:n,crmContactUrl:r,crmAttachmentUrl:i,staleDays:a,onBack:o,onUpdateStatus:s,onAddApplication:c,onUpdateApplicationStatus:u}){let[d,f]=(0,l.useState)(``),p=Array.isArray(e.applications)?e.applications:[],m=e.status===`recruiting`,g=e.discord_guild_id&&e.discord_thread_id?`https://discord.com/channels/${encodeURIComponent(e.discord_guild_id)}/${encodeURIComponent(e.discord_thread_id)}`:``,_=Bn(e,a);return(0,R.jsxs)(`div`,{className:`grid gap-5`,children:[(0,R.jsxs)(V,{className:L(!m&&`border-l-4 border-l-muted-foreground/60 bg-secondary/35`),children:[(0,R.jsxs)(H,{className:`items-start`,children:[(0,R.jsxs)(`div`,{className:`grid gap-2`,children:[(0,R.jsxs)(B,{type:`button`,variant:`ghost`,size:`sm`,className:`w-fit`,onClick:o,children:[(0,R.jsx)(h,{}),`Back to gigs`]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(zt,{className:`text-xl`,children:e.title||`Untitled gig`}),(0,R.jsxs)(`div`,{className:`mt-2 flex flex-wrap gap-1.5`,children:[(0,R.jsx)(z,{variant:e.status===`filled`?`succeeded`:e.status===`lost`?`failed`:m?`queued`:`neutral`,children:e.status_label||Rn(e.status)}),m?null:(0,R.jsx)(z,{variant:`neutral`,children:`Not recruiting`}),_===null?null:(0,R.jsxs)(z,{variant:`running`,children:[_,`d stale`]}),e.posting_type?(0,R.jsx)(z,{variant:`neutral`,children:Rn(e.posting_type)}):null,e.discord_channel_name?(0,R.jsxs)(z,{variant:`neutral`,children:[`#`,e.discord_channel_name]}):null,(e.required_skills||[]).map(e=>(0,R.jsx)(z,{variant:`queued`,children:e},e)),(e.preferred_skills||[]).map(e=>(0,R.jsx)(z,{variant:`neutral`,children:e},e))]})]})]}),(0,R.jsxs)(`div`,{className:`grid min-w-[190px] gap-2`,children:[n?(0,R.jsxs)(W,{children:[`Gig status`,(0,R.jsx)(Vt,{"aria-label":`Status for ${e.title||`gig`}`,value:e.status,disabled:t[`gig:${e.id}:status`],onChange:t=>s(e.id,t.target.value),children:In.map(e=>(0,R.jsx)(`option`,{value:e,children:Rn(e)},e))})]}):null,g?(0,R.jsxs)(`a`,{className:`inline-flex min-h-9 items-center justify-center gap-2 rounded-md border bg-secondary px-3 text-sm font-semibold`,href:g,target:`_blank`,rel:`noreferrer`,children:[(0,R.jsx)(y,{className:`size-4`}),`Discord thread`]}):null]})]}),(0,R.jsxs)(Bt,{className:`grid gap-4 lg:grid-cols-[1fr_1fr_1fr]`,children:[(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Activity`}),(0,R.jsx)(`strong`,{className:`block`,children:qt(zn(e))||`unknown`}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[`Posted `,qt(e.posted_at)||`unknown`]})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`People`}),(0,R.jsx)(`strong`,{className:`block`,children:e.application_count||p.length}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[Number(e.interested_count||0),` interested`]})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:`Discord`}),(0,R.jsx)(`strong`,{className:`block`,children:e.discord_channel_name||`No channel`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.discord_thread_id?`Thread ${e.discord_thread_id}`:`No thread`})]})]})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`People`}),(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[p.length,` candidate`,p.length===1?``:`s`]})]}),n?(0,R.jsxs)(`form`,{className:`grid gap-2 border-t p-4 md:grid-cols-[minmax(220px,1fr)_auto]`,onSubmit:t=>{t.preventDefault(),c(e.id,d).then(e=>{e&&f(``)})},children:[(0,R.jsxs)(W,{className:`min-w-0`,children:[`CRM profile`,(0,R.jsx)(U,{value:d,onChange:e=>f(e.target.value),placeholder:`https://crm.508.dev/#Contact/view/...`,"aria-label":`CRM profile for candidate`})]}),(0,R.jsxs)(B,{type:`submit`,className:`self-end`,disabled:t[`gig:${e.id}:addCandidate`]||!d.trim(),children:[(0,R.jsx)(se,{}),`Add candidate`]})]}):null,(0,R.jsx)(kn,{hidden:p.length!==0,children:`No suggested or interested people yet.`}),(0,R.jsx)(`div`,{className:L(`grid gap-3 p-4`,p.length===0&&`hidden`),children:p.map(a=>(0,R.jsx)(Qn,{gigId:e.id,application:a,loading:t,canWrite:n,crmContactUrl:r,crmAttachmentUrl:i,onUpdateApplicationStatus:u},a.id))})]})]})}function Qn({gigId:e,application:t,loading:n,canWrite:r,crmContactUrl:i,crmAttachmentUrl:a,onUpdateApplicationStatus:o}){let s=Xn(t),c=i(t.crm_contact_id),l=a(t.latest_resume_id),u=typeof t.fit_score==`number`?`${Math.round(t.fit_score)}/100`:typeof t.match_score==`number`?t.match_score.toFixed(1):``,d=typeof t.evaluation?.llm_summary==`string`?t.evaluation.llm_summary:``;return(0,R.jsxs)(`div`,{className:`grid gap-2 rounded-md border bg-background p-2`,children:[(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-2`,children:[c?(0,R.jsx)(`a`,{className:`font-extrabold text-primary`,href:c,target:`_blank`,rel:`noopener noreferrer`,children:s}):(0,R.jsx)(`strong`,{children:s}),(0,R.jsx)(z,{variant:t.status===`interested`?`succeeded`:`neutral`,children:Rn(t.status)}),(0,R.jsx)(z,{variant:`neutral`,children:Rn(t.source||`manual_add`)}),u?(0,R.jsxs)(`span`,{className:`text-xs font-bold text-muted-foreground`,children:[`Fit `,u]}):null,c?(0,R.jsx)(`a`,{className:`text-xs font-extrabold text-primary`,href:c,target:`_blank`,rel:`noopener noreferrer`,"aria-label":`Open ${s} CRM profile`,children:`CRM profile`}):null,l?(0,R.jsx)(`a`,{className:`text-xs font-extrabold text-primary`,href:l,target:`_blank`,rel:`noopener noreferrer`,children:`Resume`}):null]}),d?(0,R.jsx)(`div`,{className:`text-xs text-muted-foreground`,children:d}):null,r?(0,R.jsx)(Vt,{"aria-label":`Candidate status for ${s}`,value:t.status||`suggested`,disabled:n[`application:${t.id}:status`],onChange:n=>o(e,t.id,n.target.value),children:Ln.map(e=>(0,R.jsx)(`option`,{value:e,children:Rn(e)},e))}):null]})}function $n(e){let t=pn[e.peopleFilterKind]?.options||[];return(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`People lookup`}),(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center justify-end gap-2`,children:[e.canSync?(0,R.jsxs)(B,{id:`syncPeople`,"data-permission":`people:sync`,type:`button`,onClick:e.onSync,disabled:e.loading.syncPeople,children:[(0,R.jsx)(C,{}),`Sync people`]}):null,e.canSyncNewsletters?(0,R.jsxs)(B,{id:`syncNewsletters`,"data-permission":`people:sync`,type:`button`,variant:`secondary`,onClick:e.onSyncNewsletters,disabled:e.loading.syncNewsletters,children:[(0,R.jsx)(C,{}),`Sync newsletters`]}):null,e.crmBaseUrl?(0,R.jsx)(`a`,{id:`crmHomeLink`,className:`text-sm font-extrabold text-primary`,href:e.crmBaseUrl,target:`_blank`,rel:`noreferrer`,children:`Open CRM`}):null,(0,R.jsx)(`span`,{id:`peopleStatus`,className:`text-sm text-muted-foreground`,children:e.loading.people?`Loading`:`${e.people.length} shown`})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 border-b p-4 md:grid-cols-[minmax(0,1fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`Search CRM people cache`,(0,R.jsx)(U,{id:`peopleQuery`,value:e.peopleQuery,autoComplete:`off`,placeholder:`Name, email, CRM id, Discord, resume`,onChange:t=>e.setPeopleQuery(t.target.value),onKeyDown:t=>{t.key===`Enter`&&e.onSearch()}})]}),(0,R.jsxs)(B,{id:`searchPeople`,type:`button`,onClick:e.onSearch,disabled:e.loading.people,children:[(0,R.jsx)(ne,{}),`Search`]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 border-b bg-background p-4 md:grid-cols-[minmax(120px,.7fr)_minmax(150px,1fr)_minmax(150px,1fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`Member`,(0,R.jsxs)(Vt,{id:`peopleMember`,value:e.peopleMember,onChange:t=>e.setPeopleMember(t.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Any`}),(0,R.jsx)(`option`,{value:`true`,children:`Member`}),(0,R.jsx)(`option`,{value:`false`,children:`Not member`})]})]}),(0,R.jsxs)(W,{children:[`Add filter`,(0,R.jsx)(Vt,{id:`peopleFilterKind`,value:e.peopleFilterKind,disabled:e.peopleFilterKeys.length===0,onChange:t=>e.setPeopleFilterKind(t.target.value),children:e.peopleFilterKeys.map(e=>(0,R.jsx)(`option`,{value:e,children:pn[e].label},e))})]}),(0,R.jsxs)(W,{children:[`Value`,(0,R.jsx)(Vt,{id:`peopleFilterValue`,value:e.peopleFilterValue,onChange:t=>e.setPeopleFilterValue(t.target.value),children:t.map(([e,t])=>(0,R.jsx)(`option`,{value:e,children:t},e))})]}),(0,R.jsx)(B,{id:`addPeopleFilter`,type:`button`,onClick:e.addFilter,disabled:e.peopleFilterKeys.length===0,children:`Add filter`}),(0,R.jsx)(`div`,{id:`activePeopleFilters`,className:`md:col-span-4`,children:(0,R.jsx)(Fn,{filters:e.peopleFilters,onRemove:e.removeFilter})})]}),(0,R.jsx)(kn,{hidden:e.people.length!==0,children:`No people match this lookup.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`peopleTable`,className:L(`min-w-[900px]`,e.people.length===0&&`hidden`),"aria-label":`People lookup results`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(Dn,{className:`w-[27%]`,label:`Name`,scope:`people`,sort:e.sort,sortKey:`name`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[28%]`,label:`Status`,scope:`people`,sort:e.sort,sortKey:`status`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[20%]`,label:`Discord`,scope:`people`,sort:e.sort,sortKey:`discord`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[25%]`,label:`Resume / skills`,scope:`people`,sort:e.sort,sortKey:`resume`,onSort:(t,n)=>e.onSort(n)})]})}),(0,R.jsx)(Wt,{id:`peopleBody`,children:e.people.map(t=>{let n=t.name||t.email_508||t.email||`CRM contact`,r=e.crmContactUrl(t.crm_contact_id),i=t.profile_status||{},a=Number(i.skills_count||0),o=e.crmAttachmentUrl(t.latest_resume_id);return(0,R.jsxs)(Gt,{children:[(0,R.jsxs)(K,{children:[r?(0,R.jsx)(`a`,{className:`font-extrabold text-primary`,href:r,target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${n} in CRM`,children:n}):(0,R.jsx)(`strong`,{children:n}),(0,R.jsx)(`div`,{className:`text-sm text-muted-foreground`,children:[t.email_508||t.email,t.contact_type].filter(Boolean).join(` | `)})]}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-1.5`,children:[i.crm_active?null:(0,R.jsx)(z,{variant:`missing`,children:t.sync_status||`CRM sync issue`}),(0,R.jsx)(z,{variant:i.is_member?`succeeded`:`missing`,children:i.is_member?`Member`:`Missing Member`}),(0,R.jsx)(z,{variant:i.discord_linked?`succeeded`:`missing`,children:i.discord_linked?`Discord`:`Missing Discord`}),(0,R.jsx)(z,{variant:i.email_508?`succeeded`:`missing`,children:i.email_508?`508 email`:`Missing 508 email`}),i.latest_resume?null:(0,R.jsx)(z,{variant:`missing`,children:`Missing Resume`})]})}),(0,R.jsx)(K,{children:[t.discord_username,t.discord_user_id].filter(Boolean).join(` | `)||`Not linked`}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-1.5`,children:[o?(0,R.jsx)(`a`,{className:`inline-flex min-h-7 items-center rounded-md border bg-secondary px-2 text-xs font-extrabold`,href:o,target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${n} resume`,children:`Resume`}):(0,R.jsx)(`span`,{children:t.latest_resume_name||t.latest_resume_id||`No resume`}),(0,R.jsx)(z,{variant:a>0?`succeeded`:`missing`,children:a>0?`Skills parsed`:`Skills not parsed`})]})})]},t.crm_contact_id||n)})})]})})]})}function er(e){let t=pn[e.onboardingFilterKind]?.options||[];return(0,R.jsxs)(R.Fragment,{children:[e.canWrite?(0,R.jsx)(or,{loading:e.loading.engineerSetup,onSetup:e.onSetupEngineer}):null,(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Onboarding queue`}),(0,R.jsx)(`span`,{id:`onboardingStatus`,className:`text-sm text-muted-foreground`,children:e.loading.onboarding?`Loading`:`${e.people.length} shown`})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 border-b p-4 md:grid-cols-[minmax(0,1fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`Search prospects`,(0,R.jsx)(U,{id:`onboardingQuery`,value:e.onboardingQuery,autoComplete:`off`,placeholder:`Name, email, Discord, onboarder`,onChange:t=>e.setOnboardingQuery(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onSearch()})]}),(0,R.jsxs)(B,{id:`searchOnboarding`,type:`button`,onClick:e.onSearch,disabled:e.loading.onboarding,children:[(0,R.jsx)(ne,{}),`Search`]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 border-b bg-background p-4 md:grid-cols-[minmax(140px,.8fr)_minmax(150px,1fr)_minmax(150px,1fr)_minmax(120px,.7fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{id:`onboardingState`,value:e.onboardingState,onChange:t=>e.setOnboardingState(t.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Any state`}),hn.map(([e,t])=>(0,R.jsx)(`option`,{value:e,children:t},e))]})]}),(0,R.jsxs)(W,{children:[`Onboarder`,(0,R.jsx)(U,{id:`onboarderFilter`,value:e.onboarderFilter,autoComplete:`off`,placeholder:`Any onboarder`,onChange:t=>e.setOnboarderFilter(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onSearch()})]}),(0,R.jsxs)(W,{children:[`Add filter`,(0,R.jsx)(Vt,{id:`onboardingFilterKind`,value:e.onboardingFilterKind,disabled:e.onboardingFilterKeys.length===0,onChange:t=>e.setOnboardingFilterKind(t.target.value),children:e.onboardingFilterKeys.map(e=>(0,R.jsx)(`option`,{value:e,children:pn[e].label},e))})]}),(0,R.jsxs)(W,{children:[`Value`,(0,R.jsx)(Vt,{id:`onboardingFilterValue`,value:e.onboardingFilterValue,onChange:t=>e.setOnboardingFilterValue(t.target.value),children:t.map(([e,t])=>(0,R.jsx)(`option`,{value:e,children:t},e))})]}),(0,R.jsx)(B,{id:`addOnboardingFilter`,type:`button`,onClick:e.addFilter,disabled:e.onboardingFilterKeys.length===0,children:`Add filter`}),(0,R.jsx)(`div`,{id:`activeOnboardingFilters`,className:`md:col-span-5`,children:(0,R.jsx)(Fn,{filters:e.onboardingFilters,onRemove:e.removeFilter,suffix:`onboarding filter`})})]}),(0,R.jsx)(kn,{hidden:e.people.length!==0,children:`No prospects match this queue view.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`onboardingTable`,className:L(`min-w-[1340px]`,e.people.length===0&&`hidden`),"aria-label":`Onboarding queue`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(Dn,{className:`w-[18%]`,label:`Name`,scope:`onboarding`,sort:e.sort,sortKey:`name`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[12%]`,label:`Status`,scope:`onboarding`,sort:e.sort,sortKey:`onboarding_state`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[20%]`,label:`Onboarder`,scope:`onboarding`,sort:e.sort,sortKey:`onboarder`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[12%]`,label:`Updated`,scope:`onboarding`,sort:e.sort,sortKey:`updated`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(G,{className:`w-[15%]`,children:`Email`}),(0,R.jsx)(G,{className:`w-[12%]`,children:`Links`}),(0,R.jsx)(Dn,{className:`w-[11%]`,label:`Needs`,scope:`onboarding`,sort:e.sort,sortKey:`profile_gaps`,onSort:(t,n)=>e.onSort(n)})]})}),(0,R.jsx)(Wt,{id:`onboardingBody`,children:e.people.map(t=>(0,R.jsx)(sr,{person:t,loading:e.loading,canWrite:e.canWrite,onAssign:e.onAssign,onStatusChange:e.onStatusChange,onDraftEmail:e.onDraftEmail,onSendEmail:e.onSendEmail,crmContactUrl:e.crmContactUrl,crmAttachmentUrl:e.crmAttachmentUrl,canConfigure:e.canConfigure,onOpenConfiguration:e.onOpenConfiguration},t.crm_contact_id||t.name))})]})})]})]})}var tr=[`Female`,`Genderqueer`,`Male`,`Non-Conforming`,`Other`,`Prefer not to say`,`Transgender`],nr=[`Company Email`,`Personal Email`,`User ID`];function rr(e){let t=(e||``).trim().split(/\s+/).filter(Boolean);return t.length===0?{first:``,middle:``,last:``}:t.length===1?{first:t[0],middle:``,last:``}:t.length===2?{first:t[0],middle:``,last:t[1]}:{first:t[0],middle:t.slice(1,-1).join(` `),last:t[t.length-1]}}function ir(e){let t=(e.email||``).trim();return!t||t.toLowerCase().endsWith(`@508.dev`)?``:t}function ar(e){let t=(e.email_508||``).trim();if(t)return t;let n=(e.email||``).trim();return n.toLowerCase().endsWith(`@508.dev`)?n:``}function or({loading:e,onSetup:t}){let[n,r]=(0,l.useState)(``),[i,a]=(0,l.useState)([]),[o,s]=(0,l.useState)(!1),[c,u]=(0,l.useState)(``),[d,f]=(0,l.useState)(``),[p,m]=(0,l.useState)(``),[h,g]=(0,l.useState)(``),[_,v]=(0,l.useState)(``),[y,b]=(0,l.useState)(``),[x,ee]=(0,l.useState)(``),[te,S]=(0,l.useState)(``),[C,re]=(0,l.useState)(``),[ie,ae]=(0,l.useState)(``),[oe,ce]=(0,l.useState)(``);function w(e){let t=rr(e.name);m(t.first),g(t.middle),v(t.last),f(ar(e)),ae(ir(e)),b(e.address_country||``),r(e.name||e.email_508||e.email||``),a([]),u(``)}async function T(){let e=n.trim();if(e){s(!0),u(``);try{a(await q(`/dashboard/api/people?${new URLSearchParams({limit:`8`,query:e}).toString()}`))}catch(e){u(bn(e,`Unable to search people`)),a([])}finally{s(!1)}}}async function le(){let e={email:d,first_name:p,middle_name:h,last_name:_,country:y,personal_email:ie};x.trim()&&(e.gender=x),te.trim()&&(e.date_of_birth=te),C.trim()&&(e.date_of_joining=C),oe.trim()&&(e.prefered_email=oe),await t(e)&&(r(``),a([]),f(``),m(``),g(``),v(``),b(``),ee(``),S(``),re(``),ae(``),ce(``))}return(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Engineer setup`})}),(0,R.jsx)(Bt,{children:(0,R.jsxs)(`form`,{className:`grid gap-3`,onSubmit:e=>{e.preventDefault(),le()},children:[(0,R.jsxs)(`div`,{className:`grid gap-3 border-b pb-3 md:grid-cols-[minmax(0,1fr)_auto]`,children:[(0,R.jsxs)(W,{children:[`CRM person`,(0,R.jsx)(U,{value:n,autoComplete:`off`,placeholder:`Search name or email`,onChange:e=>r(e.target.value),onKeyDown:e=>{e.key===`Enter`&&(e.preventDefault(),T())}})]}),(0,R.jsxs)(B,{type:`button`,onClick:T,disabled:o||!n.trim(),children:[(0,R.jsx)(ne,{}),`Search`]}),c?(0,R.jsx)(`span`,{className:`text-sm font-semibold text-destructive`,children:c}):null,i.length>0?(0,R.jsx)(`div`,{className:`grid gap-2 md:col-span-2`,children:i.map(e=>{let t=e.name||e.email_508||e.email||e.crm_contact_id,n=[e.email_508||e.email,e.contact_type].filter(Boolean).join(` | `);return(0,R.jsxs)(`button`,{type:`button`,className:`grid rounded-md border bg-background px-3 py-2 text-left text-sm hover:border-primary`,onClick:()=>w(e),children:[(0,R.jsx)(`strong`,{children:t}),n?(0,R.jsx)(`span`,{className:`text-muted-foreground`,children:n}):null]},e.crm_contact_id||t)})}):null]}),(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-[minmax(0,1fr)_minmax(0,1fr)_minmax(130px,.6fr)]`,children:[(0,R.jsxs)(W,{children:[`Company email`,(0,R.jsx)(U,{value:d,autoComplete:`off`,placeholder:`engineer@508.dev`,onChange:e=>f(e.target.value)})]}),(0,R.jsxs)(W,{children:[`First name`,(0,R.jsx)(U,{value:p,autoComplete:`off`,placeholder:`First`,onChange:e=>m(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Middle name`,(0,R.jsx)(U,{value:h,autoComplete:`off`,placeholder:`Optional`,onChange:e=>g(e.target.value)})]})]}),(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-[minmax(0,1fr)_minmax(130px,.6fr)]`,children:[(0,R.jsxs)(W,{children:[`Last name`,(0,R.jsx)(U,{value:_,autoComplete:`off`,placeholder:`Last`,onChange:e=>v(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Country`,(0,R.jsx)(U,{value:y,autoComplete:`off`,placeholder:`Taiwan`,onChange:e=>b(e.target.value)})]})]}),(0,R.jsxs)(`details`,{className:`rounded-md border bg-background p-3`,children:[(0,R.jsx)(`summary`,{className:`cursor-pointer text-sm font-extrabold`,children:`Advanced options`}),(0,R.jsxs)(`div`,{className:`mt-3 grid gap-3 md:grid-cols-2`,children:[(0,R.jsxs)(W,{children:[`Gender`,(0,R.jsxs)(Vt,{value:x,onChange:e=>ee(e.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Default`}),tr.map(e=>(0,R.jsx)(`option`,{value:e,children:e},e))]})]}),(0,R.jsxs)(W,{children:[`Date of birth`,(0,R.jsx)(U,{value:te,type:`date`,autoComplete:`off`,onChange:e=>S(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Date of joining`,(0,R.jsx)(U,{value:C,type:`date`,autoComplete:`off`,onChange:e=>re(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Personal email`,(0,R.jsx)(U,{value:ie,type:`email`,autoComplete:`off`,placeholder:`Optional`,onChange:e=>ae(e.target.value)})]}),(0,R.jsxs)(W,{children:[`Preferred contact email`,(0,R.jsxs)(Vt,{value:oe,onChange:e=>ce(e.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Default`}),nr.map(e=>(0,R.jsx)(`option`,{value:e,children:e},e))]})]})]})]}),(0,R.jsx)(`div`,{className:`flex flex-wrap items-center justify-between gap-3`,children:(0,R.jsxs)(B,{id:`setupEngineer`,type:`submit`,disabled:e||!d.trim()||!p.trim(),children:[(0,R.jsx)(se,{}),`Set up engineer`]})})]})})]})}function sr({person:e,loading:t,canWrite:n,onAssign:r,onStatusChange:i,onDraftEmail:a,onSendEmail:o,crmContactUrl:s,crmAttachmentUrl:c,canConfigure:u,onOpenConfiguration:d}){let f=e.name||e.email_508||e.email||`CRM contact`,[p,m]=(0,l.useState)($t(e.onboarder)),[h,g]=(0,l.useState)(!1),[_,v]=(0,l.useState)(null),[y,b]=(0,l.useState)(null),[x,ee]=(0,l.useState)({has_contributed:_n(Xt(e))===`onboarded`,discord_joined:e.discord_user_id?`yes`:`unknown`,agreement_signed:`unknown`});(0,l.useEffect)(()=>m($t(e.onboarder)),[e.onboarder]);let S=_n(Xt(e)),ne=e.profile_status||{},ae=[[`Discord`,ne.discord_linked],[`Resume`,ne.latest_resume],[`Skills`,Number(ne.skills_count||0)>0]].filter(([,e])=>!e),oe=s(e.crm_contact_id),se=c(e.latest_resume_id),ce=_?.onboarding_email_sent_at||e.onboarding_email_sent_at,w=_?.onboarding_email_sent_by||e.onboarding_email_sent_by,T=_?.onboarding_email_recipient||e.onboarding_email_recipient,le=!_||y!==null&&y.has_contributed===x.has_contributed&&y.discord_joined===x.discord_joined&&y.agreement_signed===x.agreement_signed,E=_&&!_.onboarding_email_sent_at?le?_.can_send?``:_.recipient_email?_.reply_to_email?`Send disabled: onboarding email SMTP is not configured.`:`Send disabled: your Reply-To email is missing.`:`Send disabled: candidate email is missing.`:`Send disabled: regenerate after changing draft options.`:``,ue=E.includes(`SMTP`),de=!!t[`onboarding-email-draft:${e.crm_contact_id}`],fe=!!t[`onboarding-email-send:${e.crm_contact_id}`],pe=_?.markdown_body||``;async function D(t=x){let n=await a(e.crm_contact_id,t);n&&(v(n),b({...t}),g(!0))}async function O(){if(!_||!y||!le)return;let t=await o(e.crm_contact_id,y,_.markdown_body);t&&v(t)}return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(Gt,{children:[(0,R.jsxs)(K,{children:[oe?(0,R.jsx)(`a`,{className:`font-extrabold text-primary`,href:oe,target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${f} in CRM`,children:f}):(0,R.jsx)(`strong`,{children:f}),(0,R.jsx)(`div`,{className:`text-sm text-muted-foreground`,children:e.email_508||e.email||``})]}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid max-w-56 gap-2`,children:[(0,R.jsx)(z,{variant:Qt(Xt(e)),children:e.onboarding_status_label||Zt(Xt(e))}),n?(0,R.jsxs)(Vt,{"aria-label":`Onboarding status for ${f}`,value:S,disabled:t[`onboarding-status:${e.crm_contact_id}`],onChange:t=>i(e.crm_contact_id,t.target.value),children:[S?null:(0,R.jsx)(`option`,{value:``,disabled:!0,children:`No status`}),mn.map(([e,t])=>(0,R.jsx)(`option`,{value:e,children:t},e))]}):null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`form`,{className:`grid max-w-64 grid-cols-[minmax(100px,1fr)_auto] items-center gap-2`,onSubmit:t=>{t.preventDefault(),r(e.crm_contact_id,p)},children:[(0,R.jsx)(U,{"aria-label":`Onboarder for ${f}`,value:p,placeholder:`508 username`,onChange:e=>m(e.target.value)}),(0,R.jsx)(B,{type:`submit`,size:`sm`,"aria-label":`Save onboarder for ${f}`,disabled:t[`onboarder:${e.crm_contact_id}`],children:`Save`})]})}),(0,R.jsx)(K,{children:qt(e.onboarding_updated_at)}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-2`,children:[ce?(0,R.jsxs)(z,{variant:`succeeded`,children:[`Sent `,qt(ce)]}):(0,R.jsx)(z,{variant:`neutral`,children:`Not sent`}),T?(0,R.jsx)(`span`,{className:`text-xs text-muted-foreground`,children:T}):null,w?(0,R.jsxs)(`span`,{className:`text-xs text-muted-foreground`,children:[`By `,w]}):null,n?(0,R.jsxs)(B,{type:`button`,size:`sm`,variant:h?`outline`:`secondary`,onClick:()=>{if(h){g(!1);return}g(!0),_||D()},disabled:de,children:[(0,R.jsx)(te,{}),_?`Edit draft`:`Draft email`]}):null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-1.5`,children:[se?(0,R.jsx)(`a`,{className:`inline-flex min-h-7 items-center rounded-md border bg-secondary px-2 text-xs font-extrabold`,href:se,target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${f} resume`,children:`Resume`}):null,an(e.linkedin)?(0,R.jsx)(`a`,{className:`inline-flex min-h-7 items-center rounded-md border bg-secondary px-2 text-xs font-extrabold`,href:an(e.linkedin),target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${f} LinkedIn`,children:`LinkedIn`}):null,on(e.github_username)?(0,R.jsx)(`a`,{className:`inline-flex min-h-7 items-center rounded-md border bg-secondary px-2 text-xs font-extrabold`,href:on(e.github_username),target:`_blank`,rel:`noreferrer`,"aria-label":`Open ${f} GitHub`,children:e.github_username||`GitHub`}):null,!se&&!an(e.linkedin)&&!on(e.github_username)?`None`:null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap gap-1.5`,children:[ae.map(([e])=>(0,R.jsxs)(z,{variant:`missing`,children:[`Missing `,e]},String(e))),ae.length===0?`None`:null]})})]}),h?(0,R.jsx)(Gt,{children:(0,R.jsx)(K,{colSpan:7,className:`bg-secondary/30`,children:(0,R.jsxs)(`div`,{className:`grid gap-3 rounded-md border bg-background p-4`,children:[(0,R.jsxs)(`div`,{className:`grid gap-3 md:grid-cols-[auto_minmax(150px,220px)_minmax(150px,220px)_auto] md:items-end`,children:[(0,R.jsxs)(`label`,{className:`flex min-h-9 items-center gap-2 text-sm font-semibold`,children:[(0,R.jsx)(`input`,{type:`checkbox`,checked:x.has_contributed,onChange:e=>{ee({...x,has_contributed:e.target.checked})}}),`Contribution done`]}),(0,R.jsxs)(W,{children:[`Discord`,(0,R.jsxs)(Vt,{value:x.discord_joined,onChange:e=>ee({...x,discord_joined:e.target.value}),children:[(0,R.jsx)(`option`,{value:`unknown`,children:`Unknown`}),(0,R.jsx)(`option`,{value:`yes`,children:`Joined`}),(0,R.jsx)(`option`,{value:`no`,children:`Not joined`})]})]}),(0,R.jsxs)(W,{children:[`Agreement`,(0,R.jsxs)(Vt,{value:x.agreement_signed,onChange:e=>ee({...x,agreement_signed:e.target.value}),children:[(0,R.jsx)(`option`,{value:`unknown`,children:`Unknown`}),(0,R.jsx)(`option`,{value:`yes`,children:`Signed`}),(0,R.jsx)(`option`,{value:`no`,children:`Not signed`})]})]}),(0,R.jsxs)(B,{type:`button`,variant:`outline`,onClick:()=>D(),disabled:de,children:[(0,R.jsx)(C,{}),`Regenerate`]})]}),_?(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(`div`,{className:`grid gap-2 text-sm md:grid-cols-3`,children:[(0,R.jsxs)(`span`,{children:[(0,R.jsx)(`strong`,{children:`To:`}),` `,_.recipient_email||`Missing`]}),(0,R.jsxs)(`span`,{children:[(0,R.jsx)(`strong`,{children:`Reply-To:`}),` `,_.reply_to_email||`Missing`]}),(0,R.jsxs)(`span`,{children:[(0,R.jsx)(`strong`,{children:`From:`}),` `,_.sender_display_name||`onboarding`]})]}),(0,R.jsxs)(W,{children:[`Subject`,(0,R.jsx)(U,{value:_.subject,readOnly:!0})]}),(0,R.jsxs)(W,{children:[`Draft`,(0,R.jsx)(`textarea`,{value:pe,className:`min-h-64 w-full rounded-md border border-input bg-background px-3 py-2 font-mono text-sm text-foreground shadow-xs transition-colors placeholder:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]`,onChange:e=>v({..._,markdown_body:e.target.value})})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap items-center gap-2`,children:[(0,R.jsxs)(B,{type:`button`,variant:`default`,onClick:O,disabled:fe||!_.can_send||!le||!pe.trim(),title:E||void 0,children:[(0,R.jsx)(re,{}),fe?`Sending`:`Send`]}),E?(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[E,ue&&u?(0,R.jsxs)(B,{type:`button`,size:`sm`,variant:`ghost`,className:`ml-2`,onClick:d,children:[(0,R.jsx)(ie,{}),`Configure`]}):null]}):null,_.marker_status===`error`?(0,R.jsxs)(`span`,{className:`text-sm text-muted-foreground`,children:[`Marker not saved: `,_.marker_error||`unknown`]}):null]})]}):(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:de?`Drafting email`:`No draft loaded`})]})})}):null]})}function cr(e){return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(V,{className:`grid gap-3 p-4 md:grid-cols-4 md:items-end`,children:[(0,R.jsxs)(W,{children:[`Window`,(0,R.jsxs)(Vt,{id:`minutes`,value:e.minutes,onChange:t=>e.setMinutes(t.target.value),children:[(0,R.jsx)(`option`,{value:`15`,children:`15 minutes`}),(0,R.jsx)(`option`,{value:`60`,children:`1 hour`}),(0,R.jsx)(`option`,{value:`360`,children:`6 hours`}),(0,R.jsx)(`option`,{value:`1440`,children:`24 hours`})]})]}),(0,R.jsxs)(W,{children:[`Status`,(0,R.jsxs)(Vt,{id:`status`,value:e.status,onChange:t=>e.setStatus(t.target.value),children:[(0,R.jsx)(`option`,{value:``,children:`Any status`}),(0,R.jsx)(`option`,{value:`queued`,children:`Queued`}),(0,R.jsx)(`option`,{value:`running`,children:`Running`}),(0,R.jsx)(`option`,{value:`succeeded`,children:`Succeeded`}),(0,R.jsx)(`option`,{value:`failed`,children:`Failed`}),(0,R.jsx)(`option`,{value:`dead`,children:`Dead`}),(0,R.jsx)(`option`,{value:`canceled`,children:`Canceled`})]})]}),(0,R.jsxs)(W,{children:[`Type`,(0,R.jsx)(U,{id:`jobType`,value:e.jobType,autoComplete:`off`,placeholder:`Any type`,onChange:t=>e.setJobType(t.target.value),onKeyDown:t=>t.key===`Enter`&&e.onSearch()})]}),(0,R.jsxs)(B,{id:`refreshJobs`,type:`button`,onClick:e.onSearch,disabled:e.loading.jobs,children:[(0,R.jsx)(C,{}),`Refresh background tasks`]})]}),(0,R.jsxs)(`section`,{className:`grid gap-3 md:grid-cols-4`,"aria-label":`Background task summary`,children:[(0,R.jsx)(On,{id:`metricTotal`,label:`Total`,value:e.jobs.length}),(0,R.jsx)(On,{id:`metricQueued`,label:`Queued`,value:e.jobCounts.queued||0}),(0,R.jsx)(On,{id:`metricRunning`,label:`Running`,value:e.jobCounts.running||0}),(0,R.jsx)(On,{id:`metricFailed`,label:`Failed`,value:(e.jobCounts.failed||0)+(e.jobCounts.dead||0)})]}),(0,R.jsxs)(V,{children:[(0,R.jsx)(H,{children:(0,R.jsx)(zt,{children:`Recent background tasks`})}),(0,R.jsx)(kn,{hidden:e.jobs.length!==0,children:`No background tasks match these filters.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`jobsTable`,className:L(`min-w-[980px]`,e.jobs.length===0&&`hidden`),"aria-label":`Recent background tasks`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(Dn,{className:`w-[22%]`,label:`Task id`,scope:`jobs`,sort:e.sort,sortKey:`job_id`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[24%]`,label:`Task type`,scope:`jobs`,sort:e.sort,sortKey:`type`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[12%]`,label:`Status`,scope:`jobs`,sort:e.sort,sortKey:`status`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[12%]`,label:`Attempts`,scope:`jobs`,sort:e.sort,sortKey:`attempts`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[18%]`,label:`Updated`,scope:`jobs`,sort:e.sort,sortKey:`updated_at`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(G,{children:`Actions`})]})}),(0,R.jsx)(Wt,{id:`jobsBody`,children:e.jobs.map(t=>(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{className:`font-mono`,children:t.job_id}),(0,R.jsx)(K,{children:t.type}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:t.status||`neutral`,children:t.status})}),(0,R.jsxs)(K,{children:[t.attempts,`/`,t.max_attempts]}),(0,R.jsx)(K,{children:qt(t.updated_at)}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-2`,children:[(0,R.jsx)(B,{type:`button`,size:`sm`,variant:`outline`,"aria-label":`View details for ${t.type} task ${t.job_id}`,onClick:()=>e.onDetail(t.job_id),disabled:e.loading[`detail:${t.job_id}`],children:`Details`}),e.canWrite?(0,R.jsx)(B,{type:`button`,size:`sm`,"aria-label":`Rerun ${t.type} task ${t.job_id}`,onClick:()=>e.onRerun(t.job_id),disabled:e.loading[`rerun:${t.job_id}`],children:`Rerun`}):null]})})]},t.job_id))})]})})]}),e.jobDetail?(0,R.jsxs)(V,{id:`jobDetailPanel`,children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Task detail`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.jobDetail.job_id})]}),(0,R.jsxs)(Bt,{className:`grid gap-4`,children:[(0,R.jsx)(`div`,{className:`grid gap-3 md:grid-cols-2`,children:[[`Task type`,e.jobDetail.type],[`Status`,e.jobDetail.status],[`Attempts`,`${e.jobDetail.attempts}/${e.jobDetail.max_attempts}`],[`Updated`,qt(e.jobDetail.updated_at)],[`Created`,qt(e.jobDetail.created_at)],[`Run after`,qt(e.jobDetail.run_after)],[`Locked by`,e.jobDetail.locked_by||`None`],[`Last error`,e.jobDetail.last_error||`None`]].map(([e,t])=>(0,R.jsxs)(`div`,{className:`grid gap-1 rounded-md border bg-background p-3`,children:[(0,R.jsx)(`span`,{className:`text-[11px] font-extrabold uppercase text-muted-foreground`,children:e}),(0,R.jsx)(`strong`,{className:`break-words text-sm`,children:t})]},e))}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`h2`,{className:`mb-2 text-[15px] font-bold`,children:`Payload`}),(0,R.jsx)(`pre`,{className:`max-h-64 overflow-auto whitespace-pre-wrap break-words rounded-md border bg-background p-3 font-mono text-xs`,children:Yt(e.jobDetail.payload)||`No payload`})]}),(0,R.jsxs)(`div`,{children:[(0,R.jsx)(`h2`,{className:`mb-2 text-[15px] font-bold`,children:`Result`}),(0,R.jsx)(`pre`,{className:`max-h-64 overflow-auto whitespace-pre-wrap break-words rounded-md border bg-background p-3 font-mono text-xs`,children:Yt(e.jobDetail.result)||`No result`})]})]})]}):null]})}function lr(e){return(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Recent audit`}),(0,R.jsxs)(B,{id:`refreshAudit`,type:`button`,variant:`outline`,onClick:e.onRefresh,disabled:e.loading.audit,children:[(0,R.jsx)(C,{}),`Refresh`]})]}),(0,R.jsx)(kn,{hidden:e.events.length!==0,children:`No audit events found.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`auditTable`,className:L(`min-w-[760px]`,e.events.length===0&&`hidden`),"aria-label":`Recent audit events`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(Dn,{className:`w-[24%]`,label:`Time`,scope:`audit`,sort:e.sort,sortKey:`occurred_at`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[28%]`,label:`Actor`,scope:`audit`,sort:e.sort,sortKey:`actor`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[28%]`,label:`Action`,scope:`audit`,sort:e.sort,sortKey:`action`,onSort:(t,n)=>e.onSort(n)}),(0,R.jsx)(Dn,{className:`w-[20%]`,label:`Result`,scope:`audit`,sort:e.sort,sortKey:`result`,onSort:(t,n)=>e.onSort(n)})]})}),(0,R.jsx)(Wt,{id:`auditBody`,children:e.events.map(e=>(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:qt(e.occurred_at)}),(0,R.jsx)(K,{children:e.actor_display_name||e.actor_subject||e.actor_provider}),(0,R.jsx)(K,{children:e.action}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:e.result===`success`?`succeeded`:`failed`,children:e.result})})]},e.id||`${e.occurred_at||``}-${e.actor_subject||``}-${e.action||``}`))})]})})]})}function ur({report:e,loading:t,onRefresh:n}){let r=e?.summary||{},i=[[`Status`,e?.status_counts||{}],[`Intent`,e?.intent_counts||{}],[`Planner`,e?.planner_counts||{}]].flatMap(([e,t])=>Object.entries(t).map(([t,n])=>({label:e,value:t,count:n}))).sort((e,t)=>t.count-e.count||e.label.localeCompare(t.label)),a=Array.isArray(e?.recent_unsupported)?e.recent_unsupported:[];return(0,R.jsxs)(R.Fragment,{children:[(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Agent requests`}),(0,R.jsxs)(B,{id:`refreshAgent`,type:`button`,variant:`outline`,onClick:n,disabled:t.agent,children:[(0,R.jsx)(C,{}),`Refresh`]})]}),(0,R.jsxs)(Bt,{className:`grid gap-3 md:grid-cols-5`,children:[(0,R.jsx)(On,{id:`agentMetricTotal`,label:`Total`,value:r.total||0}),(0,R.jsx)(On,{id:`agentMetricHandled`,label:`Handled`,value:r.handled||0}),(0,R.jsx)(On,{id:`agentMetricConfirmations`,label:`Confirmations`,value:r.requires_confirmation||0}),(0,R.jsx)(On,{id:`agentMetricClarifications`,label:`Clarifications`,value:r.needs_clarification||0}),(0,R.jsx)(On,{id:`agentMetricUnsupported`,label:`Not understood`,value:r.unsupported||0})]})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Request mix`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:`Recent agent.request audit events.`})]}),(0,R.jsx)(kn,{hidden:i.length!==0,children:`No agent request data found.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`agentBreakdownTable`,className:L(`min-w-[860px]`,i.length===0&&`hidden`),"aria-label":`Agent request breakdown`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{children:`Dimension`}),(0,R.jsx)(G,{children:`Value`}),(0,R.jsx)(G,{children:`Count`})]})}),(0,R.jsx)(Wt,{id:`agentBreakdownBody`,children:i.map(e=>(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:e.label}),(0,R.jsx)(K,{children:e.value}),(0,R.jsx)(K,{children:e.count})]},`${e.label}-${e.value}`))})]})})]}),(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Not understood`}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:`Sanitized request text only.`})]}),(0,R.jsx)(kn,{hidden:a.length!==0,children:`No unsupported agent requests found.`}),(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`agentUnsupportedTable`,className:L(`min-w-[860px]`,a.length===0&&`hidden`),"aria-label":`Unsupported agent requests`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{children:`Time`}),(0,R.jsx)(G,{children:`Actor`}),(0,R.jsx)(G,{children:`Message`}),(0,R.jsx)(G,{children:`Result`})]})}),(0,R.jsx)(Wt,{id:`agentUnsupportedBody`,children:a.map(e=>(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:qt(e.occurred_at)}),(0,R.jsx)(K,{children:e.actor}),(0,R.jsx)(K,{children:e.message_sanitized}),(0,R.jsx)(K,{children:(0,R.jsx)(z,{variant:e.result===`success`?`succeeded`:`failed`,children:e.result||`unknown`})})]},`${e.occurred_at||``}-${e.actor||``}-${e.message_sanitized||``}`))})]})})]})]})}function dr({items:e,loading:t,canWrite:n,onRefresh:r,onSave:i,onClear:a,focusCategory:o,focusNonce:s}){let[c,u]=(0,l.useState)(`All`),[d,f]=(0,l.useState)(``),[p,m]=(0,l.useState)({}),h=(0,l.useMemo)(()=>{let t=new Set(e.map(e=>e.category)),n=sn.filter(e=>t.has(e.category)),r=Array.from(t).filter(e=>!cn.has(e)).sort().map(e=>({category:e,label:e,description:`Additional runtime settings.`}));return n.concat(r)},[e]),g=(0,l.useMemo)(()=>{let t=new Map;for(let n of e){if(c!==`All`&&n.category!==c)continue;let e=t.get(n.category)??[];e.push(n),t.set(n.category,e)}return Array.from(t.entries()).map(([e,t])=>{let n=cn.get(e),r=t.sort((e,t)=>Number(!un(e))-Number(!un(t))||e.label.localeCompare(t.label)),i=r.filter(un),a=r.filter(e=>!un(e));return{category:e,label:n?.label??e,description:n?.description??`Additional runtime settings.`,order:n?.index??sn.length,primaryItems:i.length?i:r,advancedItems:i.length?a:[],items:r}}).sort((e,t)=>e.order-t.order||e.label.localeCompare(t.label))},[e,c]),_=(0,l.useMemo)(()=>({configured:e.filter(e=>e.configured).length,envLocked:e.filter(e=>e.env_locked).length,missing:e.filter(e=>!e.configured).length}),[e]),v=g.reduce((e,t)=>e+t.items.length,0);(0,l.useEffect)(()=>{m(Object.fromEntries(e.map(e=>[e.key,e.is_secret?``:String(e.value??``)])))},[e]),(0,l.useEffect)(()=>{c!==`All`&&!e.some(e=>e.category===c)&&u(`All`)},[e,c]),(0,l.useEffect)(()=>{if(!o||!e.some(e=>e.category===o))return;u(o),f(o);let t=window.requestAnimationFrame?.(()=>{document.getElementById(ln(o))?.scrollIntoView({block:`start`,behavior:`smooth`})}),n=window.setTimeout(()=>f(``),4e3);return()=>{t!==void 0&&window.cancelAnimationFrame?.(t),window.clearTimeout(n)}},[o,s,e]);function y(e){return e.source===`env`?`ENV`:e.source===`database`?`DB`:`Default`}function b(e){let r=p[e.key]??``,i=!n||e.env_locked||t[`configuration:${e.key}`];return e.value_type===`bool`?(0,R.jsxs)(Vt,{"aria-label":`${e.label} value`,value:r,disabled:i,onChange:t=>m(n=>({...n,[e.key]:t.target.value})),children:[(0,R.jsx)(`option`,{value:``,children:`Default`}),(0,R.jsx)(`option`,{value:`true`,children:`True`}),(0,R.jsx)(`option`,{value:`false`,children:`False`})]}):(0,R.jsx)(U,{"aria-label":`${e.label} value`,value:r,type:e.is_secret?`password`:e.value_type===`int`?`number`:`text`,inputMode:e.value_type===`int`||e.value_type===`float`?`numeric`:`text`,placeholder:e.is_secret?`Set new value`:``,autoComplete:`off`,disabled:i,onChange:t=>m(n=>({...n,[e.key]:t.target.value}))})}function x(e,t,n){return(0,R.jsx)(`div`,{className:`overflow-x-auto`,children:(0,R.jsxs)(Ht,{id:`configurationTable-${n}`,className:`min-w-[980px]`,"aria-label":`${e} configuration settings`,children:[(0,R.jsx)(Ut,{children:(0,R.jsxs)(Gt,{children:[(0,R.jsx)(G,{className:`w-[26%]`,children:`Setting`}),(0,R.jsx)(G,{className:`w-[12%]`,children:`Source`}),(0,R.jsx)(G,{className:`w-[18%]`,children:`Active`}),(0,R.jsx)(G,{className:`w-[29%]`,children:`Value`}),(0,R.jsx)(G,{className:`w-[15%]`,children:`Actions`})]})}),(0,R.jsx)(Wt,{id:`configurationBody-${n}`,children:t.map(e=>ee(e))})]})})}function ee(e){let r=t[`configuration:${e.key}`],o=n&&!e.env_locked&&!r,s=p[e.key]??``,c=!e.is_secret&&!s.trim();return(0,R.jsxs)(Gt,{children:[(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-1`,children:[(0,R.jsx)(`strong`,{children:e.label}),(0,R.jsx)(`span`,{className:`font-mono text-xs text-muted-foreground`,children:e.key}),(0,R.jsx)(`span`,{className:`text-xs text-muted-foreground`,children:e.description}),e.restart_required?(0,R.jsx)(`div`,{children:(0,R.jsx)(z,{variant:`running`,children:`Restart`})}):null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-1.5`,children:[(0,R.jsx)(z,{variant:e.source===`env`?`running`:`neutral`,children:y(e)}),e.env_locked?(0,R.jsx)(`span`,{className:`text-xs text-muted-foreground`,children:`Environment locked`}):null]})}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`grid gap-1`,children:[(0,R.jsx)(z,{variant:e.configured?`succeeded`:`missing`,children:e.configured?`Configured`:`Missing`}),e.is_secret?(0,R.jsx)(`span`,{className:`font-mono text-xs text-muted-foreground`,children:e.masked_value||(e.configured?`Hidden`:`No secret`)}):(0,R.jsx)(`span`,{className:`break-words text-xs text-muted-foreground`,children:String(e.value??``)||`Default`}),e.is_secret&&e.secret_encryption_configured===!1?(0,R.jsx)(`span`,{className:`text-xs text-red-300`,children:`Encryption key missing`}):null]})}),(0,R.jsx)(K,{children:b(e)}),(0,R.jsx)(K,{children:(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-2`,children:[(0,R.jsx)(B,{type:`button`,size:`sm`,onClick:()=>i(e.key,s),disabled:!o||e.is_secret&&!s.trim()||c||e.is_secret&&e.secret_encryption_configured===!1,children:`Save`}),(0,R.jsx)(B,{type:`button`,size:`sm`,variant:`outline`,onClick:()=>a(e.key),disabled:!o||e.source!==`database`,children:`Clear`})]})})]},e.key)}return(0,R.jsxs)(`div`,{className:`grid gap-4`,children:[(0,R.jsxs)(V,{children:[(0,R.jsxs)(H,{children:[(0,R.jsx)(zt,{children:`Configuration`}),(0,R.jsxs)(B,{id:`refreshConfiguration`,type:`button`,variant:`outline`,onClick:r,disabled:t.configuration,children:[(0,R.jsx)(C,{}),`Refresh`]})]}),(0,R.jsxs)(Bt,{className:`grid gap-4`,children:[(0,R.jsx)(`section`,{className:`grid gap-3 sm:grid-cols-2 lg:grid-cols-4`,"aria-label":`Configuration summary`,children:[[`Total`,e.length],[`Configured`,_.configured],[`Missing`,_.missing],[`Env locked`,_.envLocked]].map(([e,t])=>(0,R.jsxs)(`div`,{className:`rounded-md border bg-background p-3`,children:[(0,R.jsx)(`span`,{className:`text-[11px] font-extrabold uppercase text-muted-foreground`,children:e}),(0,R.jsx)(`strong`,{className:`mt-1 block text-xl`,children:t})]},e))}),(0,R.jsxs)(`section`,{className:`flex flex-wrap gap-2`,"aria-label":`Configuration groups`,children:[(0,R.jsxs)(B,{type:`button`,size:`sm`,variant:c===`All`?`default`:`outline`,"aria-pressed":c===`All`,onClick:()=>u(`All`),children:[`All groups`,(0,R.jsx)(`span`,{className:`font-mono text-[11px]`,children:e.length})]}),h.map(t=>{let n=e.filter(e=>e.category===t.category).length;return(0,R.jsxs)(B,{type:`button`,size:`sm`,variant:c===t.category?`default`:`outline`,"aria-pressed":c===t.category,onClick:()=>u(t.category),children:[t.label,(0,R.jsx)(`span`,{className:`font-mono text-[11px]`,children:n})]},t.category)})]})]})]}),(0,R.jsx)(kn,{hidden:v!==0,children:`No configuration entries found.`}),g.map(e=>{let t=e.items.filter(e=>e.configured).length,n=e.items.length-t,r=e.items.some(e=>e.restart_required);return(0,R.jsxs)(V,{id:ln(e.category),className:L(`scroll-mt-4 transition-shadow`,d===e.category&&`ring-2 ring-primary ring-offset-2 ring-offset-background`),children:[(0,R.jsxs)(H,{className:`items-start`,children:[(0,R.jsxs)(`div`,{className:`grid gap-1`,children:[(0,R.jsx)(zt,{children:e.label}),(0,R.jsx)(`span`,{className:`text-sm text-muted-foreground`,children:e.description})]}),(0,R.jsxs)(`div`,{className:`flex flex-wrap justify-end gap-1.5`,children:[(0,R.jsxs)(z,{variant:`neutral`,children:[e.items.length,` settings`]}),(0,R.jsxs)(z,{variant:n?`missing`:`succeeded`,children:[t,` configured`]}),r?(0,R.jsx)(z,{variant:`running`,children:`Restart`}):null]})]}),x(e.label,e.primaryItems,e.category),e.advancedItems.length?(0,R.jsxs)(`details`,{className:`border-t bg-background/40`,children:[(0,R.jsxs)(`summary`,{className:`flex min-h-11 cursor-pointer items-center justify-between gap-3 px-4 py-3 text-sm font-extrabold`,children:[(0,R.jsx)(`span`,{children:`Advanced`}),(0,R.jsxs)(`span`,{className:`font-mono text-xs text-muted-foreground`,children:[e.advancedItems.length,` settings`]})]}),(0,R.jsx)(`div`,{className:`border-t`,children:x(`${e.label} advanced`,e.advancedItems,`${e.category}-advanced`)})]}):null]},e.category)})]})}var fr=document.getElementById(`root`);if(fr)(0,fe.createRoot)(fr).render((0,R.jsx)(l.StrictMode,{children:(0,R.jsx)(jn,{})}));else throw Error(`Missing #root container`); \ No newline at end of file diff --git a/apps/api/src/five08/backend/static/dashboard/index.html b/apps/api/src/five08/backend/static/dashboard/index.html index d1c55b83..203933dc 100644 --- a/apps/api/src/five08/backend/static/dashboard/index.html +++ b/apps/api/src/five08/backend/static/dashboard/index.html @@ -4,7 +4,7 @@ 508 Operations Dashboard - + diff --git a/apps/discord_bot/README.md b/apps/discord_bot/README.md index 0e1284ec..c2e4aaa1 100644 --- a/apps/discord_bot/README.md +++ b/apps/discord_bot/README.md @@ -150,6 +150,7 @@ Relevant configuration: - `/create-mailbox` - Description: Create a Migadu mailbox for a 508 user, optionally link it to a CRM contact, and sync `c508Email`. - Prerequisites: `MIGADU_API_USER` and `MIGADU_API_KEY` must be configured (configured in env; command will fail if missing). + - Newsletter sync: if Brevo and/or Keila are configured, the new 508 mailbox and backup email are added to the configured 508 members audience. Brevo uses `BREVO_508_MEMBERS_NEWSLETTER_LIST_ID`, or the list named by `BREVO_508_MEMBERS_NEWSLETTER_LIST_NAME` (default `508 members`) when the explicit ID is unset. Newsletter failures are shown as warnings and do not block mailbox creation. - Required role: Admin - Args: - `mailbox_username` (required): 508 mailbox username or address. If the domain is omitted, `@508.dev` is added automatically. diff --git a/apps/discord_bot/src/five08/discord_bot/cogs/crm.py b/apps/discord_bot/src/five08/discord_bot/cogs/crm.py index 7ddf0d4f..27a55bc5 100644 --- a/apps/discord_bot/src/five08/discord_bot/cogs/crm.py +++ b/apps/discord_bot/src/five08/discord_bot/cogs/crm.py @@ -51,6 +51,10 @@ ResumeProcessorConfig, ResumeProfileProcessor, ) +from five08.newsletter_sync import ( + format_newsletter_sync_warning, + sync_newsletter_contacts, +) from five08.skills import normalize_skill, normalize_skill_list from five08.discord_bot.utils.audit import DiscordAuditCogMixin from five08.discord_bot.utils.role_decorators import ( @@ -133,10 +137,12 @@ def __init__( *, mailbox_email: str, partial_success: str, + newsletter_error: str | None = None, ) -> None: super().__init__(message) self.mailbox_email = mailbox_email self.partial_success = partial_success + self.newsletter_error = newsletter_error @dataclass(frozen=True, slots=True) @@ -147,6 +153,7 @@ class MailboxProvisioningResult: created: bool crm_updated: bool backup_email: str + newsletter_error: str | None = None @dataclass(frozen=True, slots=True) @@ -3816,6 +3823,27 @@ def _migadu_client(self) -> MigaduClient: domain=self._migadu_mailbox_domain(), ) + async def _add_emails_to_newsletter(self, emails: list[str]) -> str | None: + """Best-effort subscribe mailbox and backup addresses to newsletter tools.""" + try: + result = await asyncio.to_thread( + sync_newsletter_contacts, + settings, + emails, + source="discord_create_user_accounts", + ) + except Exception as exc: + error_warning = self._sanitize_error_message_for_discord( + f"Newsletter sync failed: {exc}", + max_length=500, + ) + logger.warning("Newsletter sync warning: %s", error_warning, exc_info=True) + return error_warning + warning = format_newsletter_sync_warning(result) + if warning: + logger.warning("Newsletter sync warning: %s", warning) + return warning + def _normalize_mailbox_request(self, mailbox_username: str) -> tuple[str, str]: """Normalize a bare or full 508 mailbox request to email and local-part.""" normalized = mailbox_username.strip().lower() @@ -8284,6 +8312,12 @@ async def _create_migadu_mailbox_for_contact( created=False, crm_updated=False, backup_email="", + newsletter_error=await self._add_emails_to_newsletter( + [ + existing_email, + self._contact_text_value(contact.get("emailAddress")) or "", + ] + ), ) backup_email = self._normalize_full_email( @@ -8307,6 +8341,10 @@ async def _create_migadu_mailbox_for_contact( partial_success="mailbox_created_address_mismatch", ) + newsletter_error = await self._add_emails_to_newsletter( + [target_email, backup_email] + ) + try: await asyncio.to_thread( self.espo_api.request, @@ -8319,6 +8357,7 @@ async def _create_migadu_mailbox_for_contact( str(exc), mailbox_email=target_email, partial_success="mailbox_created_crm_update_failed", + newsletter_error=newsletter_error, ) from exc contact["c508Email"] = target_email @@ -8327,6 +8366,7 @@ async def _create_migadu_mailbox_for_contact( created=True, crm_updated=True, backup_email=backup_email, + newsletter_error=newsletter_error, ) async def _invite_outline_user_for_contact( @@ -8564,22 +8604,38 @@ async def _create_user_accounts_for_contact( "mailbox_username": mailbox_username, "mailbox_email": exc.mailbox_email, "partial_success": exc.partial_success, + "newsletter_error": exc.newsletter_error, "error": message, }, ) + newsletter_warning = ( + self._sanitize_error_message_for_discord( + exc.newsletter_error, + max_length=500, + ) + if exc.newsletter_error + else None + ) + newsletter_line = ( + f"\nNewsletter: subscription warning: `{newsletter_warning}`" + if newsletter_warning + else "\nNewsletter: added mailbox and backup email." + ) if exc.partial_success == "mailbox_created_crm_update_failed": await interaction.followup.send( "⚠️ Created the mailbox, but failed to update CRM `c508Email`.\n" f"Email: `{exc.mailbox_email}`\n" f"Error: `{message}`\n" - "SSO provisioning and Outline invite were not started.", + "SSO provisioning and Outline invite were not started." + f"{newsletter_line}", ephemeral=True, ) else: await interaction.followup.send( f"⚠️ {message}\n" f"Created mailbox: `{exc.mailbox_email}`\n" - "SSO provisioning and Outline invite were not started.", + "SSO provisioning and Outline invite were not started." + f"{newsletter_line}", ephemeral=True, ) except SSOProvisioningPartialError as exc: @@ -8704,6 +8760,8 @@ async def _create_user_accounts_for_contact( "sso_created": result.sso.created, "sso_crm_updated": result.sso.crm_updated, "outline_invited": result.outline_invited, + "newsletter_subscribed": result.mailbox.newsletter_error is None, + "newsletter_error": result.mailbox.newsletter_error, "recovery_email_error": result.sso.recovery_email_error, }, resource_type="crm_contact", @@ -8720,6 +8778,16 @@ async def _create_user_accounts_for_contact( f"(user ID `{result.sso.user_id}`).", "Outline invite: sent.", ] + if result.mailbox.newsletter_error is None: + message_lines.append("Newsletter: added mailbox and backup email.") + else: + newsletter_warning = self._sanitize_error_message_for_discord( + result.mailbox.newsletter_error, + max_length=500, + ) + message_lines.append( + f"Newsletter: subscription warning: `{newsletter_warning}`" + ) if result.sso.created: if result.sso.recovery_email_error is None: message_lines.append("SSO recovery email: sent.") diff --git a/apps/discord_bot/src/five08/discord_bot/cogs/migadu.py b/apps/discord_bot/src/five08/discord_bot/cogs/migadu.py index 8730683f..34d5041f 100644 --- a/apps/discord_bot/src/five08/discord_bot/cogs/migadu.py +++ b/apps/discord_bot/src/five08/discord_bot/cogs/migadu.py @@ -26,6 +26,10 @@ from five08.discord_bot.config import settings from five08.discord_bot.utils.audit import DiscordAuditCogMixin from five08.discord_bot.utils.role_decorators import require_role +from five08.newsletter_sync import ( + format_newsletter_sync_warning, + sync_newsletter_contacts, +) logger = logging.getLogger(__name__) @@ -55,6 +59,7 @@ class MailboxCreationOutcome: mailbox_name: str crm_contact: dict[str, Any] | None sync_error: str | None = None + newsletter_error: str | None = None def _truncate_discord_text(value: str, *, limit: int) -> str: @@ -208,6 +213,27 @@ def _migadu_client(self) -> MigaduClient: domain=self._migadu_mailbox_domain(), ) + async def _add_emails_to_newsletter(self, emails: list[str]) -> str | None: + """Best-effort subscribe mailbox and backup addresses to newsletter tools.""" + try: + result = await asyncio.to_thread( + sync_newsletter_contacts, + settings, + emails, + source="discord_create_mailbox", + ) + except Exception as exc: + error_warning = _truncate_discord_text( + f"Newsletter sync failed: {exc}", + limit=500, + ) + logger.warning("Newsletter sync warning: %s", error_warning, exc_info=True) + return error_warning + warning = format_newsletter_sync_warning(result) + if warning: + logger.warning("Newsletter sync warning: %s", warning) + return warning + def _normalize_mailbox_request(self, mailbox_username: str) -> tuple[str, str]: """ Normalize user input and derive: @@ -580,6 +606,9 @@ async def _execute_mailbox_creation( name=mailbox_name, ) created_address = str(mailbox.get("address") or context.mailbox_email) + newsletter_error = await self._add_emails_to_newsletter( + [created_address, backup_email] + ) contact_to_update = pre_resolved_contact sync_error: str | None = None @@ -617,6 +646,7 @@ async def _execute_mailbox_creation( mailbox_name=mailbox_name, crm_contact=contact_to_update, sync_error=sync_error, + newsletter_error=newsletter_error, ) def _build_mailbox_embed( @@ -650,6 +680,21 @@ def _build_mailbox_embed( value=outcome.sync_error, inline=False, ) + if outcome.newsletter_error: + embed.add_field( + name="Newsletter", + value=_truncate_discord_text( + f"Newsletter subscription warning: {outcome.newsletter_error}", + limit=1024, + ), + inline=False, + ) + else: + embed.add_field( + name="Newsletter", + value="Added mailbox and backup email to newsletter tools.", + inline=False, + ) return embed @@ -754,6 +799,7 @@ async def _handle_mailbox_creation( else None ), "crm_sync_error": outcome.sync_error, + "newsletter_error": outcome.newsletter_error, "mailbox_created": True, }, resource_type="discord_command", @@ -789,6 +835,8 @@ async def _handle_mailbox_creation( else None ), "forwarded_to": outcome.backup_email, + "newsletter_subscribed": outcome.newsletter_error is None, + "newsletter_error": outcome.newsletter_error, }, resource_type="discord_command", ) diff --git a/apps/worker/src/five08/worker/jobs.py b/apps/worker/src/five08/worker/jobs.py index 5d86de0b..3c876683 100644 --- a/apps/worker/src/five08/worker/jobs.py +++ b/apps/worker/src/five08/worker/jobs.py @@ -6,7 +6,12 @@ from email import message_from_bytes from collections.abc import Callable from typing import Any +from urllib.parse import unquote +from five08.redaction import ( + EMAIL_ADDRESS_PATTERN, + PERCENT_ENCODED_EMAIL_ADDRESS_PATTERN, +) from five08.worker.config import settings from five08.worker.crm.docuseal_processor import DocusealAgreementProcessor from five08.worker.crm.intake_form_processor import IntakeFormProcessor @@ -16,6 +21,7 @@ from five08.worker.erpnext_project_sync import ERPNextProjectSyncProcessor from five08.worker.mailbox_resume_ingest import ResumeMailboxProcessor from five08.worker.masking import mask_email +from five08.newsletter_sync import NewsletterSyncProcessor logger = logging.getLogger(__name__) @@ -164,6 +170,52 @@ def sync_projects_from_erpnext_job() -> dict[str, Any]: return processor.sync_open_projects() +def _mask_newsletter_sync_result(result: dict[str, Any]) -> dict[str, Any]: + """Mask email addresses before newsletter sync results are persisted.""" + crm_failures = result.get("crm_lookup_failures") + if isinstance(crm_failures, list): + for failure in crm_failures: + if not isinstance(failure, dict): + continue + if failure.get("mailbox"): + failure["mailbox"] = mask_email(str(failure["mailbox"])) + if failure.get("error"): + failure["error"] = _mask_emails_in_text(str(failure["error"])) + + providers = result.get("providers") + if isinstance(providers, dict): + for provider_result in providers.values(): + if not isinstance(provider_result, dict): + continue + failures = provider_result.get("failures") + if not isinstance(failures, list): + continue + for failure in failures: + if not isinstance(failure, dict): + continue + if failure.get("email"): + failure["email"] = mask_email(str(failure["email"])) + if failure.get("error"): + failure["error"] = _mask_emails_in_text(str(failure["error"])) + return result + + +def _mask_emails_in_text(text: str) -> str: + """Mask email-like substrings embedded in free-form error text.""" + text = PERCENT_ENCODED_EMAIL_ADDRESS_PATTERN.sub( + lambda match: mask_email(unquote(match.group(0))), + text, + ) + return EMAIL_ADDRESS_PATTERN.sub(lambda match: mask_email(match.group(0)), text) + + +def sync_508_members_newsletters_job() -> dict[str, Any]: + """Sync Migadu member emails into configured newsletter providers.""" + logger.info("Processing 508 members newsletter sync job") + processor = NewsletterSyncProcessor(settings) + return _mask_newsletter_sync_result(processor.sync_508_members()) + + JOB_FUNCTIONS: dict[str, Callable[..., dict[str, Any]]] = { process_webhook_event.__name__: process_webhook_event, process_contact_skills_job.__name__: process_contact_skills_job, @@ -174,5 +226,6 @@ def sync_projects_from_erpnext_job() -> dict[str, Any]: sync_people_from_crm_job.__name__: sync_people_from_crm_job, sync_person_from_crm_job.__name__: sync_person_from_crm_job, sync_projects_from_erpnext_job.__name__: sync_projects_from_erpnext_job, + sync_508_members_newsletters_job.__name__: sync_508_members_newsletters_job, process_docuseal_agreement_job.__name__: process_docuseal_agreement_job, } diff --git a/docs/configuration.md b/docs/configuration.md index 20a9cce1..f1a6ff66 100644 --- a/docs/configuration.md +++ b/docs/configuration.md @@ -36,6 +36,10 @@ and are intentionally not dashboard-configurable. Onboarding email SMTP settings are dashboard-configurable under the Onboarding category when env overrides are not set. +Newsletter sync settings are normally set from the admin dashboard +configuration page. A non-empty env or `.env` value locks the matching +dashboard field. + ## Queue And Jobs - `REDIS_URL` @@ -208,3 +212,26 @@ Agent gateway: Agent model base URLs must be HTTPS endpoints on allowed provider hosts, except the internal Docker-network Bifrost URL `http://bifrost:8080/openai` is allowed for same-host deployments. + +## Migadu Mailbox And Newsletter Sync + +- `MIGADU_API_USER`, `MIGADU_API_KEY`: required for `/create-mailbox` and `/create-user-accounts`. +- `MIGADU_MAILBOX_DOMAIN`: optional, defaults to `508.dev`. +- `BREVO_API_KEY`: optional for Brevo newsletter sync. +- `BREVO_API_BASE_URL`: optional, defaults to `https://api.brevo.com/v3`. +- `BREVO_API_TIMEOUT_SECONDS`: optional, defaults to `20.0`. +- `BREVO_508_MEMBERS_NEWSLETTER_LIST_ID`: optional explicit Brevo list ID override; use `4` for the 508 members list when setting it directly. +- `BREVO_508_MEMBERS_NEWSLETTER_LIST_NAME`: optional, defaults to `508 members` and is used to look up the list ID when the explicit ID is unset. +- `KEILA_API_KEY`: optional for Keila contact sync. +- `KEILA_API_BASE_URL`: optional, defaults to `https://app.keila.io`. +- `KEILA_API_TIMEOUT_SECONDS`: optional, defaults to `20.0`. +- `NEWSLETTER_SYNC_ENABLED`: optional, defaults to `true`; dashboard changes require an API restart because the scheduler starts at startup. +- `NEWSLETTER_SYNC_INTERVAL_SECONDS`: optional, defaults to `604800`; dashboard changes require an API restart because the scheduler sleep interval is startup-bound. +- `NEWSLETTER_SYNC_EXCLUDED_MAILBOXES`: optional comma-separated mailbox local-parts or full addresses to skip during Migadu resync. + +Mailbox and backup email subscription to configured newsletter tools is best +effort. Failures are reported as warnings and do not block mailbox or account +creation. The periodic sync uses Migadu mailboxes and password recovery emails +as the source of truth for `@508.dev`. When CRM is configured, it only syncs +mailboxes that match a CRM contact; it also skips configured excluded mailboxes +and does not re-add provider-suppressed contacts. diff --git a/packages/shared/src/five08/agent/tools.py b/packages/shared/src/five08/agent/tools.py index 0dc3d2e9..7930ef67 100644 --- a/packages/shared/src/five08/agent/tools.py +++ b/packages/shared/src/five08/agent/tools.py @@ -5,6 +5,7 @@ import itertools import re import threading +from collections.abc import Callable from dataclasses import dataclass, field from datetime import date, datetime, timezone from typing import Any @@ -27,6 +28,10 @@ ) from five08.clients.outline import OutlineClient from five08.crm_contacts import EspoContactRepository +from five08.newsletter_sync import ( + format_newsletter_sync_warning, + sync_newsletter_contacts, +) SSO_ID_FIELD = "cSsoID" @@ -75,6 +80,14 @@ class ToolRuntimeConfig: outline_base_url: str = "https://app.getoutline.com" outline_api_key: str | None = None outline_api_timeout_seconds: float = 20.0 + brevo_api_key: str | None = None + brevo_api_base_url: str = "https://api.brevo.com/v3" + brevo_api_timeout_seconds: float = 20.0 + brevo_508_members_newsletter_list_id: int | None = None + brevo_508_members_newsletter_list_name: str = "508 members" + keila_api_key: str | None = None + keila_api_base_url: str = "https://app.keila.io" + keila_api_timeout_seconds: float = 20.0 @classmethod def from_settings(cls, settings: Any) -> "ToolRuntimeConfig": @@ -127,6 +140,30 @@ def from_settings(cls, settings: Any) -> "ToolRuntimeConfig": "outline_api_timeout_seconds", 20.0, ), + brevo_api_key=getattr(settings, "brevo_api_key", None), + brevo_api_base_url=getattr( + settings, + "brevo_api_base_url", + "https://api.brevo.com/v3", + ), + brevo_api_timeout_seconds=getattr( + settings, + "brevo_api_timeout_seconds", + 20.0, + ), + brevo_508_members_newsletter_list_id=getattr( + settings, "brevo_508_members_newsletter_list_id", None + ), + brevo_508_members_newsletter_list_name=getattr( + settings, "brevo_508_members_newsletter_list_name", "508 members" + ), + keila_api_key=getattr(settings, "keila_api_key", None), + keila_api_base_url=getattr( + settings, "keila_api_base_url", "https://app.keila.io" + ), + keila_api_timeout_seconds=getattr( + settings, "keila_api_timeout_seconds", 20.0 + ), ) @@ -272,10 +309,12 @@ def __init__( *, memory_store: MemoryStore | None = None, runtime_config: ToolRuntimeConfig | None = None, + runtime_config_factory: Callable[[], ToolRuntimeConfig] | None = None, ) -> None: self.task_store = task_store or InMemoryTaskStore() self.memory_store = memory_store or InMemoryMemoryStore() - self.runtime_config = runtime_config or ToolRuntimeConfig() + self._runtime_config = runtime_config or ToolRuntimeConfig() + self._runtime_config_factory = runtime_config_factory self._manifests = { "task_read.search_tasks": ToolManifest( name="task_read.search_tasks", @@ -436,6 +475,13 @@ def __init__( ), } + @property + def runtime_config(self) -> ToolRuntimeConfig: + """Return the current external-tool runtime config.""" + if self._runtime_config_factory is not None: + return self._runtime_config_factory() + return self._runtime_config + def get(self, tool_name: str) -> ToolManifest | None: return self._manifests.get(tool_name) @@ -1389,7 +1435,28 @@ def _migadu_client(self) -> MigaduClient: ), ) - def _create_migadu_mailbox(self, arguments: dict[str, Any]) -> dict[str, Any]: + def _add_emails_to_newsletter(self, emails: list[str]) -> str | None: + try: + result = sync_newsletter_contacts( + self.runtime_config, + emails, + source="agent_account_creation", + ) + except Exception as exc: + text = " ".join(f"Newsletter sync failed: {exc}".split()).strip() + return f"{text[:197]}..." if len(text) > 200 else text + warning = format_newsletter_sync_warning(result) + if not warning: + return None + text = " ".join(warning.split()).strip() + return f"{text[:197]}..." if len(text) > 200 else text + + def _create_migadu_mailbox( + self, + arguments: dict[str, Any], + *, + subscribe_newsletter: bool = True, + ) -> dict[str, Any]: local_part = str(arguments.get("local_part") or "").strip().lower() backup_email = _normalize_full_email( arguments.get("backup_email"), @@ -1398,13 +1465,25 @@ def _create_migadu_mailbox(self, arguments: dict[str, Any]) -> dict[str, Any]: name = str(arguments.get("name") or "").strip() if not local_part or not backup_email or not name: raise ValueError("Mailbox local_part, backup_email, and name are required") - return self._migadu_client().create_mailbox( + mailbox = self._migadu_client().create_mailbox( MigaduMailboxCreateRequest( local_part=local_part, backup_email=backup_email, name=name, ) ) + if subscribe_newsletter: + mailbox_email = str(mailbox.get("address") or "").strip().lower() + if not mailbox_email: + domain = normalize_migadu_mailbox_domain( + self.runtime_config.migadu_mailbox_domain + ) + mailbox_email = f"{local_part}@{domain}" + mailbox["newsletter_error"] = self._add_emails_to_newsletter( + [mailbox_email, backup_email] + ) + mailbox["newsletter_subscribed"] = mailbox["newsletter_error"] is None + return mailbox def _create_migadu_mailbox_for_contact( self, @@ -1427,6 +1506,9 @@ def _create_migadu_mailbox_for_contact( created=False, crm_updated=False, backup_email="", + newsletter_error=self._add_emails_to_newsletter( + [existing_email, _optional_str(contact.get("emailAddress")) or ""] + ), ) backup_email = _normalize_full_email( @@ -1439,7 +1521,8 @@ def _create_migadu_mailbox_for_contact( "local_part": local_part, "backup_email": backup_email, "name": contact_name, - } + }, + subscribe_newsletter=False, ) created_address = str(mailbox.get("address") or target_email).strip().lower() if created_address != target_email: @@ -1457,6 +1540,8 @@ def _create_migadu_mailbox_for_contact( ) raise ToolPartialSuccessError(message, result) + newsletter_error = self._add_emails_to_newsletter([target_email, backup_email]) + try: self._espo_client().update_contact(contact_id, {"c508Email": target_email}) except EspoAPIError as exc: @@ -1467,6 +1552,7 @@ def _create_migadu_mailbox_for_contact( backup_email=backup_email, partial_success="mailbox_created_crm_update_failed", error=_short_error(exc), + newsletter_error=newsletter_error, ) raise ToolPartialSuccessError( "Mailbox was created, but updating CRM c508Email failed.", @@ -1478,6 +1564,7 @@ def _create_migadu_mailbox_for_contact( created=True, crm_updated=True, backup_email=backup_email, + newsletter_error=newsletter_error, ) @staticmethod @@ -1489,12 +1576,15 @@ def _mailbox_result( backup_email: str, partial_success: str | None = None, error: str | None = None, + newsletter_error: str | None = None, ) -> dict[str, Any]: result: dict[str, Any] = { "email": email, "created": created, "crm_updated": crm_updated, "backup_email": backup_email, + "newsletter_subscribed": newsletter_error is None, + "newsletter_error": newsletter_error, } if partial_success is not None: result["partial_success"] = partial_success diff --git a/packages/shared/src/five08/clients/brevo.py b/packages/shared/src/five08/clients/brevo.py new file mode 100644 index 00000000..71eecdab --- /dev/null +++ b/packages/shared/src/five08/clients/brevo.py @@ -0,0 +1,194 @@ +"""Brevo API client helpers shared across services.""" + +from __future__ import annotations + +from typing import Any +from urllib.parse import quote + +import requests + +from five08.clients.contact_email import normalize_provider_contact_email +from five08.redaction import redact_email_addresses + +BREVO_API_BASE_URL = "https://api.brevo.com/v3" +ERROR_BODY_MAX_LENGTH = 500 + + +class BrevoAPIError(RuntimeError): + """Raised when the Brevo API request fails or returns invalid data.""" + + +def _response_body_excerpt(body: object) -> str: + """Return a bounded response-body excerpt for persisted/logged errors.""" + text = redact_email_addresses(" ".join(str(body or "").split())) + if len(text) <= ERROR_BODY_MAX_LENGTH: + return text + return f"{text[:ERROR_BODY_MAX_LENGTH]}..." + + +class BrevoClient: + """Small Brevo API wrapper for newsletter contact subscriptions.""" + + def __init__( + self, + *, + api_key: str, + base_url: str = BREVO_API_BASE_URL, + timeout_seconds: float = 20.0, + ) -> None: + self.api_key = api_key + self.base_url = base_url.rstrip("/") + self.timeout_seconds = timeout_seconds + + def add_contact_to_list( + self, + *, + email: str, + list_id: int, + ) -> dict[str, Any]: + """Create or update one Brevo contact and add it to a list.""" + normalized_email = normalize_provider_contact_email(email, "Brevo") + if list_id <= 0: + raise ValueError("Brevo list ID must be a positive integer.") + + payload = { + "email": normalized_email, + "listIds": [list_id], + "updateEnabled": True, + } + headers = { + "Accept": "application/json", + "Content-Type": "application/json", + "api-key": self.api_key, + } + + try: + response = requests.post( + f"{self.base_url}/contacts", + headers=headers, + json=payload, + timeout=self.timeout_seconds, + ) + except requests.RequestException as exc: + raise BrevoAPIError(f"Brevo API request failed: {exc}") from exc + + if response.status_code not in {200, 201, 204}: + raise BrevoAPIError( + "Brevo contact subscription failed: " + f"status={response.status_code}, " + f"body={_response_body_excerpt(response.text)}" + ) + + if not response.content: + return {} + + try: + data = response.json() + except ValueError as exc: + raise BrevoAPIError("Brevo response payload must be valid JSON.") from exc + + if not isinstance(data, dict): + raise BrevoAPIError("Brevo response payload must be a JSON object.") + return data + + def get_contact(self, email: str) -> dict[str, Any] | None: + """Return one Brevo contact by email, or None when it does not exist.""" + normalized_email = normalize_provider_contact_email(email, "Brevo") + headers = { + "Accept": "application/json", + "api-key": self.api_key, + } + try: + response = requests.get( + f"{self.base_url}/contacts/{quote(normalized_email, safe='')}", + headers=headers, + timeout=self.timeout_seconds, + ) + except requests.RequestException as exc: + raise BrevoAPIError(f"Brevo API request failed: {exc}") from exc + + if response.status_code == 404: + return None + if response.status_code != 200: + raise BrevoAPIError( + "Brevo contact lookup failed: " + f"status={response.status_code}, " + f"body={_response_body_excerpt(response.text)}" + ) + + try: + data = response.json() + except ValueError as exc: + raise BrevoAPIError("Brevo response payload must be valid JSON.") from exc + + if not isinstance(data, dict): + raise BrevoAPIError("Brevo response payload must be a JSON object.") + return data + + def find_list_id_by_name(self, name: str) -> int | None: + """Find a Brevo contact list ID by exact case-insensitive name.""" + normalized_name = name.strip().casefold() + if not normalized_name: + raise ValueError("Brevo list name must be non-empty.") + + limit = 50 + offset = 0 + while True: + payload = self._get_lists_page(limit=limit, offset=offset) + lists = payload.get("lists", []) + if not isinstance(lists, list): + raise BrevoAPIError("Brevo lists payload must include a list array.") + + for item in lists: + if not isinstance(item, dict): + continue + item_name = str(item.get("name") or "").strip().casefold() + if item_name != normalized_name: + continue + list_id = item.get("id") + if not isinstance(list_id, int): + raise BrevoAPIError("Brevo list ID must be an integer.") + return list_id + + count = payload.get("count") + offset += limit + if isinstance(count, int) and offset >= count: + return None + if len(lists) < limit: + return None + + def _get_lists_page(self, *, limit: int, offset: int) -> dict[str, Any]: + headers = { + "Accept": "application/json", + "api-key": self.api_key, + } + params: dict[str, str | int] = { + "limit": limit, + "offset": offset, + "sort": "asc", + } + try: + response = requests.get( + f"{self.base_url}/contacts/lists", + headers=headers, + params=params, + timeout=self.timeout_seconds, + ) + except requests.RequestException as exc: + raise BrevoAPIError(f"Brevo API request failed: {exc}") from exc + + if response.status_code != 200: + raise BrevoAPIError( + "Brevo list lookup failed: " + f"status={response.status_code}, " + f"body={_response_body_excerpt(response.text)}" + ) + + try: + data = response.json() + except ValueError as exc: + raise BrevoAPIError("Brevo response payload must be valid JSON.") from exc + + if not isinstance(data, dict): + raise BrevoAPIError("Brevo response payload must be a JSON object.") + return data diff --git a/packages/shared/src/five08/clients/contact_email.py b/packages/shared/src/five08/clients/contact_email.py new file mode 100644 index 00000000..1c064594 --- /dev/null +++ b/packages/shared/src/five08/clients/contact_email.py @@ -0,0 +1,15 @@ +"""Shared contact email validation for provider clients.""" + +from __future__ import annotations + +from five08.onboarding_email import validate_plain_email + + +def normalize_provider_contact_email(value: str, provider_name: str) -> str: + """Normalize a provider contact email after full-address validation.""" + try: + return validate_plain_email(value, "contact email").lower() + except ValueError as exc: + raise ValueError( + f"{provider_name} contact email must be a full email address." + ) from exc diff --git a/packages/shared/src/five08/clients/keila.py b/packages/shared/src/five08/clients/keila.py new file mode 100644 index 00000000..72684c84 --- /dev/null +++ b/packages/shared/src/five08/clients/keila.py @@ -0,0 +1,175 @@ +"""Keila API client helpers shared across services.""" + +from __future__ import annotations + +from typing import Any +from urllib.parse import quote + +import requests + +from five08.clients.contact_email import normalize_provider_contact_email +from five08.redaction import redact_email_addresses + +KEILA_API_BASE_URL = "https://app.keila.io" +ERROR_BODY_MAX_LENGTH = 500 +_EXISTING_CONTACT_UNSET = object() + + +class KeilaAPIError(RuntimeError): + """Raised when the Keila API request fails or returns invalid data.""" + + +def _response_body_excerpt(body: object) -> str: + """Return a bounded response-body excerpt for persisted/logged errors.""" + text = redact_email_addresses(" ".join(str(body or "").split())) + if len(text) <= ERROR_BODY_MAX_LENGTH: + return text + return f"{text[:ERROR_BODY_MAX_LENGTH]}..." + + +class KeilaClient: + """Small Keila API wrapper for contact synchronization.""" + + def __init__( + self, + *, + api_key: str, + base_url: str = KEILA_API_BASE_URL, + timeout_seconds: float = 20.0, + ) -> None: + self.api_key = api_key + self.base_url = base_url.rstrip("/") + self.timeout_seconds = timeout_seconds + + def get_contact_by_email(self, email: str) -> dict[str, Any] | None: + """Return one Keila contact by email, or None when it does not exist.""" + normalized_email = normalize_provider_contact_email(email, "Keila") + response = self._request( + "GET", + f"/api/v1/contacts/{quote(normalized_email, safe='')}", + params={"id_type": "email"}, + allow_not_found=True, + ) + return response + + def upsert_active_contact( + self, + *, + email: str, + first_name: str | None = None, + last_name: str | None = None, + data: dict[str, Any] | None = None, + existing_contact: dict[str, Any] | None | object = _EXISTING_CONTACT_UNSET, + ) -> dict[str, Any]: + """Create or update a Keila contact without changing suppressed statuses.""" + normalized_email = normalize_provider_contact_email(email, "Keila") + + payload: dict[str, Any] = { + "email": normalized_email, + "status": "active", + "data": data or {}, + } + if first_name: + payload["first_name"] = first_name + if last_name: + payload["last_name"] = last_name + + if existing_contact is _EXISTING_CONTACT_UNSET: + existing = self.get_contact_by_email(normalized_email) + else: + if existing_contact is not None and not isinstance(existing_contact, dict): + raise TypeError("existing_contact must be a Keila contact object.") + existing = existing_contact + if existing is None: + return ( + self._request("POST", "/api/v1/contacts", json={"data": payload}) or {} + ) + + contact_id_value = existing.get("id") + if not contact_id_value: + raise KeilaAPIError( + f"Existing Keila contact {normalized_email} missing id." + ) + contact_id = str(contact_id_value) + existing_data = existing.get("data") + payload["data"] = _merge_contact_data(existing_data, data or {}) + payload.pop("status", None) + return ( + self._request( + "PATCH", + f"/api/v1/contacts/{contact_id}", + json={"data": payload}, + ) + or {} + ) + + def _request( + self, + method: str, + path: str, + *, + params: dict[str, str] | None = None, + json: dict[str, Any] | None = None, + allow_not_found: bool = False, + ) -> dict[str, Any] | None: + headers = { + "Accept": "application/json", + "Authorization": f"Bearer {self.api_key}", + } + if json is not None: + headers["Content-Type"] = "application/json" + + try: + response = requests.request( + method, + f"{self.base_url}{path}", + headers=headers, + params=params, + json=json, + timeout=self.timeout_seconds, + ) + except requests.RequestException as exc: + raise KeilaAPIError(f"Keila API request failed: {exc}") from exc + + if allow_not_found and response.status_code == 404: + return None + if not 200 <= response.status_code < 300: + raise KeilaAPIError( + "Keila API request failed: " + f"status={response.status_code}, " + f"body={_response_body_excerpt(response.text)}" + ) + if not response.content: + return {} + try: + data = response.json() + except ValueError as exc: + raise KeilaAPIError("Keila response payload must be valid JSON.") from exc + if not isinstance(data, dict): + raise KeilaAPIError("Keila response payload must be a JSON object.") + nested = data.get("data") + if isinstance(nested, dict): + return nested + return data + + +def _merge_contact_data( + existing_data: object, + new_data: dict[str, Any], +) -> dict[str, Any]: + """Merge Keila contact data while preserving existing audience tags.""" + existing = existing_data if isinstance(existing_data, dict) else {} + merged = {**existing, **new_data} + existing_audiences = existing.get("audiences") + new_audiences = new_data.get("audiences") + if isinstance(existing_audiences, list) and isinstance(new_audiences, list): + merged["audiences"] = _unique_values([*existing_audiences, *new_audiences]) + return merged + + +def _unique_values(values: list[Any]) -> list[Any]: + unique: list[Any] = [] + for value in values: + if value not in unique: + unique.append(value) + return unique diff --git a/packages/shared/src/five08/clients/migadu.py b/packages/shared/src/five08/clients/migadu.py index ddf60b27..5d41ae26 100644 --- a/packages/shared/src/five08/clients/migadu.py +++ b/packages/shared/src/five08/clients/migadu.py @@ -7,13 +7,24 @@ import requests +from five08.redaction import redact_email_addresses + MIGADU_API_BASE_URL = "https://api.migadu.com/v1" +ERROR_BODY_MAX_LENGTH = 500 class MigaduAPIError(RuntimeError): """Raised when the Migadu API request fails or returns invalid data.""" +def _response_body_excerpt(body: object) -> str: + """Return a bounded response-body excerpt for persisted/logged errors.""" + text = redact_email_addresses(" ".join(str(body or "").split())) + if len(text) <= ERROR_BODY_MAX_LENGTH: + return text + return f"{text[:ERROR_BODY_MAX_LENGTH]}..." + + def normalize_migadu_mailbox_domain(domain: str | None) -> str: """Normalize the configured Migadu mailbox domain.""" normalized = (domain or "508.dev").strip().lower().lstrip(".") @@ -31,6 +42,15 @@ class MigaduMailboxCreateRequest: name: str +@dataclass(frozen=True, slots=True) +class MigaduMailbox: + """Mailbox fields needed for member audience sync.""" + + address: str + name: str + password_recovery_email: str | None + + class MigaduClient: """Small Migadu API wrapper for mailbox creation.""" @@ -72,7 +92,8 @@ def create_mailbox(self, request: MigaduMailboxCreateRequest) -> dict[str, Any]: if response.status_code not in {200, 201}: raise MigaduAPIError( "Migadu mailbox creation failed: " - f"status={response.status_code}, body={response.text}" + f"status={response.status_code}, " + f"body={_response_body_excerpt(response.text)}" ) try: @@ -84,3 +105,50 @@ def create_mailbox(self, request: MigaduMailboxCreateRequest) -> dict[str, Any]: raise MigaduAPIError("Migadu response payload must be a JSON object.") return data + + def list_mailboxes(self) -> list[MigaduMailbox]: + """List mailboxes for the configured domain.""" + try: + response = requests.get( + f"{self.base_url}/domains/{self.domain}/mailboxes", + auth=(self.username, self.api_key), + timeout=self.timeout_seconds, + ) + except requests.RequestException as exc: + raise MigaduAPIError(f"Migadu API request failed: {exc}") from exc + + if response.status_code != 200: + raise MigaduAPIError( + "Migadu mailbox listing failed: " + f"status={response.status_code}, " + f"body={_response_body_excerpt(response.text)}" + ) + + try: + data = response.json() + except ValueError as exc: + raise MigaduAPIError("Migadu response payload must be valid JSON.") from exc + + if not isinstance(data, dict): + raise MigaduAPIError("Migadu response payload must be a JSON object.") + + raw_mailboxes = data.get("mailboxes", []) + if not isinstance(raw_mailboxes, list): + raise MigaduAPIError("Migadu response payload must include mailboxes list.") + + mailboxes: list[MigaduMailbox] = [] + for item in raw_mailboxes: + if not isinstance(item, dict): + continue + address = str(item.get("address") or "").strip().lower() + if not address: + continue + recovery = str(item.get("password_recovery_email") or "").strip().lower() + mailboxes.append( + MigaduMailbox( + address=address, + name=str(item.get("name") or "").strip(), + password_recovery_email=recovery or None, + ) + ) + return mailboxes diff --git a/packages/shared/src/five08/newsletter_sync.py b/packages/shared/src/five08/newsletter_sync.py new file mode 100644 index 00000000..623ccaea --- /dev/null +++ b/packages/shared/src/five08/newsletter_sync.py @@ -0,0 +1,454 @@ +"""508 member newsletter audience synchronization.""" + +from __future__ import annotations + +from dataclasses import dataclass +from typing import Any, Iterable, Protocol + +from five08.clients.brevo import BrevoClient +from five08.clients.espo import EspoAPIError, EspoClient +from five08.clients.keila import KeilaClient +from five08.clients.migadu import MigaduClient, MigaduMailbox +from five08.redaction import redact_email_addresses + +CRM_BLOCKED_TYPES = {"inactive member", "rejected", "blocked"} +CRM_BLOCKED_ONBOARDING_STATES = {"rejected", "waitlist"} +PROVIDER_SUPPRESSED_STATUSES = {"unsubscribed", "unreachable", "blocked"} + + +class CRMContactLookupError(RuntimeError): + """Raised when CRM block-state lookup fails during member sync.""" + + +@dataclass(frozen=True, slots=True) +class NewsletterContact: + """One email address derived from one Migadu mailbox.""" + + email: str + mailbox_email: str + name: str + source: str + + +class NewsletterProvider(Protocol): + """Provider interface for additive newsletter subscription sync.""" + + name: str + + def ensure_contact(self, contact: NewsletterContact) -> str: + """Ensure contact exists, returning a sync status key.""" + + +def _split_name(full_name: str) -> tuple[str | None, str | None]: + normalized = full_name.strip() + if not normalized: + return None, None + parts = normalized.rsplit(" ", 1) + if len(parts) == 1: + return parts[0], None + return parts[0], parts[1] + + +def _normalized_emails_for_mailbox(mailbox: MigaduMailbox) -> list[NewsletterContact]: + contacts = [ + NewsletterContact( + email=mailbox.address, + mailbox_email=mailbox.address, + name=mailbox.name, + source="migadu_mailbox", + ) + ] + if mailbox.password_recovery_email: + contacts.append( + NewsletterContact( + email=mailbox.password_recovery_email, + mailbox_email=mailbox.address, + name=mailbox.name, + source="migadu_password_recovery_email", + ) + ) + deduped: list[NewsletterContact] = [] + seen: set[str] = set() + for contact in contacts: + if contact.email in seen: + continue + seen.add(contact.email) + deduped.append(contact) + return deduped + + +def _normalized_csv_set(value: str) -> set[str]: + return {item.strip().lower() for item in value.split(",") if item.strip()} + + +def _mailbox_local_part(email: str) -> str: + return email.split("@", 1)[0].strip().lower() + + +def _is_mailbox_excluded(email: str, excluded_mailboxes: set[str]) -> bool: + normalized_email = email.strip().lower() + if normalized_email in excluded_mailboxes: + return True + local_part = _mailbox_local_part(normalized_email) + return bool(local_part and local_part in excluded_mailboxes) + + +def _is_crm_blocked(contact: dict[str, Any] | None) -> bool: + if contact is None: + return False + contact_type = str(contact.get("type") or "").strip().casefold() + onboarding = str(contact.get("cOnboardingState") or "").strip().casefold() + return ( + contact_type in CRM_BLOCKED_TYPES or onboarding in CRM_BLOCKED_ONBOARDING_STATES + ) + + +def _contains_list_id(value: object, list_id: int) -> bool: + if not isinstance(value, list): + return False + return any(str(item).strip() == str(list_id) for item in value) + + +class BrevoNewsletterProvider: + """Brevo implementation for the 508 members newsletter list.""" + + name = "brevo" + + def __init__( + self, + client: BrevoClient, + *, + list_id: int | None, + list_name: str, + ) -> None: + self.client = client + self.list_id = list_id + self.list_name = list_name + self._list_id_lookup_completed = list_id is not None + self._resolved_list_id = list_id + + def _list_id(self) -> int | None: + if not self._list_id_lookup_completed: + self._resolved_list_id = self.client.find_list_id_by_name(self.list_name) + self._list_id_lookup_completed = True + return self._resolved_list_id + + def ensure_contact(self, contact: NewsletterContact) -> str: + list_id = self._list_id() + if list_id is None: + return "skipped_list_missing" + + existing = self.client.get_contact(contact.email) + if existing is not None and ( + bool(existing.get("emailBlacklisted")) + or str(existing.get("status") or "").strip().casefold() + in PROVIDER_SUPPRESSED_STATUSES + ): + return "skipped_provider_suppressed" + + if existing is not None: + if _contains_list_id(existing.get("listUnsubscribed"), list_id): + return "skipped_provider_suppressed" + self.client.add_contact_to_list(email=contact.email, list_id=list_id) + return "synced" + + +class KeilaNewsletterProvider: + """Keila implementation using project contacts and contact data tags.""" + + name = "keila" + + def __init__(self, client: KeilaClient) -> None: + self.client = client + + def ensure_contact(self, contact: NewsletterContact) -> str: + existing = self.client.get_contact_by_email(contact.email) + if existing is not None: + status = str(existing.get("status") or "").strip().casefold() + if status in PROVIDER_SUPPRESSED_STATUSES: + return "skipped_provider_suppressed" + + first_name, last_name = _split_name(contact.name) + self.client.upsert_active_contact( + email=contact.email, + first_name=first_name, + last_name=last_name, + data={ + "audiences": ["508_members"], + "source": contact.source, + "mailbox_email": contact.mailbox_email, + }, + existing_contact=existing, + ) + return "synced" + + +class NewsletterSyncProcessor: + """Synchronize Migadu member mailboxes into configured newsletter providers.""" + + def __init__(self, settings: Any) -> None: + self.settings = settings + self.excluded_mailboxes = _normalized_csv_set( + settings.newsletter_sync_excluded_mailboxes + ) + + def sync_508_members(self) -> dict[str, Any]: + providers = build_newsletter_providers(self.settings) + result: dict[str, Any] = { + "mailboxes_scanned": 0, + "system_mailboxes_skipped": 0, + "crm_blocked_skipped": 0, + "crm_unmatched_skipped": 0, + "crm_lookup_failed_skipped": 0, + "contacts_considered": 0, + "providers": { + provider.name: {"synced": 0, "skipped": 0, "failed": 0} + for provider in providers + }, + } + if not providers: + result["warning"] = "no_newsletter_providers_configured" + return result + + crm_lookup_enabled = self._crm_lookup_enabled() + for mailbox in self._migadu_client().list_mailboxes(): + result["mailboxes_scanned"] += 1 + if _is_mailbox_excluded(mailbox.address, self.excluded_mailboxes): + result["system_mailboxes_skipped"] += 1 + continue + + try: + crm_contacts = self._list_crm_contacts(mailbox) + except CRMContactLookupError as exc: + result["crm_lookup_failed_skipped"] += 1 + failures = result.setdefault("crm_lookup_failures", []) + if isinstance(failures, list) and len(failures) < 20: + failures.append({"mailbox": mailbox.address, "error": str(exc)}) + continue + if crm_lookup_enabled and not crm_contacts: + result["crm_unmatched_skipped"] += 1 + continue + if any(_is_crm_blocked(contact) for contact in crm_contacts): + result["crm_blocked_skipped"] += 1 + continue + + for contact in _normalized_emails_for_mailbox(mailbox): + result["contacts_considered"] += 1 + for provider in providers: + provider_result = result["providers"][provider.name] + try: + status = provider.ensure_contact(contact) + except Exception as exc: + provider_result["failed"] += 1 + failures = provider_result.setdefault("failures", []) + if isinstance(failures, list) and len(failures) < 20: + failures.append({"email": contact.email, "error": str(exc)}) + continue + if status == "synced": + provider_result["synced"] += 1 + else: + provider_result["skipped"] += 1 + statuses = provider_result.setdefault("statuses", {}) + if isinstance(statuses, dict): + statuses[status] = int(statuses.get(status, 0)) + 1 + return result + + def _migadu_client(self) -> MigaduClient: + return MigaduClient( + username=_required(self.settings.migadu_api_user, "MIGADU_API_USER"), + api_key=_required(self.settings.migadu_api_key, "MIGADU_API_KEY"), + domain=self.settings.migadu_mailbox_domain, + ) + + def _crm_client(self) -> EspoClient | None: + base_url = getattr(self.settings, "espo_base_url", None) + api_key = getattr(self.settings, "espo_api_key", None) + if not base_url or not api_key: + return None + return EspoClient(base_url, api_key) + + def _crm_lookup_enabled(self) -> bool: + return self._crm_client() is not None + + def _list_crm_contacts(self, mailbox: MigaduMailbox) -> list[dict[str, Any]]: + client = self._crm_client() + if client is None: + return [] + filters: list[dict[str, Any]] = [ + {"type": "equals", "attribute": "c508Email", "value": mailbox.address}, + {"type": "equals", "attribute": "emailAddress", "value": mailbox.address}, + ] + if mailbox.password_recovery_email: + filters.append( + { + "type": "equals", + "attribute": "emailAddress", + "value": mailbox.password_recovery_email, + } + ) + try: + response = client.list_contacts( + { + "where": [{"type": "or", "value": filters}], + "maxSize": 20, + "select": "id,name,emailAddress,c508Email,type,cOnboardingState", + } + ) + except EspoAPIError as exc: + raise CRMContactLookupError( + f"CRM contact lookup failed for {mailbox.address}: {exc}" + ) from exc + contacts = response.get("list", []) + if not isinstance(contacts, list): + return [] + return [contact for contact in contacts if isinstance(contact, dict)] + + +def _required(value: str | None, name: str) -> str: + normalized = (value or "").strip() + if not normalized: + raise ValueError(f"{name} is required.") + return normalized + + +def build_newsletter_providers(settings: Any) -> list[NewsletterProvider]: + """Build configured newsletter providers from shared-like settings.""" + providers: list[NewsletterProvider] = [] + brevo_api_key = str(getattr(settings, "brevo_api_key", "") or "").strip() + if brevo_api_key: + list_name = ( + str( + getattr( + settings, "brevo_508_members_newsletter_list_name", "508 members" + ) + or "" + ).strip() + or "508 members" + ) + providers.append( + BrevoNewsletterProvider( + BrevoClient( + api_key=brevo_api_key, + base_url=getattr( + settings, "brevo_api_base_url", "https://api.brevo.com/v3" + ), + timeout_seconds=getattr( + settings, "brevo_api_timeout_seconds", 20.0 + ), + ), + list_id=getattr(settings, "brevo_508_members_newsletter_list_id", None), + list_name=list_name, + ) + ) + + keila_api_key = str(getattr(settings, "keila_api_key", "") or "").strip() + if keila_api_key: + providers.append( + KeilaNewsletterProvider( + KeilaClient( + api_key=keila_api_key, + base_url=getattr( + settings, "keila_api_base_url", "https://app.keila.io" + ), + timeout_seconds=getattr( + settings, "keila_api_timeout_seconds", 20.0 + ), + ) + ) + ) + return providers + + +def sync_newsletter_contacts( + settings: Any, + emails: Iterable[str], + *, + name: str = "", + mailbox_email: str | None = None, + source: str = "account_creation", +) -> dict[str, Any]: + """Best-effort additive sync for known member emails at creation time.""" + providers = build_newsletter_providers(settings) + result: dict[str, Any] = { + "contacts_considered": 0, + "providers": { + provider.name: {"synced": 0, "skipped": 0, "failed": 0} + for provider in providers + }, + } + if not providers: + result["warning"] = "no_newsletter_providers_configured" + return result + + seen: set[str] = set() + default_mailbox_email = (mailbox_email or "").strip().lower() + for email in emails: + normalized_email = email.strip().lower() + if not normalized_email or normalized_email in seen: + continue + seen.add(normalized_email) + if not default_mailbox_email: + default_mailbox_email = normalized_email + result["contacts_considered"] += 1 + contact = NewsletterContact( + email=normalized_email, + mailbox_email=default_mailbox_email, + name=name, + source=source, + ) + for provider in providers: + provider_result = result["providers"][provider.name] + try: + status = provider.ensure_contact(contact) + except Exception as exc: + provider_result["failed"] += 1 + failures = provider_result.setdefault("failures", []) + if isinstance(failures, list) and len(failures) < 20: + failures.append({"email": normalized_email, "error": str(exc)}) + continue + if status == "synced": + provider_result["synced"] += 1 + else: + provider_result["skipped"] += 1 + statuses = provider_result.setdefault("statuses", {}) + if isinstance(statuses, dict): + statuses[status] = int(statuses.get(status, 0)) + 1 + return result + + +def format_newsletter_sync_warning(result: dict[str, Any]) -> str | None: + """Format direct-provisioning sync failures for user-visible warnings.""" + if result.get("warning") == "no_newsletter_providers_configured": + return "No newsletter providers are configured." + + messages: list[str] = [] + providers = result.get("providers", {}) + if not isinstance(providers, dict): + return None + + for provider_name, provider_result in providers.items(): + if not isinstance(provider_result, dict): + continue + failed = int(provider_result.get("failed") or 0) + failures = provider_result.get("failures") + if failed and isinstance(failures, list) and failures: + detail = "; ".join( + redact_email_addresses(str(item.get("error") or "unknown error")) + for item in failures[:3] + if isinstance(item, dict) + ) + messages.append(f"{provider_name} failed for {failed} contact(s): {detail}") + elif failed: + messages.append(f"{provider_name} failed for {failed} contact(s)") + + statuses = provider_result.get("statuses") + if isinstance(statuses, dict): + if statuses.get("skipped_list_missing"): + messages.append(f"{provider_name} list was not found") + suppressed = int(statuses.get("skipped_provider_suppressed") or 0) + if suppressed: + messages.append( + f"{provider_name} skipped {suppressed} suppressed contact(s)" + ) + + return "; ".join(messages) if messages else None diff --git a/packages/shared/src/five08/redaction.py b/packages/shared/src/five08/redaction.py new file mode 100644 index 00000000..9cf2d7b8 --- /dev/null +++ b/packages/shared/src/five08/redaction.py @@ -0,0 +1,19 @@ +"""Small redaction helpers for persisted/logged diagnostic strings.""" + +from __future__ import annotations + +import re + +EMAIL_ADDRESS_PATTERN = re.compile( + r"(? str: + """Replace email-like substrings in text with a stable placeholder.""" + text = PERCENT_ENCODED_EMAIL_ADDRESS_PATTERN.sub("[redacted-email]", text) + return EMAIL_ADDRESS_PATTERN.sub("[redacted-email]", text) diff --git a/packages/shared/src/five08/runtime_config.py b/packages/shared/src/five08/runtime_config.py index cc711595..1181267a 100644 --- a/packages/shared/src/five08/runtime_config.py +++ b/packages/shared/src/five08/runtime_config.py @@ -90,6 +90,110 @@ class RuntimeConfigDBSnapshot: is_secret=True, env_names=("OUTLINE_API_KEY",), ), + RuntimeConfigDefinition( + key="BREVO_API_KEY", + attr="brevo_api_key", + label="Brevo API key", + category="Newsletter", + description="Brevo API key used to add 508 member contacts to the newsletter list.", + is_secret=True, + env_names=("BREVO_API_KEY",), + ), + RuntimeConfigDefinition( + key="BREVO_API_BASE_URL", + attr="brevo_api_base_url", + label="Brevo API base URL", + category="Newsletter", + description="Brevo API endpoint for newsletter contact sync.", + value_type="url", + env_names=("BREVO_API_BASE_URL",), + ), + RuntimeConfigDefinition( + key="BREVO_API_TIMEOUT_SECONDS", + attr="brevo_api_timeout_seconds", + label="Brevo API timeout seconds", + category="Newsletter", + description="Network timeout for Brevo newsletter requests.", + value_type="float", + env_names=("BREVO_API_TIMEOUT_SECONDS",), + min_value=1, + ), + RuntimeConfigDefinition( + key="BREVO_508_MEMBERS_NEWSLETTER_LIST_ID", + attr="brevo_508_members_newsletter_list_id", + label="Brevo 508 members list ID", + category="Newsletter", + description="Optional explicit Brevo list ID; when unset the list is resolved by name.", + value_type="int", + env_names=("BREVO_508_MEMBERS_NEWSLETTER_LIST_ID",), + min_value=1, + ), + RuntimeConfigDefinition( + key="BREVO_508_MEMBERS_NEWSLETTER_LIST_NAME", + attr="brevo_508_members_newsletter_list_name", + label="Brevo 508 members list name", + category="Newsletter", + description="Brevo list name used when the explicit list ID is unset.", + env_names=("BREVO_508_MEMBERS_NEWSLETTER_LIST_NAME",), + ), + RuntimeConfigDefinition( + key="KEILA_API_KEY", + attr="keila_api_key", + label="Keila API key", + category="Newsletter", + description="Keila API key used to tag 508 member contacts.", + is_secret=True, + env_names=("KEILA_API_KEY",), + ), + RuntimeConfigDefinition( + key="KEILA_API_BASE_URL", + attr="keila_api_base_url", + label="Keila API base URL", + category="Newsletter", + description="Keila API endpoint for newsletter contact sync.", + value_type="url", + env_names=("KEILA_API_BASE_URL",), + ), + RuntimeConfigDefinition( + key="KEILA_API_TIMEOUT_SECONDS", + attr="keila_api_timeout_seconds", + label="Keila API timeout seconds", + category="Newsletter", + description="Network timeout for Keila newsletter requests.", + value_type="float", + env_names=("KEILA_API_TIMEOUT_SECONDS",), + min_value=1, + ), + RuntimeConfigDefinition( + key="NEWSLETTER_SYNC_ENABLED", + attr="newsletter_sync_enabled", + label="Newsletter sync enabled", + category="Newsletter", + description="Whether the API starts the recurring 508 members newsletter sync scheduler.", + value_type="bool", + env_names=("NEWSLETTER_SYNC_ENABLED",), + restart_required=True, + ), + RuntimeConfigDefinition( + key="NEWSLETTER_SYNC_INTERVAL_SECONDS", + attr="newsletter_sync_interval_seconds", + label="Newsletter sync interval seconds", + category="Newsletter", + description="Seconds between recurring 508 members newsletter sync enqueue attempts.", + value_type="int", + env_names=("NEWSLETTER_SYNC_INTERVAL_SECONDS",), + restart_required=True, + min_value=60, + ), + RuntimeConfigDefinition( + key="NEWSLETTER_SYNC_EXCLUDED_MAILBOXES", + attr="newsletter_sync_excluded_mailboxes", + label="Newsletter excluded mailboxes", + category="Newsletter", + description="Comma-separated Migadu mailbox local-parts or full addresses skipped by the 508 members resync.", + value_type="csv", + env_names=("NEWSLETTER_SYNC_EXCLUDED_MAILBOXES",), + ), RuntimeConfigDefinition( key="DOCUSEAL_BASE_URL", attr="docuseal_base_url", diff --git a/packages/shared/src/five08/settings.py b/packages/shared/src/five08/settings.py index 900b859e..c213bd28 100644 --- a/packages/shared/src/five08/settings.py +++ b/packages/shared/src/five08/settings.py @@ -77,6 +77,17 @@ class SharedSettings(BaseSettings): outline_base_url: str = "https://app.getoutline.com" outline_api_key: str | None = None outline_api_timeout_seconds: float = 20.0 + brevo_api_key: str | None = None + brevo_api_base_url: str = "https://api.brevo.com/v3" + brevo_api_timeout_seconds: float = 20.0 + brevo_508_members_newsletter_list_id: int | None = Field(default=None, ge=1) + brevo_508_members_newsletter_list_name: str = "508 members" + keila_api_key: str | None = None + keila_api_base_url: str = "https://app.keila.io" + keila_api_timeout_seconds: float = 20.0 + newsletter_sync_enabled: bool = True + newsletter_sync_interval_seconds: int = 604800 + newsletter_sync_excluded_mailboxes: str = "" onboarding_email_smtp_server: str | None = Field( default=None, validation_alias=AliasChoices( @@ -181,6 +192,28 @@ def _normalize_docuseal_member_agreement_template_id( ) from exc raise TypeError("DOCUSEAL_MEMBER_AGREEMENT_TEMPLATE_ID must be an integer") + @field_validator("brevo_508_members_newsletter_list_id", mode="before") + @classmethod + def _normalize_brevo_508_members_newsletter_list_id( + cls, + value: object, + ) -> int | None: + if value is None: + return None + if isinstance(value, int): + return value + if isinstance(value, str): + normalized = value.strip() + if not normalized: + return None + try: + return int(normalized) + except ValueError as exc: + raise ValueError( + "BREVO_508_MEMBERS_NEWSLETTER_LIST_ID must be an integer" + ) from exc + raise TypeError("BREVO_508_MEMBERS_NEWSLETTER_LIST_ID must be an integer") + @classmethod def _skip_dotenv(cls) -> bool: if os.getenv("ENVIRONMENT", "").strip().lower() == "test": diff --git a/tests/unit/test_agent_gateway.py b/tests/unit/test_agent_gateway.py index cbfcd24b..c48d6d2c 100644 --- a/tests/unit/test_agent_gateway.py +++ b/tests/unit/test_agent_gateway.py @@ -1866,6 +1866,8 @@ def test_user_accounts_create_is_admin_only() -> None: def _account_runtime_config( *, outline_api_key: str | None = "outline-key", + brevo_api_key: str | None = None, + keila_api_key: str | None = None, ) -> ToolRuntimeConfig: return ToolRuntimeConfig( espo_base_url="https://crm.example", @@ -1876,6 +1878,9 @@ def _account_runtime_config( authentik_api_token="authentik-token", authentik_recovery_email_stage_id="stage-1", outline_api_key=outline_api_key, + brevo_api_key=brevo_api_key, + brevo_508_members_newsletter_list_id=4, + keila_api_key=keila_api_key, ) @@ -2068,15 +2073,87 @@ def invite_user( self.invites.append(invite) return invite + class FakeBrevoClient: + contacts: dict[str, dict[str, Any]] = {} + subscriptions: list[dict[str, Any]] = [] + + def __init__( + self, + *, + api_key: str, + base_url: str = "https://api.brevo.com/v3", + timeout_seconds: float = 20.0, + ) -> None: + self.api_key = api_key + self.base_url = base_url + self.timeout_seconds = timeout_seconds + + def add_contact_to_list(self, *, email: str, list_id: int) -> dict[str, Any]: + self.subscriptions.append({"email": email, "list_id": list_id}) + return {"id": len(self.subscriptions)} + + def get_contact(self, email: str) -> dict[str, Any] | None: + return self.contacts.get(email) + + def find_list_id_by_name(self, name: str) -> int | None: + return 4 if name == "508 members" else None + + class FakeKeilaClient: + contacts: dict[str, dict[str, Any]] = {} + upserts: list[dict[str, Any]] = [] + + def __init__( + self, + *, + api_key: str, + base_url: str = "https://app.keila.io", + timeout_seconds: float = 20.0, + ) -> None: + self.api_key = api_key + self.base_url = base_url + self.timeout_seconds = timeout_seconds + + def get_contact_by_email(self, email: str) -> dict[str, Any] | None: + return self.contacts.get(email) + + def upsert_active_contact( + self, + *, + email: str, + first_name: str | None = None, + last_name: str | None = None, + data: dict[str, Any] | None = None, + existing_contact: dict[str, Any] | None | object = None, + ) -> dict[str, Any]: + self.upserts.append( + { + "email": email, + "first_name": first_name, + "last_name": last_name, + "data": data, + "existing_contact": existing_contact, + } + ) + return {"id": str(len(self.upserts))} + + FakeBrevoClient.contacts = {} + FakeBrevoClient.subscriptions = [] + FakeKeilaClient.contacts = {} + FakeKeilaClient.upserts = [] + monkeypatch.setattr("five08.agent.tools.EspoClient", FakeEspoClient) monkeypatch.setattr("five08.agent.tools.AuthentikClient", FakeAuthentikClient) monkeypatch.setattr("five08.agent.tools.MigaduClient", FakeMigaduClient) monkeypatch.setattr("five08.agent.tools.OutlineClient", FakeOutlineClient) + monkeypatch.setattr("five08.newsletter_sync.BrevoClient", FakeBrevoClient) + monkeypatch.setattr("five08.newsletter_sync.KeilaClient", FakeKeilaClient) return SimpleNamespace( espo=FakeEspoClient, authentik=FakeAuthentikClient, migadu=FakeMigaduClient, outline=FakeOutlineClient, + brevo=FakeBrevoClient, + keila=FakeKeilaClient, events=events, ) @@ -2451,6 +2528,162 @@ def test_user_accounts_tool_executes_all_steps( assert ("contact-1", {"cSsoID": "42"}) in fakes.espo.updates +def test_user_accounts_tool_subscribes_mailbox_and_backup_email_to_brevo( + monkeypatch: pytest.MonkeyPatch, +) -> None: + fakes = _install_account_tool_fakes(monkeypatch) + registry = ToolRegistry(runtime_config=_account_runtime_config(brevo_api_key="key")) + + result = registry.execute( + "account_write.create_user_accounts", + {"contact_id": "contact-1", "mailbox_username": "jane@508.dev"}, + organization_id="org-1", + actor_id="123", + actor_scopes={ + "mailbox:create", + "user:manage", + "integration:manage", + "crm:contact:read", + "crm:contact:update", + }, + ) + + assert result["mailbox"]["newsletter_subscribed"] is True + assert result["mailbox"]["newsletter_error"] is None + assert fakes.brevo.subscriptions == [ + {"email": "jane@508.dev", "list_id": 4}, + {"email": "jane@example.com", "list_id": 4}, + ] + + +def test_user_accounts_tool_reads_dynamic_newsletter_runtime_config( + monkeypatch: pytest.MonkeyPatch, +) -> None: + fakes = _install_account_tool_fakes(monkeypatch) + runtime_values = {"brevo_api_key": None} + registry = ToolRegistry( + runtime_config_factory=lambda: _account_runtime_config( + brevo_api_key=runtime_values["brevo_api_key"] + ) + ) + runtime_values["brevo_api_key"] = "key" + + result = registry.execute( + "account_write.create_user_accounts", + {"contact_id": "contact-1", "mailbox_username": "jane@508.dev"}, + organization_id="org-1", + actor_id="123", + actor_scopes={ + "mailbox:create", + "user:manage", + "integration:manage", + "crm:contact:read", + "crm:contact:update", + }, + ) + + assert result["mailbox"]["newsletter_subscribed"] is True + assert fakes.brevo.subscriptions == [ + {"email": "jane@508.dev", "list_id": 4}, + {"email": "jane@example.com", "list_id": 4}, + ] + + +def test_user_accounts_tool_reports_suppressed_newsletter_contact( + monkeypatch: pytest.MonkeyPatch, +) -> None: + fakes = _install_account_tool_fakes(monkeypatch) + fakes.brevo.contacts = {"jane@example.com": {"emailBlacklisted": True}} + registry = ToolRegistry(runtime_config=_account_runtime_config(brevo_api_key="key")) + + result = registry.execute( + "account_write.create_user_accounts", + {"contact_id": "contact-1", "mailbox_username": "jane@508.dev"}, + organization_id="org-1", + actor_id="123", + actor_scopes={ + "mailbox:create", + "user:manage", + "integration:manage", + "crm:contact:read", + "crm:contact:update", + }, + ) + + assert result["mailbox"]["newsletter_subscribed"] is False + assert result["mailbox"]["newsletter_error"] == ( + "brevo skipped 1 suppressed contact(s)" + ) + assert fakes.brevo.subscriptions == [{"email": "jane@508.dev", "list_id": 4}] + + +def test_user_accounts_tool_reports_newsletter_sync_exceptions( + monkeypatch: pytest.MonkeyPatch, +) -> None: + _install_account_tool_fakes(monkeypatch) + + def _raise_newsletter_error(*args: object, **kwargs: object) -> dict[str, object]: + raise RuntimeError("provider exploded") + + monkeypatch.setattr( + "five08.agent.tools.sync_newsletter_contacts", + _raise_newsletter_error, + ) + registry = ToolRegistry(runtime_config=_account_runtime_config(brevo_api_key="key")) + + result = registry.execute( + "account_write.create_user_accounts", + {"contact_id": "contact-1", "mailbox_username": "jane@508.dev"}, + organization_id="org-1", + actor_id="123", + actor_scopes={ + "mailbox:create", + "user:manage", + "integration:manage", + "crm:contact:read", + "crm:contact:update", + }, + ) + + assert result["mailbox"]["newsletter_subscribed"] is False + assert result["mailbox"]["newsletter_error"] == ( + "Newsletter sync failed: provider exploded" + ) + + +def test_user_accounts_tool_subscribes_mailbox_and_backup_email_to_keila( + monkeypatch: pytest.MonkeyPatch, +) -> None: + fakes = _install_account_tool_fakes(monkeypatch) + registry = ToolRegistry(runtime_config=_account_runtime_config(keila_api_key="key")) + + result = registry.execute( + "account_write.create_user_accounts", + {"contact_id": "contact-1", "mailbox_username": "jane@508.dev"}, + organization_id="org-1", + actor_id="123", + actor_scopes={ + "mailbox:create", + "user:manage", + "integration:manage", + "crm:contact:read", + "crm:contact:update", + }, + ) + + assert result["mailbox"]["newsletter_subscribed"] is True + assert result["mailbox"]["newsletter_error"] is None + assert [item["email"] for item in fakes.keila.upserts] == [ + "jane@508.dev", + "jane@example.com", + ] + assert fakes.keila.upserts[0]["data"] == { + "audiences": ["508_members"], + "source": "agent_account_creation", + "mailbox_email": "jane@508.dev", + } + + def test_user_accounts_tool_preflights_before_mailbox_creation( monkeypatch: pytest.MonkeyPatch, ) -> None: diff --git a/tests/unit/test_backend_api.py b/tests/unit/test_backend_api.py index 7e3ca110..1c7dc095 100644 --- a/tests/unit/test_backend_api.py +++ b/tests/unit/test_backend_api.py @@ -7030,6 +7030,105 @@ def test_dashboard_sync_people_workflows_engineer_is_dry_run( mock_insert.assert_not_called() +def test_dashboard_sync_newsletters_audits_discord_session(client: TestClient) -> None: + session = api.AuthSession( + subject="123456789", + email="admin@508.dev", + display_name="Discord Admin", + groups=["discord_admin"], + is_admin=True, + id_token="id-token-1", + expires_at=4_102_444_800, + actor_provider=api.ActorProvider.DISCORD.value, + crm_contact_id="contact-123", + ) + + with ( + patch( + "five08.backend.api._current_session", + new_callable=AsyncMock, + return_value=("session-1", session), + ), + patch( + "five08.backend.api._enqueue_newsletter_sync_job", + new_callable=AsyncMock, + return_value=Mock(id="job-newsletter-1", created=True), + ), + patch("five08.backend.api.insert_audit_event") as mock_insert, + ): + response = client.post("/dashboard/api/sync/newsletters") + + assert response.status_code == 202 + assert response.json()["job_id"] == "job-newsletter-1" + audit_payload = mock_insert.call_args.args[1] + assert audit_payload.source == api.AuditSource.ADMIN_DASHBOARD + assert audit_payload.action == "newsletter.508_members_sync" + assert audit_payload.result == api.AuditResult.SUCCESS + assert audit_payload.actor_provider == api.ActorProvider.DISCORD + assert audit_payload.actor_subject == "123456789" + assert audit_payload.resource_type == "newsletter_sync" + assert audit_payload.resource_id == "job-newsletter-1" + assert audit_payload.metadata is not None + assert audit_payload.metadata["source"] == "dashboard" + + +@pytest.mark.asyncio +async def test_manual_newsletter_sync_idempotency_keys_are_unique() -> None: + with patch( + "five08.backend.api.enqueue_job", + side_effect=[ + Mock(id="job-newsletter-1", created=True), + Mock(id="job-newsletter-2", created=True), + ], + ) as mock_enqueue: + await api._enqueue_newsletter_sync_job(Mock(), reason="dashboard") + await api._enqueue_newsletter_sync_job(Mock(), reason="dashboard") + + keys = [item.kwargs["idempotency_key"] for item in mock_enqueue.call_args_list] + assert keys[0] != keys[1] + assert all(key.startswith("newsletter-sync:508-members:dashboard:") for key in keys) + + +def test_dashboard_sync_newsletters_workflows_engineer_is_dry_run( + client: TestClient, +) -> None: + session = api.AuthSession( + subject="workflows-1", + email="workflows@508.dev", + display_name="Workflows Engineer", + groups=["Workflows Engineer"], + is_admin=False, + id_token="", + expires_at=4_102_444_800, + actor_provider=api.ActorProvider.DISCORD.value, + ) + + with ( + patch( + "five08.backend.api._current_session", + new_callable=AsyncMock, + return_value=("session-1", session), + ), + patch( + "five08.backend.api._enqueue_newsletter_sync_job", + new_callable=AsyncMock, + ) as mock_enqueue, + patch("five08.backend.api.insert_audit_event") as mock_insert, + ): + response = client.post("/dashboard/api/sync/newsletters") + + assert response.status_code == 200 + assert response.json()["status"] == "dry_run" + assert response.json()["would_enqueue"]["job_type"] == ( + "sync_508_members_newsletters_job" + ) + assert response.json()["would_enqueue"]["idempotency_key_pattern"] == ( + "newsletter-sync:508-members:dashboard::" + ) + mock_enqueue.assert_not_called() + mock_insert.assert_not_called() + + def test_dashboard_sync_projects_workflows_engineer_is_dry_run( client: TestClient, ) -> None: diff --git a/tests/unit/test_brevo_client.py b/tests/unit/test_brevo_client.py new file mode 100644 index 00000000..36c34279 --- /dev/null +++ b/tests/unit/test_brevo_client.py @@ -0,0 +1,199 @@ +"""Unit tests for the shared Brevo API client.""" + +from unittest.mock import Mock, patch + +import pytest +import requests + +from five08.clients.brevo import BrevoAPIError, BrevoClient + + +def test_add_contact_to_list_posts_create_or_update_payload() -> None: + response = Mock() + response.status_code = 201 + response.content = b'{"id": 21}' + response.json.return_value = {"id": 21} + + with patch("five08.clients.brevo.requests.post", return_value=response) as post: + result = BrevoClient( + api_key="brevo-key", + timeout_seconds=7.0, + ).add_contact_to_list(email="Jane@Example.com", list_id=4) + + post.assert_called_once_with( + "https://api.brevo.com/v3/contacts", + headers={ + "Accept": "application/json", + "Content-Type": "application/json", + "api-key": "brevo-key", + }, + json={ + "email": "jane@example.com", + "listIds": [4], + "updateEnabled": True, + }, + timeout=7.0, + ) + assert result == {"id": 21} + + +def test_add_contact_to_list_accepts_empty_success_body() -> None: + response = Mock() + response.status_code = 204 + response.content = b"" + + with patch("five08.clients.brevo.requests.post", return_value=response): + result = BrevoClient(api_key="brevo-key").add_contact_to_list( + email="jane@example.com", + list_id=4, + ) + + assert result == {} + + +def test_add_contact_to_list_raises_on_request_error() -> None: + with patch( + "five08.clients.brevo.requests.post", + side_effect=requests.Timeout("timed out"), + ): + with pytest.raises(BrevoAPIError, match="request failed"): + BrevoClient(api_key="brevo-key").add_contact_to_list( + email="jane@example.com", + list_id=4, + ) + + +def test_add_contact_to_list_truncates_error_response_body() -> None: + response = Mock() + response.status_code = 400 + response.text = f"email=jane@example.com {'x' * 600}" + + with patch("five08.clients.brevo.requests.post", return_value=response): + with pytest.raises(BrevoAPIError) as exc_info: + BrevoClient(api_key="brevo-key").add_contact_to_list( + email="jane@example.com", + list_id=4, + ) + + message = str(exc_info.value) + assert "jane@example.com" not in message + assert "[redacted-email]" in message + assert len(message) < 600 + + +def test_get_contact_fetches_contact_by_email() -> None: + response = Mock() + response.status_code = 200 + response.json.return_value = {"email": "jane@example.com"} + + with patch("five08.clients.brevo.requests.get", return_value=response) as get: + result = BrevoClient(api_key="brevo-key").get_contact("Jane@Example.com") + + get.assert_called_once_with( + "https://api.brevo.com/v3/contacts/jane%40example.com", + headers={"Accept": "application/json", "api-key": "brevo-key"}, + timeout=20.0, + ) + assert result == {"email": "jane@example.com"} + + +def test_get_contact_url_encodes_plus_in_email() -> None: + response = Mock() + response.status_code = 200 + response.json.return_value = {"email": "jane+tag@example.com"} + + with patch("five08.clients.brevo.requests.get", return_value=response) as get: + result = BrevoClient(api_key="brevo-key").get_contact("Jane+Tag@Example.com") + + get.assert_called_once_with( + "https://api.brevo.com/v3/contacts/jane%2Btag%40example.com", + headers={"Accept": "application/json", "api-key": "brevo-key"}, + timeout=20.0, + ) + assert result == {"email": "jane+tag@example.com"} + + +def test_get_contact_returns_none_for_missing_contact() -> None: + response = Mock() + response.status_code = 404 + + with patch("five08.clients.brevo.requests.get", return_value=response): + result = BrevoClient(api_key="brevo-key").get_contact("jane@example.com") + + assert result is None + + +@pytest.mark.parametrize("email", ["", "not-an-email", "a@b", "jane @example.com"]) +def test_get_contact_rejects_invalid_email(email: str) -> None: + with pytest.raises(ValueError, match="full email address"): + BrevoClient(api_key="brevo-key").get_contact(email) + + +@pytest.mark.parametrize("email", ["", "not-an-email", "a@b", "jane @example.com"]) +def test_add_contact_to_list_rejects_invalid_email(email: str) -> None: + with pytest.raises(ValueError, match="full email address"): + BrevoClient(api_key="brevo-key").add_contact_to_list(email=email, list_id=4) + + +def test_get_contact_truncates_error_response_body() -> None: + response = Mock() + response.status_code = 500 + response.text = f"email=jane@example.com {'x' * 600}" + + with patch("five08.clients.brevo.requests.get", return_value=response): + with pytest.raises(BrevoAPIError) as exc_info: + BrevoClient(api_key="brevo-key").get_contact("jane@example.com") + + message = str(exc_info.value) + assert "jane@example.com" not in message + assert "[redacted-email]" in message + assert len(message) < 600 + + +def test_find_list_id_by_name_gets_matching_list() -> None: + response = Mock() + response.status_code = 200 + response.json.return_value = { + "count": 2, + "lists": [ + {"id": 3, "name": "Other"}, + {"id": 4, "name": "508 members"}, + ], + } + + with patch("five08.clients.brevo.requests.get", return_value=response) as get: + list_id = BrevoClient(api_key="brevo-key").find_list_id_by_name("508 Members") + + get.assert_called_once_with( + "https://api.brevo.com/v3/contacts/lists", + headers={"Accept": "application/json", "api-key": "brevo-key"}, + params={"limit": 50, "offset": 0, "sort": "asc"}, + timeout=20.0, + ) + assert list_id == 4 + + +def test_find_list_id_by_name_returns_none_when_missing() -> None: + response = Mock() + response.status_code = 200 + response.json.return_value = {"count": 1, "lists": [{"id": 3, "name": "Other"}]} + + with patch("five08.clients.brevo.requests.get", return_value=response): + list_id = BrevoClient(api_key="brevo-key").find_list_id_by_name("508 members") + + assert list_id is None + + +def test_find_list_id_by_name_truncates_error_response_body() -> None: + response = Mock() + response.status_code = 503 + response.text = f"email=jane@example.com {'x' * 600}" + + with patch("five08.clients.brevo.requests.get", return_value=response): + with pytest.raises(BrevoAPIError) as exc_info: + BrevoClient(api_key="brevo-key").find_list_id_by_name("508 members") + + message = str(exc_info.value) + assert "jane@example.com" not in message + assert "[redacted-email]" in message + assert len(message) < 600 diff --git a/tests/unit/test_crm_create_sso_user.py b/tests/unit/test_crm_create_sso_user.py index 7923f676..b9ea8c8c 100644 --- a/tests/unit/test_crm_create_sso_user.py +++ b/tests/unit/test_crm_create_sso_user.py @@ -44,6 +44,21 @@ def cog(mock_espo_api: Mock) -> CRMCog: return CRMCog(Mock()) +@pytest.mark.asyncio +async def test_add_emails_to_newsletter_returns_warning_on_unexpected_error( + cog: CRMCog, +) -> None: + with patch( + "five08.discord_bot.cogs.crm.sync_newsletter_contacts", + side_effect=RuntimeError("provider `exploded`"), + ): + warning = await cog._add_emails_to_newsletter( + ["jane@508.dev", "jane@example.com"] + ) + + assert warning == "Newsletter sync failed: provider 'exploded'" + + @pytest.mark.asyncio async def test_create_sso_user_creates_links_and_sends_recovery_email( cog: CRMCog, mock_interaction: AsyncMock, mock_espo_api: Mock @@ -521,6 +536,11 @@ async def test_create_user_accounts_creates_mailbox_sso_and_outline_invite( patch.object(cog, "_migadu_client", return_value=migadu_client), patch.object(cog, "_authentik_client", return_value=authentik_client), patch.object(cog, "_outline_client", return_value=outline_client), + patch.object( + cog, + "_add_emails_to_newsletter", + new=AsyncMock(return_value=None), + ) as mock_newsletter, patch.object(cog, "_audit_command_safe") as mock_audit, ): mock_espo_api.request.return_value = {"id": "crm-123"} @@ -546,6 +566,9 @@ async def test_create_user_accounts_creates_mailbox_sso_and_outline_invite( name="Jane Doe", role="member", ) + mock_newsletter.assert_awaited_once_with( + ["jane@508.dev", "jane.personal@example.com"] + ) assert mock_espo_api.request.call_args_list[0].args == ( "PUT", "Contact/crm-123", @@ -560,6 +583,7 @@ async def test_create_user_accounts_creates_mailbox_sso_and_outline_invite( assert "User accounts are ready" in message assert "Email: `jane@508.dev`" in message assert "Outline invite: sent." in message + assert "Newsletter: added mailbox and backup email." in message assert mock_interaction.followup.send.call_args.kwargs["ephemeral"] is True assert mock_audit.call_args.kwargs["metadata"]["outline_invited"] is True diff --git a/tests/unit/test_keila_client.py b/tests/unit/test_keila_client.py new file mode 100644 index 00000000..077fd26c --- /dev/null +++ b/tests/unit/test_keila_client.py @@ -0,0 +1,263 @@ +"""Unit tests for the shared Keila API client.""" + +from unittest.mock import Mock, patch + +import pytest +import requests + +from five08.clients.keila import KeilaAPIError, KeilaClient + + +def test_get_contact_by_email_fetches_contact() -> None: + response = Mock() + response.status_code = 200 + response.content = b'{"data":{"id":"contact-1","email":"jane@example.com"}}' + response.json.return_value = { + "data": {"id": "contact-1", "email": "jane@example.com"} + } + + with patch( + "five08.clients.keila.requests.request", return_value=response + ) as request: + result = KeilaClient(api_key="keila-key").get_contact_by_email( + "Jane@Example.com" + ) + + request.assert_called_once_with( + "GET", + "https://app.keila.io/api/v1/contacts/jane%40example.com", + headers={ + "Accept": "application/json", + "Authorization": "Bearer keila-key", + }, + params={"id_type": "email"}, + json=None, + timeout=20.0, + ) + assert result == {"id": "contact-1", "email": "jane@example.com"} + + +def test_get_contact_by_email_url_encodes_plus_in_email() -> None: + response = Mock() + response.status_code = 200 + response.content = b'{"data":{"id":"contact-1","email":"jane+tag@example.com"}}' + response.json.return_value = { + "data": {"id": "contact-1", "email": "jane+tag@example.com"} + } + + with patch( + "five08.clients.keila.requests.request", return_value=response + ) as request: + result = KeilaClient(api_key="keila-key").get_contact_by_email( + "Jane+Tag@Example.com" + ) + + request.assert_called_once_with( + "GET", + "https://app.keila.io/api/v1/contacts/jane%2Btag%40example.com", + headers={ + "Accept": "application/json", + "Authorization": "Bearer keila-key", + }, + params={"id_type": "email"}, + json=None, + timeout=20.0, + ) + assert result == {"id": "contact-1", "email": "jane+tag@example.com"} + + +def test_get_contact_by_email_returns_none_for_missing_contact() -> None: + response = Mock() + response.status_code = 404 + + with patch("five08.clients.keila.requests.request", return_value=response): + result = KeilaClient(api_key="keila-key").get_contact_by_email( + "jane@example.com" + ) + + assert result is None + + +@pytest.mark.parametrize("email", ["", "not-an-email", "a@b", "jane @example.com"]) +def test_get_contact_by_email_rejects_invalid_email(email: str) -> None: + with pytest.raises(ValueError, match="full email address"): + KeilaClient(api_key="keila-key").get_contact_by_email(email) + + +@pytest.mark.parametrize("email", ["", "not-an-email", "a@b", "jane @example.com"]) +def test_upsert_active_contact_rejects_invalid_email(email: str) -> None: + with pytest.raises(ValueError, match="full email address"): + KeilaClient(api_key="keila-key").upsert_active_contact( + email=email, + data={"audiences": ["508_members"]}, + existing_contact=None, + ) + + +def test_upsert_active_contact_creates_missing_contact() -> None: + missing = Mock() + missing.status_code = 404 + created = Mock() + created.status_code = 201 + created.content = b'{"data":{"id":"contact-1"}}' + created.json.return_value = {"data": {"id": "contact-1"}} + + with patch( + "five08.clients.keila.requests.request", + side_effect=[missing, created], + ) as request: + result = KeilaClient(api_key="keila-key").upsert_active_contact( + email="jane@example.com", + first_name="Jane", + last_name="Doe", + data={"audiences": ["508_members"]}, + ) + + assert request.call_args_list[1].kwargs["json"] == { + "data": { + "email": "jane@example.com", + "status": "active", + "data": {"audiences": ["508_members"]}, + "first_name": "Jane", + "last_name": "Doe", + } + } + assert result == {"id": "contact-1"} + + +def test_upsert_active_contact_updates_existing_contact_without_status() -> None: + existing = Mock() + existing.status_code = 200 + existing.content = b'{"data":{"id":"contact-1","status":"active"}}' + existing.json.return_value = {"data": {"id": "contact-1", "status": "active"}} + updated = Mock() + updated.status_code = 200 + updated.content = b'{"data":{"id":"contact-1"}}' + updated.json.return_value = {"data": {"id": "contact-1"}} + + with patch( + "five08.clients.keila.requests.request", + side_effect=[existing, updated], + ) as request: + result = KeilaClient(api_key="keila-key").upsert_active_contact( + email="jane@example.com", + data={"audiences": ["508_members"]}, + ) + + assert request.call_args_list[1].args == ( + "PATCH", + "https://app.keila.io/api/v1/contacts/contact-1", + ) + assert request.call_args_list[1].kwargs["json"] == { + "data": { + "email": "jane@example.com", + "data": {"audiences": ["508_members"]}, + } + } + assert result == {"id": "contact-1"} + + +def test_upsert_active_contact_preserves_existing_contact_data() -> None: + existing = Mock() + existing.status_code = 200 + existing.content = ( + b'{"data":{"id":"contact-1","status":"active",' + b'"data":{"form":"member-intake","audiences":["old"]}}}' + ) + existing.json.return_value = { + "data": { + "id": "contact-1", + "status": "active", + "data": {"form": "member-intake", "audiences": ["old"]}, + } + } + updated = Mock() + updated.status_code = 200 + updated.content = b'{"data":{"id":"contact-1"}}' + updated.json.return_value = {"data": {"id": "contact-1"}} + + with patch( + "five08.clients.keila.requests.request", + side_effect=[existing, updated], + ) as request: + result = KeilaClient(api_key="keila-key").upsert_active_contact( + email="jane@example.com", + data={"audiences": ["508_members"], "mailbox_email": "jane@example.com"}, + ) + + assert request.call_args_list[1].kwargs["json"] == { + "data": { + "email": "jane@example.com", + "data": { + "form": "member-intake", + "audiences": ["old", "508_members"], + "mailbox_email": "jane@example.com", + }, + } + } + assert result == {"id": "contact-1"} + + +def test_upsert_active_contact_uses_supplied_existing_contact_without_lookup() -> None: + updated = Mock() + updated.status_code = 200 + updated.content = b'{"data":{"id":"contact-1"}}' + updated.json.return_value = {"data": {"id": "contact-1"}} + + with patch( + "five08.clients.keila.requests.request", + return_value=updated, + ) as request: + result = KeilaClient(api_key="keila-key").upsert_active_contact( + email="jane@example.com", + data={"audiences": ["508_members"]}, + existing_contact={"id": "contact-1", "email": "jane@example.com"}, + ) + + request.assert_called_once() + assert request.call_args.args == ( + "PATCH", + "https://app.keila.io/api/v1/contacts/contact-1", + ) + assert result == {"id": "contact-1"} + + +def test_upsert_active_contact_requires_existing_contact_id() -> None: + existing = Mock() + existing.status_code = 200 + existing.content = b'{"data":{"email":"jane@example.com","status":"active"}}' + existing.json.return_value = { + "data": {"email": "jane@example.com", "status": "active"} + } + + with patch("five08.clients.keila.requests.request", return_value=existing): + with pytest.raises(KeilaAPIError, match="missing id"): + KeilaClient(api_key="keila-key").upsert_active_contact( + email="jane@example.com", + data={"audiences": ["508_members"]}, + ) + + +def test_keila_client_raises_on_request_error() -> None: + with patch( + "five08.clients.keila.requests.request", + side_effect=requests.Timeout("timed out"), + ): + with pytest.raises(KeilaAPIError, match="request failed"): + KeilaClient(api_key="keila-key").get_contact_by_email("jane@example.com") + + +def test_keila_client_truncates_error_response_body() -> None: + response = Mock() + response.status_code = 500 + response.text = f"email=jane@example.com {'x' * 600}" + response.content = response.text.encode() + + with patch("five08.clients.keila.requests.request", return_value=response): + with pytest.raises(KeilaAPIError) as exc_info: + KeilaClient(api_key="keila-key").get_contact_by_email("jane@example.com") + + message = str(exc_info.value) + assert "jane@example.com" not in message + assert "[redacted-email]" in message + assert len(message) < 600 diff --git a/tests/unit/test_migadu_client.py b/tests/unit/test_migadu_client.py new file mode 100644 index 00000000..b1f7063c --- /dev/null +++ b/tests/unit/test_migadu_client.py @@ -0,0 +1,26 @@ +"""Unit tests for the shared Migadu API client.""" + +from unittest.mock import Mock, patch + +import pytest + +from five08.clients.migadu import MigaduAPIError, MigaduClient + + +def test_list_mailboxes_truncates_error_response_body() -> None: + response = Mock() + response.status_code = 500 + response.text = f"email=jane@508.dev {'x' * 600}" + + with patch("five08.clients.migadu.requests.get", return_value=response): + with pytest.raises(MigaduAPIError) as exc_info: + MigaduClient( + username="migadu-user", + api_key="migadu-key", + domain="508.dev", + ).list_mailboxes() + + message = str(exc_info.value) + assert "jane@508.dev" not in message + assert "[redacted-email]" in message + assert len(message) < 600 diff --git a/tests/unit/test_migadu_create_mailbox.py b/tests/unit/test_migadu_create_mailbox.py index c84c4b1d..11cb3aa0 100644 --- a/tests/unit/test_migadu_create_mailbox.py +++ b/tests/unit/test_migadu_create_mailbox.py @@ -32,6 +32,11 @@ def migadu_cog(mock_bot: Mock) -> MigaduCog: mock_settings.migadu_api_user = "migadu-user" mock_settings.migadu_api_key = "migadu-key" mock_settings.migadu_mailbox_domain = "508.dev" + mock_settings.brevo_api_key = None + mock_settings.brevo_api_base_url = "https://api.brevo.com/v3" + mock_settings.brevo_api_timeout_seconds = 20.0 + mock_settings.brevo_508_members_newsletter_list_id = None + mock_settings.brevo_508_members_newsletter_list_name = "508 members" cog = MigaduCog(mock_bot) cog.espo_api = Mock() return cog @@ -70,6 +75,21 @@ def test_normalize_mailbox_request_rejects_non_508_domain( migadu_cog._normalize_mailbox_request("alice@gmail.com") +@pytest.mark.asyncio +async def test_add_emails_to_newsletter_returns_warning_on_unexpected_error( + migadu_cog: MigaduCog, +) -> None: + with patch( + "five08.discord_bot.cogs.migadu.sync_newsletter_contacts", + side_effect=RuntimeError("provider exploded"), + ): + warning = await migadu_cog._add_emails_to_newsletter( + ["alice@508.dev", "alice@gmail.com"] + ) + + assert warning == "Newsletter sync failed: provider exploded" + + @pytest.mark.asyncio async def test_create_mailbox_command_success_with_crm_defaults_and_sync( migadu_cog: MigaduCog, @@ -90,6 +110,7 @@ async def test_create_mailbox_command_success_with_crm_defaults_and_sync( migadu_cog._create_migadu_mailbox = AsyncMock( return_value={"address": "alice@508.dev"} ) + migadu_cog._add_emails_to_newsletter = AsyncMock(return_value=None) await migadu_cog.create_mailbox.callback( migadu_cog, @@ -107,6 +128,9 @@ async def test_create_mailbox_command_success_with_crm_defaults_and_sync( "contact-1", {"c508Email": "alice@508.dev"}, ) + migadu_cog._add_emails_to_newsletter.assert_awaited_once_with( + ["alice@508.dev", "alice@gmail.com"] + ) _args, kwargs = mock_interaction.followup.send.call_args assert kwargs["embed"].title == "✅ Mailbox Created" diff --git a/tests/unit/test_newsletter_sync.py b/tests/unit/test_newsletter_sync.py new file mode 100644 index 00000000..d3de09c7 --- /dev/null +++ b/tests/unit/test_newsletter_sync.py @@ -0,0 +1,501 @@ +"""Unit tests for Migadu-backed newsletter audience sync.""" + +from __future__ import annotations + +from types import SimpleNamespace +from typing import Any + +import pytest + +from five08.clients.migadu import MigaduMailbox +from five08.clients.espo import EspoAPIError +from five08.newsletter_sync import ( + NewsletterSyncProcessor, + build_newsletter_providers, + format_newsletter_sync_warning, + sync_newsletter_contacts, +) + + +class FakeMigaduClient: + mailboxes: list[MigaduMailbox] = [] + + def __init__( + self, + *, + username: str, + api_key: str, + domain: str, + ) -> None: + self.username = username + self.api_key = api_key + self.domain = domain + + def list_mailboxes(self) -> list[MigaduMailbox]: + return list(self.mailboxes) + + +class FakeBrevoClient: + contacts: dict[str, dict[str, Any]] = {} + subscriptions: list[dict[str, Any]] = [] + list_lookup_names: list[str] = [] + contact_lookup_emails: list[str] = [] + + def __init__( + self, + *, + api_key: str, + base_url: str = "https://api.brevo.com/v3", + timeout_seconds: float = 20.0, + ) -> None: + self.api_key = api_key + self.base_url = base_url + self.timeout_seconds = timeout_seconds + + def get_contact(self, email: str) -> dict[str, Any] | None: + self.contact_lookup_emails.append(email) + return self.contacts.get(email) + + def add_contact_to_list(self, *, email: str, list_id: int) -> dict[str, Any]: + self.subscriptions.append({"email": email, "list_id": list_id}) + return {"id": len(self.subscriptions)} + + def find_list_id_by_name(self, name: str) -> int | None: + self.list_lookup_names.append(name) + return 4 if name == "508 members" else None + + +class FakeKeilaClient: + contacts: dict[str, dict[str, Any]] = {} + upserts: list[dict[str, Any]] = [] + lookups: list[str] = [] + + def __init__( + self, + *, + api_key: str, + base_url: str = "https://app.keila.io", + timeout_seconds: float = 20.0, + ) -> None: + self.api_key = api_key + self.base_url = base_url + self.timeout_seconds = timeout_seconds + + def get_contact_by_email(self, email: str) -> dict[str, Any] | None: + self.lookups.append(email) + return self.contacts.get(email) + + def upsert_active_contact( + self, + *, + email: str, + first_name: str | None = None, + last_name: str | None = None, + data: dict[str, Any] | None = None, + existing_contact: dict[str, Any] | None | object = None, + ) -> dict[str, Any]: + self.upserts.append( + { + "email": email, + "first_name": first_name, + "last_name": last_name, + "data": data, + "existing_contact": existing_contact, + } + ) + return {"id": str(len(self.upserts))} + + +class FakeEspoClient: + contacts: list[dict[str, Any]] = [] + raise_error = False + + def __init__( + self, + base_url: str, + api_key: str, + timeout_seconds: float = 20.0, + ) -> None: + self.base_url = base_url + self.api_key = api_key + self.timeout_seconds = timeout_seconds + + def list_contacts(self, params: dict[str, Any]) -> dict[str, Any]: + if self.raise_error: + raise EspoAPIError("CRM unavailable") + return {"list": [dict(item) for item in self.contacts]} + + +@pytest.fixture(autouse=True) +def reset_fakes(monkeypatch: pytest.MonkeyPatch) -> None: + FakeMigaduClient.mailboxes = [] + FakeBrevoClient.contacts = {} + FakeBrevoClient.subscriptions = [] + FakeBrevoClient.list_lookup_names = [] + FakeBrevoClient.contact_lookup_emails = [] + FakeKeilaClient.contacts = {} + FakeKeilaClient.upserts = [] + FakeKeilaClient.lookups = [] + FakeEspoClient.contacts = [] + FakeEspoClient.raise_error = False + monkeypatch.setattr("five08.newsletter_sync.MigaduClient", FakeMigaduClient) + monkeypatch.setattr("five08.newsletter_sync.BrevoClient", FakeBrevoClient) + monkeypatch.setattr("five08.newsletter_sync.KeilaClient", FakeKeilaClient) + monkeypatch.setattr("five08.newsletter_sync.EspoClient", FakeEspoClient) + + +def _settings(**overrides: Any) -> SimpleNamespace: + values: dict[str, Any] = { + "migadu_api_user": "migadu-user", + "migadu_api_key": "migadu-key", + "migadu_mailbox_domain": "508.dev", + "brevo_api_key": "brevo-key", + "brevo_508_members_newsletter_list_id": 4, + "keila_api_key": "keila-key", + "newsletter_sync_excluded_mailboxes": "system", + } + values.update(overrides) + return SimpleNamespace(**values) + + +def test_sync_508_members_adds_mailbox_and_backup_email_to_configured_providers() -> ( + None +): + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ), + MigaduMailbox( + address="system@508.dev", + name="System", + password_recovery_email="ops@example.com", + ), + ] + + result = NewsletterSyncProcessor(_settings()).sync_508_members() + + assert result["mailboxes_scanned"] == 2 + assert result["system_mailboxes_skipped"] == 1 + assert result["contacts_considered"] == 2 + assert FakeBrevoClient.subscriptions == [ + {"email": "jane@508.dev", "list_id": 4}, + {"email": "jane@example.com", "list_id": 4}, + ] + assert [item["email"] for item in FakeKeilaClient.upserts] == [ + "jane@508.dev", + "jane@example.com", + ] + assert result["providers"]["brevo"]["synced"] == 2 + assert result["providers"]["keila"]["synced"] == 2 + + +def test_sync_508_members_skips_provider_suppressed_contacts() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + FakeBrevoClient.contacts = {"jane@example.com": {"emailBlacklisted": True}} + FakeKeilaClient.contacts = {"jane@example.com": {"status": "unsubscribed"}} + + result = NewsletterSyncProcessor(_settings()).sync_508_members() + + assert FakeBrevoClient.subscriptions == [{"email": "jane@508.dev", "list_id": 4}] + assert [item["email"] for item in FakeKeilaClient.upserts] == ["jane@508.dev"] + assert result["providers"]["brevo"]["statuses"] == { + "synced": 1, + "skipped_provider_suppressed": 1, + } + assert result["providers"]["keila"]["statuses"] == { + "synced": 1, + "skipped_provider_suppressed": 1, + } + + +def test_sync_508_members_allows_sms_only_brevo_blacklist() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email=None, + ) + ] + FakeBrevoClient.contacts = { + "jane@508.dev": {"smsBlacklisted": True, "emailBlacklisted": False} + } + + result = NewsletterSyncProcessor(_settings(keila_api_key=None)).sync_508_members() + + assert FakeBrevoClient.subscriptions == [{"email": "jane@508.dev", "list_id": 4}] + assert result["providers"]["brevo"]["statuses"] == {"synced": 1} + + +def test_sync_508_members_skips_brevo_list_unsubscribed_contacts() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + FakeBrevoClient.contacts = {"jane@example.com": {"listUnsubscribed": ["4"]}} + + result = NewsletterSyncProcessor(_settings()).sync_508_members() + + assert FakeBrevoClient.subscriptions == [{"email": "jane@508.dev", "list_id": 4}] + assert result["providers"]["brevo"]["statuses"] == { + "synced": 1, + "skipped_provider_suppressed": 1, + } + + +def test_sync_508_members_caches_brevo_list_lookup_by_name() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + + result = NewsletterSyncProcessor( + _settings( + brevo_508_members_newsletter_list_id=None, + brevo_508_members_newsletter_list_name="508 members", + keila_api_key=None, + ) + ).sync_508_members() + + assert result["providers"]["brevo"]["synced"] == 2 + assert FakeBrevoClient.list_lookup_names == ["508 members"] + + +def test_sync_508_members_skips_missing_brevo_list_before_contact_lookup() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + + result = NewsletterSyncProcessor( + _settings( + brevo_508_members_newsletter_list_id=None, + brevo_508_members_newsletter_list_name="Missing list", + keila_api_key=None, + ) + ).sync_508_members() + + assert result["providers"]["brevo"]["synced"] == 0 + assert result["providers"]["brevo"]["statuses"] == {"skipped_list_missing": 2} + assert FakeBrevoClient.list_lookup_names == ["Missing list"] + assert FakeBrevoClient.contact_lookup_emails == [] + + +def test_sync_508_members_avoids_duplicate_keila_contact_lookups() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + + result = NewsletterSyncProcessor(_settings(brevo_api_key=None)).sync_508_members() + + assert result["providers"]["keila"]["synced"] == 2 + assert FakeKeilaClient.lookups == ["jane@508.dev", "jane@example.com"] + assert [item["existing_contact"] for item in FakeKeilaClient.upserts] == [ + None, + None, + ] + + +def test_sync_508_members_skips_crm_blocked_mailboxes() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + FakeEspoClient.contacts = [{"id": "contact-1", "type": "Inactive Member"}] + + result = NewsletterSyncProcessor( + _settings(espo_base_url="https://crm.example", espo_api_key="espo-key") + ).sync_508_members() + + assert result["crm_blocked_skipped"] == 1 + assert result["contacts_considered"] == 0 + assert FakeBrevoClient.subscriptions == [] + assert FakeKeilaClient.upserts == [] + + +def test_sync_508_members_skips_crm_unmatched_mailboxes_when_crm_configured() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="service@508.dev", + name="Service Account", + password_recovery_email="ops@example.com", + ) + ] + + result = NewsletterSyncProcessor( + _settings(espo_base_url="https://crm.example", espo_api_key="espo-key") + ).sync_508_members() + + assert result["crm_unmatched_skipped"] == 1 + assert result["contacts_considered"] == 0 + assert FakeBrevoClient.subscriptions == [] + assert FakeKeilaClient.upserts == [] + + +def test_sync_508_members_syncs_crm_matched_mailboxes_when_crm_configured() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + FakeEspoClient.contacts = [{"id": "contact-1", "type": "Member"}] + + result = NewsletterSyncProcessor( + _settings(espo_base_url="https://crm.example", espo_api_key="espo-key") + ).sync_508_members() + + assert result["crm_unmatched_skipped"] == 0 + assert result["contacts_considered"] == 2 + assert FakeBrevoClient.subscriptions == [ + {"email": "jane@508.dev", "list_id": 4}, + {"email": "jane@example.com", "list_id": 4}, + ] + + +def test_sync_508_members_skips_mailbox_when_any_crm_match_is_blocked() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + FakeEspoClient.contacts = [ + {"id": "contact-1", "type": "Member"}, + {"id": "contact-2", "type": "Inactive Member"}, + ] + + result = NewsletterSyncProcessor( + _settings(espo_base_url="https://crm.example", espo_api_key="espo-key") + ).sync_508_members() + + assert result["crm_blocked_skipped"] == 1 + assert result["contacts_considered"] == 0 + assert FakeBrevoClient.subscriptions == [] + assert FakeKeilaClient.upserts == [] + + +def test_build_newsletter_providers_uses_default_list_name_when_blank() -> None: + providers = build_newsletter_providers( + _settings( + brevo_508_members_newsletter_list_id=None, + brevo_508_members_newsletter_list_name=" ", + ) + ) + + assert len(providers) == 2 + assert providers[0].list_name == "508 members" + + +def test_format_newsletter_sync_warning_reports_suppressed_skips() -> None: + warning = format_newsletter_sync_warning( + { + "providers": { + "brevo": { + "synced": 1, + "skipped": 1, + "failed": 0, + "statuses": {"skipped_provider_suppressed": 1}, + } + } + } + ) + + assert warning == "brevo skipped 1 suppressed contact(s)" + + +def test_format_newsletter_sync_warning_redacts_failure_emails() -> None: + warning = format_newsletter_sync_warning( + { + "providers": { + "keila": { + "failed": 1, + "failures": [ + { + "email": "jane@example.com", + "error": ( + "lookup failed for jane@example.com at " + "/contacts/jane%40example.com" + ), + } + ], + } + } + } + ) + + assert warning == ( + "keila failed for 1 contact(s): lookup failed for [redacted-email] " + "at /contacts/[redacted-email]" + ) + assert "jane@example.com" not in warning + assert "jane%40example.com" not in warning + + +def test_sync_newsletter_contacts_uses_first_email_as_default_mailbox_pointer() -> None: + result = sync_newsletter_contacts( + _settings(brevo_api_key=None), + ["jane@508.dev", "jane@example.com"], + source="test", + ) + + assert result["providers"]["keila"]["synced"] == 2 + assert [item["email"] for item in FakeKeilaClient.upserts] == [ + "jane@508.dev", + "jane@example.com", + ] + assert [item["data"]["mailbox_email"] for item in FakeKeilaClient.upserts] == [ + "jane@508.dev", + "jane@508.dev", + ] + + +def test_sync_508_members_skips_mailbox_when_crm_lookup_fails() -> None: + FakeMigaduClient.mailboxes = [ + MigaduMailbox( + address="jane@508.dev", + name="Jane Doe", + password_recovery_email="jane@example.com", + ) + ] + FakeEspoClient.raise_error = True + + result = NewsletterSyncProcessor( + _settings(espo_base_url="https://crm.example", espo_api_key="espo-key") + ).sync_508_members() + + assert result["crm_lookup_failed_skipped"] == 1 + assert result["contacts_considered"] == 0 + assert result["crm_lookup_failures"] == [ + { + "mailbox": "jane@508.dev", + "error": "CRM contact lookup failed for jane@508.dev: CRM unavailable", + } + ] + assert FakeBrevoClient.subscriptions == [] + assert FakeKeilaClient.upserts == [] diff --git a/tests/unit/test_runtime_config.py b/tests/unit/test_runtime_config.py index 0dd46eab..edd884af 100644 --- a/tests/unit/test_runtime_config.py +++ b/tests/unit/test_runtime_config.py @@ -297,6 +297,8 @@ def test_runtime_config_numeric_bounds_are_preserved( "key", [ "DISCORD_LOGS_WEBHOOK_URL", + "NEWSLETTER_SYNC_ENABLED", + "NEWSLETTER_SYNC_INTERVAL_SECONDS", ], ) def test_startup_bound_runtime_config_is_restart_required(key: str) -> None: @@ -305,6 +307,29 @@ def test_startup_bound_runtime_config_is_restart_required(key: str) -> None: assert definition.restart_required is True +@pytest.mark.parametrize( + "key", + [ + "BREVO_API_KEY", + "BREVO_API_BASE_URL", + "BREVO_API_TIMEOUT_SECONDS", + "BREVO_508_MEMBERS_NEWSLETTER_LIST_ID", + "BREVO_508_MEMBERS_NEWSLETTER_LIST_NAME", + "KEILA_API_KEY", + "KEILA_API_BASE_URL", + "KEILA_API_TIMEOUT_SECONDS", + "NEWSLETTER_SYNC_ENABLED", + "NEWSLETTER_SYNC_INTERVAL_SECONDS", + "NEWSLETTER_SYNC_EXCLUDED_MAILBOXES", + ], +) +def test_newsletter_settings_are_dashboard_configurable(key: str) -> None: + definition = runtime_config_definition_for_key(key) + + assert definition is not None + assert definition.category == "Newsletter" + + def test_core_crm_auth_and_mailbox_settings_are_not_dashboard_configurable() -> None: assert runtime_config_definition_for_key("ESPO_BASE_URL") is None assert runtime_config_definition_for_key("ESPO_API_KEY") is None diff --git a/tests/unit/test_shared_settings.py b/tests/unit/test_shared_settings.py index db5f974c..6802009e 100644 --- a/tests/unit/test_shared_settings.py +++ b/tests/unit/test_shared_settings.py @@ -99,6 +99,11 @@ def test_shared_settings_expose_agent_external_tool_credentials() -> None: migadu_api_user="migadu-user", migadu_api_key="migadu-key", migadu_mailbox_domain="mail.example.com", + brevo_api_key="brevo-key", + brevo_508_members_newsletter_list_id=4, + brevo_508_members_newsletter_list_name="508 members", + keila_api_key="keila-key", + keila_api_base_url="https://keila.example", ) runtime_config = ToolRuntimeConfig.from_settings(settings) @@ -107,6 +112,11 @@ def test_shared_settings_expose_agent_external_tool_credentials() -> None: assert runtime_config.migadu_api_user == "migadu-user" assert runtime_config.migadu_api_key == "migadu-key" assert runtime_config.migadu_mailbox_domain == "mail.example.com" + assert runtime_config.brevo_api_key == "brevo-key" + assert runtime_config.brevo_508_members_newsletter_list_id == 4 + assert runtime_config.brevo_508_members_newsletter_list_name == "508 members" + assert runtime_config.keila_api_key == "keila-key" + assert runtime_config.keila_api_base_url == "https://keila.example" def test_shared_settings_docuseal_template_id_accepts_numeric_string() -> None: @@ -125,6 +135,20 @@ def test_shared_settings_docuseal_template_id_rejects_non_numeric_string() -> No SharedSettings(docuseal_member_agreement_template_id="abc") +def test_shared_settings_brevo_members_list_id_accepts_blank_string_as_none() -> None: + """Blank Brevo list IDs from env should leave list-name lookup enabled.""" + settings = SharedSettings(brevo_508_members_newsletter_list_id=" ") + + assert settings.brevo_508_members_newsletter_list_id is None + + +def test_shared_settings_brevo_members_list_id_accepts_numeric_string() -> None: + """Numeric Brevo list IDs from env should coerce to integers.""" + settings = SharedSettings(brevo_508_members_newsletter_list_id="4") + + assert settings.brevo_508_members_newsletter_list_id == 4 + + def test_local_service_defaults_target_host_runtime( monkeypatch: pytest.MonkeyPatch, ) -> None: diff --git a/tests/unit/test_worker_newsletter_sync.py b/tests/unit/test_worker_newsletter_sync.py new file mode 100644 index 00000000..1aa6324d --- /dev/null +++ b/tests/unit/test_worker_newsletter_sync.py @@ -0,0 +1,65 @@ +"""Unit tests for worker newsletter sync job result handling.""" + +import pytest + +from five08.worker import jobs + + +def test_sync_508_members_newsletters_job_masks_failure_emails( + monkeypatch: pytest.MonkeyPatch, +) -> None: + class FakeNewsletterSyncProcessor: + def __init__(self, settings: object) -> None: + self.settings = settings + + def sync_508_members(self) -> dict[str, object]: + return { + "crm_lookup_failures": [ + { + "mailbox": "jane@508.dev", + "error": "CRM lookup failed for jane@508.dev", + } + ], + "providers": { + "brevo": { + "failures": [ + { + "email": "jane@example.com", + "error": ( + "provider unavailable for /contacts/" + "jane%40example.com" + ), + } + ] + }, + "keila": {"failures": "not-a-list"}, + }, + } + + monkeypatch.setattr( + jobs, + "NewsletterSyncProcessor", + FakeNewsletterSyncProcessor, + ) + + result = jobs.sync_508_members_newsletters_job() + + assert result == { + "crm_lookup_failures": [ + { + "mailbox": "j***@5****...", + "error": "CRM lookup failed for j***@5****...", + } + ], + "providers": { + "brevo": { + "failures": [ + { + "email": "j***@e****...", + "error": ("provider unavailable for /contacts/j***@e****..."), + } + ] + }, + "keila": {"failures": "not-a-list"}, + }, + }