Content inside supertokenslib
(The raw file follows this syntax highlighted file.)
console.log("{{.Version}}")
import * as R from "ramda";import anime from "animejs/lib/anime.es.js";import {create} from "free-style"
const l=console.log;const neon=R.allPass([R.isNotNil,R.complement(R.isEmpty)]);const eon=R.anyPass([R.isNil,R.isEmpty])
const store=(o,...ihs)=>{let s=o;let hs=ihs;return {state:()=>s,setState:(o)=>{
if(R.equals(s,o)){return};s=o;R.map(R.call,hs)
},subscribe:(h)=>hs=R.append(h,hs)}}
const d=document;const dce=(e)=>d.createElement(e);const dctn=(e)=>d.createTextNode(e)
const dhac=(c)=>d.head.appendChild(c);const dbac=(c)=>d.body.appendChild(c);const lcc="st"+crypto.randomUUID()
const mot=R.tap((e)=>{if(eon(e[lcc].added)&&eon(e[lcc].removed)){return}
(new MutationObserver((ms,o)=>R.map((m)=>{
if(R.includes(e,m.addedNodes)&&neon(e[lcc].added)){e[lcc].added(e)}
if(R.includes(e,m.removedNodes)&&neon(e[lcc].removed)){o.disconnect();e[lcc].removed(e)}
})(ms))).observe(window.document,{childList:true,subtree:true})})
const e=(t,a,...cs)=>{ // Based on hscrpt - use 'className' for class and all lowercase for handlers
const el=dce(t);
R.mapObjIndexed((v,k)=>el[k]=v,R.dissoc("lifecycle",a));
el[lcc]={};R.mapObjIndexed((v,k)=>el[lcc][k]=v,a.lifecycle);
R.map((c)=>{if(c){el.appendChild(R.is(String,c) ? dctn(c) : c)}},cs);
return mot(el)}
const ge=(id)=>d.getElementById(id);const valueByID=(id)=>ge(id).value;const disableID=(id)=>ge(id).disabled=true;
const focusIfVisible=(id)=>{if(neon(ge(id))&&(ge(id).offsetParent!=null)){ ge(id).focus() }}
const ssw=(s)=>"{{.Cfg.SuperTokensWebLibBase}}"+s
const addjs=(src)=>{const s=dce("script");s.setAttribute("src", src);s.async=false;dbac(s)}
addjs(ssw("/website.js"));addjs(ssw("/supertokens.js"));addjs(ssw("/session.js"));addjs(ssw("/passwordless.js"));addjs(ssw("/emailverification.js"))
const addcss=(o)=>{const s=create();const r=s.registerStyle(o);const se=dce("style");se.textContent=s.getStyles();dhac(se);return r}
const stWrapperClass=addcss({boxSizing:"border-box",position:"absolute",top:"12px",right:"12px",zIndex:999,
display:"flex",alignItems:"flex-end",flexDirection:"column"})
const stMenuWrapperClass=addcss({backgroundColor:"white",minWidth:"10em",display:"flex",flexDirection:"column",
border:"black solid 1px",paddingTop:"4px",div:{minHeight:"10px",marginBottom:"4px"}})
const stPlainObj={padding:"4px",cursor:"pointer",marginLeft:"12px",fontVariant:"small-caps"}
const stPlainClass=addcss(stPlainObj)
const stPlainMenuClass=addcss(R.mergeRight(stPlainObj,
{marginLeft:"0px",paddingLeft:"12px",display:"inline-block",width:"calc(100% - 16px)","&:hover":{backgroundColor:"#e6e6e6"}}))
const stButtonClass=addcss({width:"32px",height:"32px",padding:"4px",border:0,cursor:"pointer",borderRadius:"16px",marginLeft:"12px"})
const stImgClass=addcss({width:"24px",height:"24px"})
const stFormClass=addcss({display:"flex",flexDirection:"column",padding:"4px",backgroundColor:"white","input": {marginBottom:"10px"}})
const stInputCentered=addcss({textAlign:"center"});const stOpacity0=addcss({opacity:"0%"});const stMarginTop10=addcss({marginTop:"10px"})
const flexItemsCenter={display:"flex",alignItems:"center"};const stFlexItemsCenter=addcss(flexItemsCenter)
const stNonPlainEmailWrapper=addcss(R.mergeRight(flexItemsCenter,{width:"32px",height:"32px",justifyContent:"center"}))
const fadeInOut=(t)=>()=>{anime({targets:t,opacity:["0%","100%"],duration:1000,easing:"easeInOutQuad",complete:()=>{
anime({targets:t,opacity:["100%","0%"],duration:1000,delay:2000,easing:"easeInOutQuad",complete:()=>{anime.remove(t)}})}})}
const truncated=R.replace(/(.)[^@]*(.)@(.).*(.\.[^.]+)/,"$1-$2@$3-$4")
window.onload=()=>{ // TODO - pull out as many funcs as possible
const plain=R.equals("{{.Cfg.PlainUI}}", "true");const prefixedPath=(s)=>"{{.Cfg.APIBasePath}}"+s;
const isLoggedIn=()=>R.equals("{{.LoggedIn}}","true");const userEmail=()=>"{{.UserEmail}}";const userID=()=>"{{.UserID}}"
const uuid="st"+crypto.randomUUID()
const s=store({init:true,label:"...",stState:"init",showMenu:false,menuItems:[]})
const mstate=(s,o)=>s.setState(pre(R.mergeRight(s.state(),o)))
const onEnter=(f)=>(e)=>{if(e.key==="Enter" || e.keyCode===13){f()}}
const callSendOTP=()=>{globalThis.stAuth.sendOTP(valueByID(uuid+"email"));disableID(uuid+"email");disableID(uuid+"emailButton")}
const callVerifyOTP=()=>{globalThis.stAuth.verifyOTP(valueByID(uuid+"otp"));disableID(uuid+"otp");disableID(uuid+"otpButton")}
const handleClose=()=>{s.setState(R.mergeRight(s.state(),{stState:"init",displayState:c.init()}));h.state("loggedOut")}
const c={init:()=>null,
enterEmail:()=>e("div", {className:stFormClass},
e("input", {id:uuid+"email",type:"text",placeholder:"your.email@address.com",onkeyup:onEnter(callSendOTP),className:stInputCentered}),
e("button", {id:uuid+"emailButton",onclick:callSendOTP}, "Request one-time code"),
e("button",{className:stMarginTop10,onclick:handleClose},"Close")),
enterOTP:()=>e("div", {className:stFormClass},
e("input", {id:uuid+"otp",type:"text",placeholder:"One-time code in email",onkeyup:onEnter(callVerifyOTP),className:stInputCentered}),
e("button", {id:uuid+"otpButton",onclick:callVerifyOTP}, "Verify code"),
e("button",{className:stMarginTop10,onclick:handleClose},"Close")),
showEmail:()=>(plain
? null
: e("div", {className:stNonPlainEmailWrapper},
e("div",{id:uuid+"userEmail",className:stOpacity0},userEmail()),
e("button",{className:stButtonClass, onclick:fadeInOut("#"+uuid+"userEmail")},
e("img",{className:stImgClass,src:(isLoggedIn() ? prefixedPath("/person_outline.svg") : null)}))))}
const pre=(o)=>({
"init":R.identity, // TODO - see if we can live without this...
"loggedOut":R.identity,
"checkEmail":R.mergeLeft({displayState:c.enterOTP()}),
"wrongOTP":R.identity, "oldOTP":R.identity, "failedOTP":R.identity,
"signUpSuccess":R.tap(()=>window.location.reload()),
"signInSuccess":R.tap(()=>window.location.reload()),
"loggedIn":R.mergeLeft({displayState:c.showEmail()}),
"logoutCompleted":R.tap(()=>window.location.reload()),
}[o.stState](o))
const handleMenu=()=>{if(!isLoggedIn()){handleClose()}mstate(s,({showMenu:!s.state().showMenu}))}
const handleMenuMouseLeave=()=>mstate(s,({showMenu:false}))
const handle=()=>{handleMenuMouseLeave();if(s.state().stState==="loggedOut"){ mstate(s,{displayState:c.enterEmail()})
}else if(s.state().stState==="loggedIn"){globalThis.stAuth.logout()}}
let etc=null
const checkForMenuClose=(e)=>{if(neon(etc)&&!etc.contains(e.target)){handleMenuMouseLeave()}}
const removeClickOffTracker=(el)=>{etc=el;document.removeEventListener("click",checkForMenuClose,true)}
const addClickOffTracker=(el)=>{etc=el;removeClickOffTracker(el);document.addEventListener("click",checkForMenuClose,true)}
const setMenuItems=(mis)=>mstate(s,{menuItems:R.map(
([label,fn])=>e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:()=>{handleMenuMouseLeave();fn()}},label)),
mis)})
const aui=()=>e("div",{id:uuid,className:stWrapperClass,
style:"opacity:"+(s.state().stState=="init" ? 0: 100)+"%;background:rgba(0,0,0,0)",
onmouseleave:handleMenuMouseLeave,lifecycle:{added:addClickOffTracker,removed:removeClickOffTracker}},
e("button",{style:"width:32px;height:32px;padding:0;border:0;background-color:white;cursor:pointer",onclick:handleMenu},
e("img",{style:"width:32px;height:32px",src:prefixedPath("/menu.svg")})),
s.state().displayState,
(!s.state().showMenu ? null
: e("div",{className:stMenuWrapperClass},
(isLoggedIn() ? null : e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:handle},"login"))),
(!isLoggedIn() ? null : e("div",{className:stPlainMenuClass,style:"cursor:default;font-variant:normal"},
truncated(userEmail()))),
...s.state().menuItems,
(!isLoggedIn() ? null : e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:handle},"logout"))),
)),
)
s.subscribe(()=>{ge(uuid).replaceWith(aui());focusIfVisible(uuid+"email");focusIfVisible(uuid+"otp")})
dbac(aui())
const h={catch:(msg,err)=>showError(msg,err), state:(state,data)=>mstate(s,{stState:state,stData:data})}
const hge=(msg,err)=>h.catch(msg+(err.isSuperTokensGeneralError ? " (some other failure)" : ""),err)
supertokens.init({appInfo:{apiDomain:"{{.Cfg.APIDomain}}",apiBasePath:"{{.Cfg.APIBasePath}}",appName:"{{.Cfg.AppName}}"},
recipeList:[supertokensEmailVerification.init(),supertokensSession.init(),supertokensPasswordless.init()]});
const sendOTP=async(email)=>{try{
const resp=await supertokensPasswordless.createCode({email})
if(resp.status != "SIGN_IN_UP_NOT_ALLOWED"){h.state("checkEmail")}
}catch(err){hge("sendOTP",err)}}
const verifyOTP=async(otp)=>{try{
const resp=await supertokensPasswordless.consumeCode({userInputCode:otp});
if(resp.status === "OK"){
if(resp.createdNewRecipeUser && resp.user.loginMethods.length === 1){h.state("signUpSuccess",resp.user.id)
}else{h.state("signInSuccess",resp.user.id)}
}else if(resp.status === "INCORRECT_USER_INPUT_CODE_ERROR"){h.state("wrongOTP")
}else if(resp.status === "EXPIRED_USER_INPUT_CODE_ERROR"){h.state("oldOTP")
}else{h.state("failedOTP")}
}catch(err){hge("handleOTP",err)}}
const logout=async()=>{await supertokensSession.signOut();h.state("logoutCompleted")}
globalThis.stAuth={sendOTP,verifyOTP,logout,isLoggedIn,userEmail,userID,setMenuItems}
if(neon(globalThis.authHooks)){globalThis.authHooks.forEach(([l,f])=>{const p=h[l];h[l]=(a,b)=>{p(a,b);f(a,b)}})}
if(neon(globalThis.menuItems)){setMenuItems(globalThis.menuItems)}
if(isLoggedIn()){h.state("loggedIn",userEmail())}else{h.state("loggedOut")}
anime({targets:"#"+uuid,opacity:"100%",duration:1000,easing:"easeInOutQuad",complete:()=>anime.remove("#"+uuid)})
}
console.log("{{.Version}}") import * as R from "ramda";import anime from "animejs/lib/anime.es.js";import {create} from "free-style" const l=console.log;const neon=R.allPass([R.isNotNil,R.complement(R.isEmpty)]);const eon=R.anyPass([R.isNil,R.isEmpty]) const store=(o,...ihs)=>{let s=o;let hs=ihs;return {state:()=>s,setState:(o)=>{ if(R.equals(s,o)){return};s=o;R.map(R.call,hs) },subscribe:(h)=>hs=R.append(h,hs)}} const d=document;const dce=(e)=>d.createElement(e);const dctn=(e)=>d.createTextNode(e) const dhac=(c)=>d.head.appendChild(c);const dbac=(c)=>d.body.appendChild(c);const lcc="st"+crypto.randomUUID() const mot=R.tap((e)=>{if(eon(e[lcc].added)&&eon(e[lcc].removed)){return} (new MutationObserver((ms,o)=>R.map((m)=>{ if(R.includes(e,m.addedNodes)&&neon(e[lcc].added)){e[lcc].added(e)} if(R.includes(e,m.removedNodes)&&neon(e[lcc].removed)){o.disconnect();e[lcc].removed(e)} })(ms))).observe(window.document,{childList:true,subtree:true})}) const e=(t,a,...cs)=>{ // Based on hscrpt - use 'className' for class and all lowercase for handlers const el=dce(t); R.mapObjIndexed((v,k)=>el[k]=v,R.dissoc("lifecycle",a)); el[lcc]={};R.mapObjIndexed((v,k)=>el[lcc][k]=v,a.lifecycle); R.map((c)=>{if(c){el.appendChild(R.is(String,c) ? dctn(c) : c)}},cs); return mot(el)} const ge=(id)=>d.getElementById(id);const valueByID=(id)=>ge(id).value;const disableID=(id)=>ge(id).disabled=true; const focusIfVisible=(id)=>{if(neon(ge(id))&&(ge(id).offsetParent!=null)){ ge(id).focus() }} const ssw=(s)=>"{{.Cfg.SuperTokensWebLibBase}}"+s const addjs=(src)=>{const s=dce("script");s.setAttribute("src", src);s.async=false;dbac(s)} addjs(ssw("/website.js"));addjs(ssw("/supertokens.js"));addjs(ssw("/session.js"));addjs(ssw("/passwordless.js"));addjs(ssw("/emailverification.js")) const addcss=(o)=>{const s=create();const r=s.registerStyle(o);const se=dce("style");se.textContent=s.getStyles();dhac(se);return r} const stWrapperClass=addcss({boxSizing:"border-box",position:"absolute",top:"12px",right:"12px",zIndex:999, display:"flex",alignItems:"flex-end",flexDirection:"column"}) const stMenuWrapperClass=addcss({backgroundColor:"white",minWidth:"10em",display:"flex",flexDirection:"column", border:"black solid 1px",paddingTop:"4px",div:{minHeight:"10px",marginBottom:"4px"}}) const stPlainObj={padding:"4px",cursor:"pointer",marginLeft:"12px",fontVariant:"small-caps"} const stPlainClass=addcss(stPlainObj) const stPlainMenuClass=addcss(R.mergeRight(stPlainObj, {marginLeft:"0px",paddingLeft:"12px",display:"inline-block",width:"calc(100% - 16px)","&:hover":{backgroundColor:"#e6e6e6"}})) const stButtonClass=addcss({width:"32px",height:"32px",padding:"4px",border:0,cursor:"pointer",borderRadius:"16px",marginLeft:"12px"}) const stImgClass=addcss({width:"24px",height:"24px"}) const stFormClass=addcss({display:"flex",flexDirection:"column",padding:"4px",backgroundColor:"white","input": {marginBottom:"10px"}}) const stInputCentered=addcss({textAlign:"center"});const stOpacity0=addcss({opacity:"0%"});const stMarginTop10=addcss({marginTop:"10px"}) const flexItemsCenter={display:"flex",alignItems:"center"};const stFlexItemsCenter=addcss(flexItemsCenter) const stNonPlainEmailWrapper=addcss(R.mergeRight(flexItemsCenter,{width:"32px",height:"32px",justifyContent:"center"})) const fadeInOut=(t)=>()=>{anime({targets:t,opacity:["0%","100%"],duration:1000,easing:"easeInOutQuad",complete:()=>{ anime({targets:t,opacity:["100%","0%"],duration:1000,delay:2000,easing:"easeInOutQuad",complete:()=>{anime.remove(t)}})}})} const truncated=R.replace(/(.)[^@]*(.)@(.).*(.\.[^.]+)/,"$1-$2@$3-$4") window.onload=()=>{ // TODO - pull out as many funcs as possible const plain=R.equals("{{.Cfg.PlainUI}}", "true");const prefixedPath=(s)=>"{{.Cfg.APIBasePath}}"+s; const isLoggedIn=()=>R.equals("{{.LoggedIn}}","true");const userEmail=()=>"{{.UserEmail}}";const userID=()=>"{{.UserID}}" const uuid="st"+crypto.randomUUID() const s=store({init:true,label:"...",stState:"init",showMenu:false,menuItems:[]}) const mstate=(s,o)=>s.setState(pre(R.mergeRight(s.state(),o))) const onEnter=(f)=>(e)=>{if(e.key==="Enter" || e.keyCode===13){f()}} const callSendOTP=()=>{globalThis.stAuth.sendOTP(valueByID(uuid+"email"));disableID(uuid+"email");disableID(uuid+"emailButton")} const callVerifyOTP=()=>{globalThis.stAuth.verifyOTP(valueByID(uuid+"otp"));disableID(uuid+"otp");disableID(uuid+"otpButton")} const handleClose=()=>{s.setState(R.mergeRight(s.state(),{stState:"init",displayState:c.init()}));h.state("loggedOut")} const c={init:()=>null, enterEmail:()=>e("div", {className:stFormClass}, e("input", {id:uuid+"email",type:"text",placeholder:"your.email@address.com",onkeyup:onEnter(callSendOTP),className:stInputCentered}), e("button", {id:uuid+"emailButton",onclick:callSendOTP}, "Request one-time code"), e("button",{className:stMarginTop10,onclick:handleClose},"Close")), enterOTP:()=>e("div", {className:stFormClass}, e("input", {id:uuid+"otp",type:"text",placeholder:"One-time code in email",onkeyup:onEnter(callVerifyOTP),className:stInputCentered}), e("button", {id:uuid+"otpButton",onclick:callVerifyOTP}, "Verify code"), e("button",{className:stMarginTop10,onclick:handleClose},"Close")), showEmail:()=>(plain ? null : e("div", {className:stNonPlainEmailWrapper}, e("div",{id:uuid+"userEmail",className:stOpacity0},userEmail()), e("button",{className:stButtonClass, onclick:fadeInOut("#"+uuid+"userEmail")}, e("img",{className:stImgClass,src:(isLoggedIn() ? prefixedPath("/person_outline.svg") : null)}))))} const pre=(o)=>({ "init":R.identity, // TODO - see if we can live without this... "loggedOut":R.identity, "checkEmail":R.mergeLeft({displayState:c.enterOTP()}), "wrongOTP":R.identity, "oldOTP":R.identity, "failedOTP":R.identity, "signUpSuccess":R.tap(()=>window.location.reload()), "signInSuccess":R.tap(()=>window.location.reload()), "loggedIn":R.mergeLeft({displayState:c.showEmail()}), "logoutCompleted":R.tap(()=>window.location.reload()), }[o.stState](o)) const handleMenu=()=>{if(!isLoggedIn()){handleClose()}mstate(s,({showMenu:!s.state().showMenu}))} const handleMenuMouseLeave=()=>mstate(s,({showMenu:false})) const handle=()=>{handleMenuMouseLeave();if(s.state().stState==="loggedOut"){ mstate(s,{displayState:c.enterEmail()}) }else if(s.state().stState==="loggedIn"){globalThis.stAuth.logout()}} let etc=null const checkForMenuClose=(e)=>{if(neon(etc)&&!etc.contains(e.target)){handleMenuMouseLeave()}} const removeClickOffTracker=(el)=>{etc=el;document.removeEventListener("click",checkForMenuClose,true)} const addClickOffTracker=(el)=>{etc=el;removeClickOffTracker(el);document.addEventListener("click",checkForMenuClose,true)} const setMenuItems=(mis)=>mstate(s,{menuItems:R.map( ([label,fn])=>e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:()=>{handleMenuMouseLeave();fn()}},label)), mis)}) const aui=()=>e("div",{id:uuid,className:stWrapperClass, style:"opacity:"+(s.state().stState=="init" ? 0: 100)+"%;background:rgba(0,0,0,0)", onmouseleave:handleMenuMouseLeave,lifecycle:{added:addClickOffTracker,removed:removeClickOffTracker}}, e("button",{style:"width:32px;height:32px;padding:0;border:0;background-color:white;cursor:pointer",onclick:handleMenu}, e("img",{style:"width:32px;height:32px",src:prefixedPath("/menu.svg")})), s.state().displayState, (!s.state().showMenu ? null : e("div",{className:stMenuWrapperClass}, (isLoggedIn() ? null : e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:handle},"login"))), (!isLoggedIn() ? null : e("div",{className:stPlainMenuClass,style:"cursor:default;font-variant:normal"}, truncated(userEmail()))), ...s.state().menuItems, (!isLoggedIn() ? null : e("div",{},e("a",{href:"#",className:stPlainMenuClass,onclick:handle},"logout"))), )), ) s.subscribe(()=>{ge(uuid).replaceWith(aui());focusIfVisible(uuid+"email");focusIfVisible(uuid+"otp")}) dbac(aui()) const h={catch:(msg,err)=>showError(msg,err), state:(state,data)=>mstate(s,{stState:state,stData:data})} const hge=(msg,err)=>h.catch(msg+(err.isSuperTokensGeneralError ? " (some other failure)" : ""),err) supertokens.init({appInfo:{apiDomain:"{{.Cfg.APIDomain}}",apiBasePath:"{{.Cfg.APIBasePath}}",appName:"{{.Cfg.AppName}}"}, recipeList:[supertokensEmailVerification.init(),supertokensSession.init(),supertokensPasswordless.init()]}); const sendOTP=async(email)=>{try{ const resp=await supertokensPasswordless.createCode({email}) if(resp.status != "SIGN_IN_UP_NOT_ALLOWED"){h.state("checkEmail")} }catch(err){hge("sendOTP",err)}} const verifyOTP=async(otp)=>{try{ const resp=await supertokensPasswordless.consumeCode({userInputCode:otp}); if(resp.status === "OK"){ if(resp.createdNewRecipeUser && resp.user.loginMethods.length === 1){h.state("signUpSuccess",resp.user.id) }else{h.state("signInSuccess",resp.user.id)} }else if(resp.status === "INCORRECT_USER_INPUT_CODE_ERROR"){h.state("wrongOTP") }else if(resp.status === "EXPIRED_USER_INPUT_CODE_ERROR"){h.state("oldOTP") }else{h.state("failedOTP")} }catch(err){hge("handleOTP",err)}} const logout=async()=>{await supertokensSession.signOut();h.state("logoutCompleted")} globalThis.stAuth={sendOTP,verifyOTP,logout,isLoggedIn,userEmail,userID,setMenuItems} if(neon(globalThis.authHooks)){globalThis.authHooks.forEach(([l,f])=>{const p=h[l];h[l]=(a,b)=>{p(a,b);f(a,b)}})} if(neon(globalThis.menuItems)){setMenuItems(globalThis.menuItems)} if(isLoggedIn()){h.state("loggedIn",userEmail())}else{h.state("loggedOut")} anime({targets:"#"+uuid,opacity:"100%",duration:1000,easing:"easeInOutQuad",complete:()=>anime.remove("#"+uuid)}) }