// functions in this modeule are used for verifying that a student is who they 
// say they are

function set_shared_or_private(shared_or_private){
    //reads or writes to local storage whether this computer is shared or private
    BookStorage.isPrivateComputer(shared_or_private==="private")
    const location = shared_or_private==="private" ? AutoStorage.STORAGE_TYPE.LOCAL : AutoStorage.STORAGE_TYPE.SESSION
    BookStorage.updateStorageType(location)
    show_shared_private_details()
  
}

function show_shared_private_details(){
    // show or hide the details of each authentication method appropriately
    if(BookStorage.isPrivateComputer()){
        for(const elem of document.getElementsByClassName("shared-computer")){
            elem.style.display="none"
        }
        for(const elem of document.getElementsByClassName("private-computer")){
            elem.style.display=""
        }
    }else{
        for(const elem of document.getElementsByClassName("shared-computer")){
            elem.style.display=""
        }
        for(const elem of document.getElementsByClassName("private-computer")){
            elem.style.display="none"
        }
    }
      
}

async function choose_configuration(evt) {
  //console.log("evt",evt)


  // can be called by passing in the checkbox to operate on, or by clicking the checkbox
  let checkbox=evt
  if(evt.target){
    // it looks as though we are clicking on the checkbox
    checkbox = evt.target
  }

  const config_id = checkbox.id

  // set the dbType to the id of hte box just checked
  let change=false
  let response=null
  switch (config_id) {
    case "book":
      change=true
      break
    case "sqlite":
      change=true
      break
    case "dataworld":
      //console.log("setting dw token")
      checkbox.innerHTML="hourglass_empty"
      checkbox.classList.add("blink")
      if(!evt.ctrlKey){//only trying to check existing config is ctrl is not down when clicked
        response=get_from_book_storage("dwToken")
        if (response){
          response = await check_dw_connection(response)
          if(response.status!=="success"){
            response=null
          }
        }
      }

      if(response){  
        change=true
      }else{  
        // either there is no dwToken or it did not work
        response = await get_dw_token()
        if(response){
          BookStorage.dwToken(response)
          change=true
        }
      }
      checkbox.classList.remove("blink")
      break
    case "oracle":
      checkbox.innerHTML="hourglass_empty"
      checkbox.classList.add("blink")
      // check to see if we can run an oracle query
      
      let succesfullyExecutedQuery=false
      if(!evt.ctrlKey){//only trying to check existing config is ctrl is not down when clicked

        let endpoint = BookStorage.loginType()==="gas" ? get_gas_endpoint() : null
        // it's okay to attempt the query if we have an endpoint OR we the logintype is paid

        if(endpoint || BookStorage.loginType()==="paid"){
          try{
              response = await run_db_query({connection:"sql_book_connection", engine:"oracle",endpoint:endpoint, query:"select count(*) as one from canvas.museum"})  
              console.log(response)
              if (response?.items[0]?.resultSet?.items[0]?.one) {
                succesfullyExecutedQuery = true
              }else{
                succesfullyExecutedQuery = false
              }
          }catch(e){
            succesfullyExecutedQuery = false
          }
        }
      }
      if(succesfullyExecutedQuery){
        // we succesfully executed an oracle query
        change=true
      }else{
        response = await get_oracle_configuration()
        console.log("response from getting oracle", response)
        if(response=="connected"){
          change=true
        }
      }
      checkbox.classList.remove("blink")
      break
  }
  checkbox.classList.remove("blink")
  checkbox.innerHTML="check_box_outline_blank"
  if(change){
    BookStorage.dbType(config_id)
    set_book_label()
    for (box of document.querySelectorAll("." + checkbox.dataset.set)) {
      box.getElementsByTagName("span")[0].innerHTML = "check_box_outline_blank"
      box.nextElementSibling.style.display = "none"
    }
    checkbox.innerHTML = "check_box"
    tag("config-" + config_id).style.display = ""
    configure_book()
  }
}


async function choose_ai_method(evt) {
  // executed when the users clicks the an AI option checkbox on the congiguration page
  const checkbox=evt.target
  if(checkbox.innerHTML==="check_box"){return}
  BookStorage.aiType(checkbox.id)

  for (box of document.querySelectorAll("." + checkbox.dataset.set)) {
    box.getElementsByTagName("span")[0].innerHTML = "check_box_outline_blank"
    box.nextElementSibling.style.display = "none"
  }

  checkbox.innerHTML = "check_box"
  configure_book()
}




async function choose_login_method(evt) {
  //console.log("calling choose login method");

  // can be called by passing in the checkbox to operate on, or by clicking the checkbox
  let checkbox=evt
  if(evt.target){
    // it looks as though we are clicking on the checkbox
    checkbox = evt.target
  }
    const login_id = checkbox.id
  
    let change = false
    let auth_message="You are now configured to not require a password when working on this computer."
    let new_package

    checkbox.innerHTML="hourglass_empty"
    checkbox.classList.add("blink")

    login_id === "login-paid" ? firebase.auth.signOut(firebase.auth.getAuth()) : null

    switch (login_id) {
      case "login-none":
        set_login_type()
        change=true
        message({
          message:auth_message,
          title:"Authentication Method Changed",
          seconds:8,
          show:true
        })
        if(BookStorage.dbType()==="oracle"){
            // Oracle not allowed with this login type
            swtich_db("oracle","sqlite")
        }
        break
      case "login-local":
        auth_message="You are now logged in to this computer.  If you close your browser, you will need to supply your password again to access your saved settings."
        new_package = await set_local_password()
        if(new_package.password){
          // we have a new local password.  Use it to encrypt values in bookStorage
          set_login_type(new_package.password)
          BookStorage.localCanary(ncrypt("logged in",BookStorage.token()))
          change=true
          message({
            message:auth_message ,
            title:"Authentication Method Changed",
            seconds:8,
            show:true
          })
          if(BookStorage.dbType()==="oracle"){
            // Oracle not allowed with this login type
            swtich_db("oracle","sqlite")
        }
  
        }
        break
      case "login-gas":
        //console.log("getting deployment id",BookStorage.deploymentId())
        // first check to see if we have a valid token
        if(!evt.ctrlKey){//only trying to check existing config is ctrl is not down when clicked
          auth_message="You are now logged in using your own authentication server.  If you close your browser, you will need to supply your password again to access your saved settings."
          if(get_from_book_storage("deploymentId") && get_from_book_storage("gasToken")){
              // we ahave a token and deployment it, lets see if it is valid fist...
              const payload = {
                  mode: "connect"
              }
              const response = await server_post(payload, get_gas_endpoint())
              if (response.status === "connected") {
                  // we have connected and we are done
                  change=true
                  set_login_type(uuid())
                  gas_to_book_storage(response,get_from_book_storage("deploymentId"))
                  message({
                      message:auth_message,
                      title:"Authentication Method Changed",
                      seconds:8,
                      show:true
                  })
                  
                  break
              }
          }
        }        
        // set a new password if necessary
        new_package = await set_gas_password()
        if(new_package && new_package.status && new_package.status==="success"){
            BookStorage.deploymentId(new_package.deployment_id)
            // we have a new information for gas authentication.  Use it to encrypt values in bookStorage
            change=true
            set_login_type(new_package.password)
            BookStorage.localCanaryBookStrage=BookStorage.token  // used to hold the local-login token so we can recall it 
            message({
            message:auth_message,
            title:"Authentication Method Changed",
            seconds:8,
            show:true
            })
    
        }
        
        break  
      case "login-paid":
        //console.log("firebase-login")
        change = true // renders the check box
        // window.paidSection()
        paidUserInfo()
        break
    }
    checkbox.classList.remove("blink")
    checkbox.innerHTML="check_box_outline_blank"

    if(change){
      BookStorage.loginType(login_id.split("-")[1])
      // clear other checkmarks 
      for (box of document.querySelectorAll("." + checkbox.dataset.set)) {
        box.getElementsByTagName("span")[0].innerHTML = "check_box_outline_blank"
        box.nextElementSibling.style.display = "none"
      }
      // check the correct one 
      checkbox.innerHTML = "check_box"
      tag("config-" + login_id).style.display = "block"
      

      //debugger
      set_book_label()  
      configure_book()  
    }
    function swtich_db(from_db,to_db){
        // changes bookstroage dbtype and the checkbox.  does no verification
        BookStorage.dbType(to_db)
        tag(from_db).innerHTML="check_box_outline_blank"
        tag(to_db).innerHTML="check_box"
    }
  
  }

async function abandon_machine(){
    
    const button_label="Yes, clear all data."
    const button = await ui_message_box(
        "This will clear all data from this machine, resetting it as if no one has used this book here.  Are you sure you want to clear the data? ",
        "Are you sure?",
        [button_label, "Oops, don't clear my data."]
    )
    
    if(button===button_label){
        localStorage.clear()
        sessionStorage.clear()
        window.location.reload()
    }
}

function set_login_type(password="Samuel Davies"){
    BookStorage.changeEncryptionKey(password)
    if(BookStorage.isPrivateComputer()){
        BookStorage.updateStorageType(AutoStorage.STORAGE_TYPE.LOCAL)
    }else{
        BookStorage.updateStorageType(AutoStorage.STORAGE_TYPE.SESSION)
    }  
}

  

async function login_local(){
    // allows the user to supply the password for a local login and verifies if it is correct
    const value = await new Promise(
      (resolve, reject)=>{
        document.body.append(ui_modal("Login to this Computer",login_local_form(resolve),resolve))
        tag("pwd").focus()
      }
    )
    return value
  
    // currently only used here, but could be used elsewhere.  then we will move it out of this fn
    function login_local_form(resolve = () => { }, reject = () => { }) {
      const form = ui_form("auth-container ui-inline-block")
      
      const password_block = ui_input_password_reveal("ui-input ui-full-width")
      const password_input = password_block.getElementsByTagName("input")[0]
      password_input.id="pwd"
      const error_message = ui_div("ui-bg-clear error-message ")
      const submit = ui_btn("login")
      submit.classList.add("mt3")
      submit.onclick = (e) => {
          e.preventDefault()
          
          if(password_input.value.length===0){
            error_message.replaceChildren(ui_span("", "Password is required."))
          }else if(! BookStorage.login(password_input.value)){
            error_message.replaceChildren(ui_span("", "The password in incorrect."))
          }else{
            // passed all checks
            //submit.replaceChildren(ui_loading())
            submit.disabled = true

            error_message.replaceChildren()
            tag("overlay").remove()
            resolve("success")
  
          }
      }
  
      const switch_btn = ui_btn("Switch to No Authentication")
      switch_btn.classList.add("mt3")
      switch_btn.onclick = (e) => {
          e.preventDefault()
          set_login_type()
          tag("overlay").remove()
          resolve("no login")
      }
  
      const or=ui_h2("","OR")
      or.style.textAlign="center"
      form.append(
          ui_p("","This book is configured for Local Login. Please supply the password you chose previously to access your saved settings")  ,
          ui_label("", "Password"), ui_br(), password_block, ui_br(),
          submit,
          error_message,
          or,
          ui_p("",'Switch to "No authentication" to start from scratch.  This will clear any saved settings you have.')  ,
          switch_btn
      )
      return form
    }
  }
  

  
  async function login_gas(){
    // allows the user to supply the password for a local login and verifies if it is correct
    
    // first, check to see if we are already logged in
    if(get_from_book_storage("deploymentId") && get_from_book_storage("gasToken")){
        // we ahave a token and deployment it, lets see if it is valid fist...
        const payload = {
            mode: "connect"
        }
        const response = await server_post(payload)
        if (response.status === "connected") {

            set_login_type(uuid())// generate a password for use with encryption.  I does not need to be knowable because this login time is verified at google server
            gas_to_book_storage(response, get_from_book_storage("deploymentId"))
            message({
                message:"You are now logged in using your own authentication server.",
                title:"Authentication Method Changed",
                seconds:8,
                show:true
            })
            return {status:"success"}
        }
    }

    const value = await new Promise(
      (resolve, reject)=>{
        document.body.append(ui_modal("Login to your Authentication Server",login_gas_form(resolve),resolve))
        tag("depl-id").focus()
      }
    )
    return value
  
    // currently only used here, but could be used elsewhere.  then we will move it out of this fn
    function login_gas_form(resolve = () => { }, reject = () => { }) {
      const form = ui_form("auth-container ui-inline-block")
  
      const switch_btn = ui_btn("Switch to No Authentication")
      switch_btn.classList.add("mt3")
      switch_btn.onclick = (e) => {
          e.preventDefault()
          set_login_type("none")
          tag("overlay").remove()
          resolve("no login")
      }
  
      const or=ui_h2("","OR")
      or.style.textAlign="center"
      form.append(
          gas_authentication_block(resolve),
          or,
          ui_p("",'Switch to "No authentication" to start from scratch.  This will clear any saved settings that may be stored on this machine.  You can always log in to your authentication server later to access the work and settings saved there.')  ,
          switch_btn
      )
      return form
    }
    
  
  }


  
  async function set_local_password(){
  // sets the password for a local login
    const value = await new Promise(
      (resolve, reject)=>{
        document.body.append(ui_modal("Set Local Password",set_local_password_form(resolve),resolve))
        tag("pwd").focus()
      }
    )
    return value
  
    // currently only used here, but could be used elsewhere.  then we will move it out of this fn
    function set_local_password_form(resolve = () => { }, reject = () => { }) {
      const form = ui_form("auth-container ui-inline-block")
      const password_block = ui_input_password_reveal("ui-input ui-full-width")
      const password_input = password_block.getElementsByTagName("input")[0]
      password_input.id="pwd"
      const submit = ui_btn("Set Password")
      const error_message = ui_div("ui-bg-clear error-message ")
      submit.classList.add("mt3")
      submit.onclick = (e) => {
          e.preventDefault()
          
          if(password_input.value.length===0){
            error_message.replaceChildren(ui_span("", "Password is required."))
          }else{
            // passed all checks
            //submit.replaceChildren(ui_loading())
            submit.disabled = true
            error_message.replaceChildren()
            tag("overlay").remove()
            resolve({password:password_input.value})
          }
      }
      form.append(
          ui_p("","Choose a password that you will use to authenticate on this computer.   We'll use it to encrypt your settings and work so even if someone gets ahold of this computer, they wont be able to read your data."),
          ui_label("", "Password"), ui_br(), password_block, ui_br(),
          submit,
          error_message,
      )
      return form
    }
    
  
  }
  
  
  
  
  async function set_gas_password(){
    // sets the password for a local login
      const value = await new Promise(
        (resolve, reject)=>{
          document.body.append(ui_modal("Connect to your Authentication Server",set_gas_password_form(resolve),resolve))
          tag("depl-id").focus()
        }
      )
      return value
    
      // currently only used here, but could be used elsewhere.  then we will move it out of this fn
      function set_gas_password_form(resolve = () => { }, reject = () => { }) {
        const form = ui_form("auth-container ui-inline-block")
        form.append(
            gas_authentication_block(resolve)
        )
        return form
      }
      
    
    }
    
    
function gas_authentication_block(resolve){
   
    const deployment_id_input = ui_input("ui-input ui-full-width")
    deployment_id_input.id="depl-id"

    const password_block = ui_input_password_reveal("ui-input ui-full-width")
    const password_input = password_block.getElementsByTagName("input")[0]
    const password_div=ui_div("", ui_label("", "Password"), ui_br(), password_block)
    password_div.style.display="none"

    const reset_code_input = ui_input("ui-input ui-full-width")
    const reset_div=ui_div("", ui_label("", "Password Reset Code"), ui_br(), reset_code_input)
    reset_div.style.display="none"

    const reset = ui_btn("Reset Password")       
    reset.style.display="none"
    const reset_message = ui_div("ui-bg-clear")

    reset.classList.add("mt3")
    reset.onclick = async (e) => {
        e.preventDefault()
        reset.disabled = true
        const payload = {
            mode: "request-reset",
          }
          const response = await server_post(payload,get_gas_endpoint(deployment_id_input.value))
          reset.disabled = false
          //console.log("response", response)
          if (response.status === "success") {
            reset_message.replaceChildren(ui_span("", 'A reset code has been sent to the email address attached to the deployment id you entered. Enter the code in the "Reset Code" field above and enter the password you want to use.'))
            reset_div.style.display="block"
            // message({
            //   message: `A reset code has been sent to the email address attached to the deployment id you enterd. Enter the code in the "Reset Code" field below and choose a new password.  You can also change the username if you like.`,
            //   title: "Password",
            //   seconds: 8,
            //   show: true
            // })
            // tag("reset-code").style.display = null
            // tag("reset-button").style.display = "none"
          } else if (response.status === "failure") {
            message({
              message: response.message,
              title: "Reset Request Failed",
              seconds: 8,
              kind: "error",
              show: true
            })
          } else {
            message({
              message: result,
              title: "Unexpected Error",
              seconds: 8,
              kind: "error",
              show: true
            })
          }
      

    }

    const submit = ui_btn("Connect")       
    const error_message = ui_div("ui-bg-clear error-message ")
    submit.classList.add("mt3")
    submit.onclick = async (e) => {
        e.preventDefault()
        
        if(e.target.innerHTML==="Connect" || e.target.innerHTML==="Login"){
            // initial connection to see if the server responding    
            if(deployment_id_input.value.length===0){
                error_message.replaceChildren(ui_span("", "Deployment ID is required."))
            }else if(e.target.innerHTML==="Login" && password_input.value.length===0){
                error_message.replaceChildren(ui_span("", "Password is required."))
            }else{
                // passed all checks
                //submit.replaceChildren(ui_loading())
                submit.disabled = true
                error_message.replaceChildren()
                reset.style.display="none"

                const payload = {
                    mode: "connect"
                }

                if(password_input.value){
                    payload.password=password_input.value
                }
                
                if (reset_code_input.value) {
                    payload.resetToken = reset_code_input.value
                }
                
                  const response = await frelayServerPost(payload,get_gas_endpoint(deployment_id_input.value))
                  submit.disabled = false
                  if(response.status==="error"){
                    // the server was contacted but because we have not sent a password, we could not yet login
                    error_message.replaceChildren(ui_span("", "Unable to contact server using the Deployment ID given.  Please verify and try again."))
                  }else if(response.status==="alive"){
                    BookStorage.deploymentId(deployment_id_input.value)
                    // the server was contacted but because we have not sent a password, we could not yet login
                    submit.innerHTML="Login"
                    password_div.style.display="block"
                  }else if(response.status==="invalid-password"){
                    BookStorage.deploymentId(deployment_id_input.value)
                    
                    if(submit.innerHTML==="Login"){
                        error_message.replaceChildren(ui_span("", "Invalid Password.  Try again or use the button below to change your password."))
                        reset.style.display=""
                    }else{
                        submit.innerHTML="Login"
                        password_div.style.display="block"
                    }
                    
                  }else if(response.status==="connected"){
                        // The existing token was valid, so we are connected
                        tag("overlay").remove()
                        gas_to_book_storage(response, deployment_id_input.value)
                        resolve({status:"success", deployment_id:deployment_id_input.value})            
                  }else{

                  }

            }
        }else if(e.target.innerHTML==="login"){

        }
    }

    return ui_div("",
        ui_p("",'Enter the "Deployment ID" that you copied when you set up your authentication server on Google Apps Script.'),
        ui_label("", "Deployment ID"), ui_br(), deployment_id_input, ui_br(),
        password_div,
        reset_div,
        submit,
        error_message,
        reset,
        reset_message,
    )


}    
    
  
  
function paste_settings_import(){
    // imports stdent data to a a textbox for copy and pasting
  
  let data=tag("copy-paste").value

  try{
    data=JSON.parse(data)
  }catch(e){
    message({
      message: "There is an error in the import data.",
      title: "Import Error",
      seconds: 8,
      kind: "error",
      show: true
    })
    return
  }
  //console.log("data", data)

  //console.log("get_book_db_type()",get_book_db_type())
  object_to_bookstorage(data)
  //console.log("get_book_db_type tag",tag(get_book_db_type()))

  // run the same code that gets run when the settings page loads
  choose_configuration(tag(get_book_db_type()))
  choose_login_method(tag("login-"+BookStorage.loginType()))
  fill_server_configuration()  

  message({
    message: "Your information has been restored",
    title: "Import Succeeded",
    seconds: 4,
    show: true
  })
  


}

function paste_settings_export(){
  const data=bookstorage_to_object()
  //console.log(data)

  tag("copy-paste").value=JSON.stringify(data,null,2)

  message({
    message: "Your information can now be copied or edited.",
    title: "Export Successful",
    seconds: 8,
    show: true
  })

}


function export_student_data(){
    // exports stdent data to a file to be imported with import_student_date
    const password=tag("import-export-password").value
  
  
    const data=bookstorage_to_object()
  
    const data_file=(
      password ? 
      ncrypt(JSON.stringify(data),password)
      : 
      JSON.stringify(data,null,2)
    )
  
    //console.log(data_file)
  
    download_text_file(data_file, "sql-book-settings.json")
    message({
      message: "Your information file has been exported.",
      title: "File Downloaded",
      seconds: 8,
      show: true
    })
}


function import_student_data(raw_data){
    // imports a file that was exported with export_stduent_data
    //console.log("at import stduetn data")
  
    let data
    try{// try to import with out password, just in case...
      data=JSON.parse(raw_data) 
    } catch(e){
      // could not decrypt withhout PW, try with
      data=dcrypt(raw_data,tag("import-export-password").value)
      
      if(data==="failed"){  
        message({
          message: "The password you provided is not correct.  Please try again.",
          title: "File Not Imported",
          seconds: 8,
          kind: "error",
          show: true
        })
        return
      }    
    }

    object_to_bookstorage(data)

  
    // run the same code that gets run when the settings page loads
    choose_configuration(tag(get_book_db_type()))
    choose_login_method(tag("login-"+BookStorage.loginType()))
    fill_server_configuration()  
  
    message({
      message: "Your information has been restored",
      title: "File Imported",
      seconds: 8,
      show: true
    })
  }

  function fill_server_configuration() {
    // computer shared or private
    
    if(BookStorage.isPrivateComputer()){
      tag("private_computer").checked=true
    }else{
      tag("shared_computer").checked=true
    }
    
    
  
    // prefill the server configuration with whatever is known about it
    //console.log("url", url)
    const url_params = get_params()
    //console.log("url_params", url_params)
    if (url_params.data) {
      // a data block has been sent in link, these take precedence
      data = JSON.parse(atob(url_params.data))
      //console.log("data", data)
      tag("gas-username").value = data.gasUsername
  
      if(data.deploymentId){
        tag("deployment-id").value = data.deploymentId
        BookStorage.deploymentId=data.deploymentId
      }
    }
  
    data_from_book_storage_to_element(document.body)  
  
  
  }
  
  
function data_from_book_storage_to_element(element=document.body){
    // looks at all input tags in element and uses thier ids to try to pull data from bookStorage
    for (const [var_name,id] of Object.entries(get_id_map_from_element(element))){
      try{
        tag(id).value=BookStorage[var_name]()
      }catch(e){
        //console.error(e)
        // on error, ignore.  It means there is no matching value in bookStorage
      }
    }
  }
  
  function init_user_profile(){
    // sets up the user-profile page when it is loaded
    tag(get_from_book_storage("aiType", "gpt")).innerHTML="check_box"
    tag(get_from_book_storage("dbType", "sqlite")).innerHTML="check_box"
    show_shared_private_details()   
  
    // check the current authentication method
    const elem = tag("login-"+BookStorage.loginType())
    elem.innerHTML="check_box"
    elem.parentNode.nextElementSibling.style.display="block"

    // speacial things we do for each login type
    if (BookStorage.loginType() === "paid") {
      //console.log("paid");
      paidUserInfo() // this renders the UI for this section
    }else{
      // default handling
    }

    fill_server_configuration()
    tag('uploadInput').addEventListener('change', function() {
            const fr=new FileReader();
            fr.onload=function(){import_student_data(fr.result)}
            fr.readAsText(this.files[0]);
    })
  }

function save_student_data(){
  // saves the stduent data
  const promises = []
  for(const [key,value] of Object.entries(get_data_from_element(tag("student-data")))){
      // set value in book storage
      promises.push(BookStorage[key](value))
  }

  Promise.all(promises).then(()=>{
    message({
      message: "Successfully saved your user information.",
      title: "User Information Saved",
      seconds: 8,
      show: true
    })
  }).catch(err=>{
    message({
      message: err.message,
      title: "Problem Saving value to server",
      kind: "error",
      show: true
  })
  })
}


async function get_dw_token(){
  // an awaitable dialog box
    const value = await new Promise(
      (resolve, reject)=>{
        document.body.append(ui_modal("Read/Write Token",get_dw_token_form(resolve),resolve))
        tag("token-input").focus()
      }
    )
    return value
  
    // currently only used here, but could be used elsewhere.  then we will move it out of this fn
    function get_dw_token_form(resolve = () => { }, reject = () => { }) {
      const form = ui_form("auth-container ui-inline-block")
      const token_input = ui_input("ui-input ui-full-width mb2")
      token_input.id="token-input"
      token_input.value=get_from_book_storage("dwToken")
      const button_label="Save"
      const submit = ui_btn(button_label)
      const error_message = ui_div("ui-bg-clear error-message ")
      const message=ui_p()
      message.innerHTML='The data.world <a href="https://data.world/settings/advanced" target="blank">Read/Write token</a> authorizes this site to interact with data.world on your behalf so you can execute queries and see your results.'
      submit.onclick = async (e) => {
        error_message.replaceChildren()
        e.preventDefault()

        if(token_input.value.length===0){
          error_message.replaceChildren(ui_span("", "The token is required."))
        }else{
          submit.replaceChildren(ui_loading())
          submit.disabled = true
          const result=await check_dw_connection(token_input.value)
          if(result.status==="success"){
            error_message.replaceChildren()
            tag("overlay").remove()
            resolve(token_input.value)
          }else{
            error_message.replaceChildren(ui_span("", result.message))
            submit.replaceChildren(button_label)
            submit.disabled = false
          }
        }
      }
      form.append(
        message,
        ui_label("", "data.world Read/Write Token"), ui_br(),
        token_input,
        submit,
        error_message
      )
      return form
    }
  
  }

async function get_oracle_configuration(){
  // an awaitable dialog box
    const value = await new Promise(
      (resolve, reject)=>{
        document.body.append(ui_modal("Oracle Autonomous Database",get_oracle_configuration_form(resolve),resolve))
        tag("url-input").focus()
      }
    )
    return value
  
    // currently only used here, but could be used elsewhere.  then we will move it out of this fn
    function get_oracle_configuration_form(resolve = () => { }, reject = () => { }) {
      const form = ui_form("auth-container ui-inline-block")
      const url_input = ui_input("ui-input ui-full-width mb2")
      url_input.id="url-input"
      url_input.value=get_from_book_storage("oracleURL")
      let button_label
      const error_message = ui_div("ui-bg-clear error-message")
      const message=ui_p("mt0")

      const username_input = ui_input("ui-input ui-full-width mb2")
      username_input.id="username-input"
      username_input.value="ADMIN"


      const password_block = ui_input_password_reveal("ui-input ui-full-width")
      const password_input = password_block.getElementsByTagName("input")[0]
      password_input.id="oracle-admin-pwd"

      const script_message  =   ui_div("script-message")
      script_message.id="script-message"
      script_message.style.display="none"
      
      const auth_block =  ui_div("",
        ui_label("", "Administrator Username"), ui_br(),username_input,
        ui_label("", "Administrator Password"), ui_br(),password_block,
      )

      const url_block = ui_div("",ui_label("", "Oracle Connection URL"), ui_br(),url_input)

      const connection_block=ui_div("",url_block,auth_block,script_message)

      if(get_from_book_storage("loginType")==='gas' || get_from_book_storage("loginType")==='paid'){
        //login type allows oracle
        message.innerHTML='Provide the the connection URL for any user on your Oracle Autonomous Database.  You may need to follow the <a target="_blank" href="https://goog.le.com">instructions to configure your Oracle Autonomous Database</a> to work with this book.'
        button_label="Initiate Connection"
      }else{
        //login type does not allow oracle
        message.innerHTML='To use your Oracle Autonomous Database, you must configure this book to authenticate with either "Your Own Authentication Server" or with "This Book\'s Authentication Server".'
        button_label="OK"
        connection_block.style.display="none"
      } 

      const submit = ui_btn(button_label)
      submit.style.marginTop="1rem"

      submit.onclick = async (e) => {
        //console.log("clicked", e)
        error_message.replaceChildren()
        e.preventDefault()

        switch(e.target.innerHTML){
            case "OK":
                tag("overlay").remove()
                resolve(null)
                return
                break
            case "Close":
                if(submit.dataset.result==="error"){
                    tag("overlay").remove()
                    resolve(null)
                }else{
                    tag("overlay").remove()
                    resolve("connected")
                }
                return
                break
            case "Configure Datasets":

                if(username_input.value.length===0 || password_input.value.length===0){
                    error_message.replaceChildren(ui_span("", "The username and password are both required."))
                }else{
                    // we have a username and password, let's check if the are valid
                    submit.replaceChildren(ui_loading())
                    const response = await check_oracle_connection(url_input.value,username_input.value,password_input.value)
                    //response.code=10    
                    switch(response.code){
                        case 3: //Credentials supplied are unauthorized as server
                            // ok, we have authenticated with the admin username and passord
                            url_block.style.display="none"
                            auth_block.style.display="block"
                            submit.innerHTML="Configure Datasets"
                            error_message.replaceChildren(ui_span("", "The username and pasword you supplied are unable to connect.  Please check them and try again."))
                            break
                        case 10: //good credentials, not configured
                            // ok, we have authenticated with the admin username and passord, time to bulid
                            url_block.style.display="none"
                            auth_block.style.display="none"
                            script_message.style.display="block"
                            submit.style.display="none"
                            submit.dataset.result =  await configure_book_db(username_input.value, password_input.value, url_input.value)
                            submit.style.display=""
                            submit.innerHTML="Close"
                            break
                        default:    
                            if(response.message){
                                error_message.replaceChildren(ui_span("", response.message))  
                            }else{
                              error_message.replaceChildren(ui_span("", "Something very unexpected happened."))  
                            }
                    }
                }    
                break
            case "Initiate Connection":
              //debugger
                if(url_input.value.length===0){
                    error_message.replaceChildren(ui_span("", "The connection URL is required."))
                }else{
                    // try to talk to the server at the url
                    submit.replaceChildren(ui_loading())
                    const response = await check_oracle_connection(url_input.value,username_input.value,password_input.value)
                    //response.code=10
                    switch(response.code){
                        case 10: //connected but no datqabase
                            message.innerHTML = "Successfully connected to your server, yay! It looks as though we need to configure this book's data on your server.  Provide the additional information below."
                            url_block.style.display="none"
                            auth_block.style.display="block"
                            submit.innerHTML="Configure Datasets"
                            break
                        case 20: //fully configured
                            tag("overlay").remove()
                            resolve("connected")
                            break
                        default:   
                            //console.log("response",response) 
                            submit.innerHTML="Initiate Connection"
                            if(response.message){
                                error_message.replaceChildren(ui_span("", response.message))  
                            }else{
                              error_message.replaceChildren(ui_span("", "Something very unexpected happened."))  
                            }
                    }
                    // if(response.status="success"){
                    //     //console.log("oracle responded")
                    // }else{
                    //     //console.log("oracle failed")
                    // }

                    // submit.replaceChildren(ui_loading())
                    // submit.disabled = true
                    // const result=await check_dw_connection(url_input.value)
                    // if(result.status==="success"){
                    //     error_message.replaceChildren()
                    //     tag("overlay").remove()
                    //     resolve(url_input.value)
                    // }else{
                    //     error_message.replaceChildren(ui_span("", result.message))
                    //     submit.replaceChildren(button_label)
                    //     submit.disabled = false
                    // }
                }
                break
            default:
                console.error("button Label not recognized")    
                resolve(null)
                return                
        }

      }
      form.append(
        message,
        connection_block,
        submit,
        error_message
      )
      return form
    }
  
  }

  async function check_oracle_connection(url, username, password){
    // accepts a token and checks to see if we can connect to data.world
    const payload = {
        mode: "check-oracle-connection",
        url: url
    }
    if(username){payload.username=username}
    if(password){payload.password=password}
    console.log("check_oracle_connection",payload)
    return await server_post(payload)
  }

  async function check_dw_connection(token){
    // accepts a token and checks to see if we can connect to data.world
    const url = `https://api.data.world/v0/sql/hudsonu/hudsonu`
    const options = {
      headers: {
        'Content-Type': 'application/json',
        'Authorization': 'Bearer ' + token,
      },
      'method': 'POST',
      'body': JSON.stringify({
        "query": "select * from approval",
        "includeTableSchema": true,
        "queryRunToken": uuid()
      })
    }
    let response
    try {
      response = await fetch(url, options)
    } catch (e) {
      ;console.log("http error", e.message)
      return {status:"failure", message:e.message}
    }
    try{
      if(response.status===200){
        // OK--everything is valid
        return {status:"success", message:"Connection successful."}
      }else if(response.status===401){
        // unauthorized
        return {status:"failure", message:"The Read/Write token was rejected by data.world.  Please check it and try again."}
      }else if(response.status===404){
        return {status:"failure", message:"Contacted data.world, but could not find the verification project.  This is unexpected."}
      }
    }catch(e){  
      return {status:"failure", message:"Unexpected error: " + e.message}
    }
  }

const paidUserInfo = (() => {
  const {
    getAuth,
    onAuthStateChanged,
    signOut,
  } = firebase.auth
  const { getFunctions, httpsCallableFromURL } = firebase.functions

  const auth = getAuth();

  const loading = ui_div("center",
    ui_loading("")
  )

  function profile(user) {
    return ui_div("center",
      ui_div("user-info",
        ui_div("user-icon", user.email[0].toUpperCase()),
        ui_div("user-email", user.email),
        signOutUI(auth),
      ),
    )
  }

  function signOutUI(firebaseAuth) {
    const signOutBtn = ui_btn_error("Sign Out")
    signOutBtn.onclick = async () => await signOut(firebaseAuth)
    return signOutBtn
  }

  /**
   * This method must be called once the user has been authenticated
   * @returns a url string that can be used to log in the user to thier stripe account or an 
   * empty string if there was an error in the backend.
   */
  async function getStripePortalLink() {
    try {
      const functions = getFunctions()
      const createPortalLink = httpsCallableFromURL(
        functions,
        "https://us-west3-prime-micron-374215.cloudfunctions.net/ext-firestore-stripe-payments-createPortalLink"
      );
      const { data } = await createPortalLink({
        returnUrl: window.location.origin,
      })
      return data.url
    } catch (error) {
      return ""
    }
  }


  async function productAndPriceData() {
    const products = []
    const querySnapshot = await getDocs(collection(db, "products"));
    querySnapshot.forEach(async (productDoc) => {
      const product_data = productDoc.data()
      if (product_data.active) {
        products.push({
          docID: productDoc.id,
          ...product_data
        })
      }
    })
    const prices_snapshot_promise = []
    for (const product of products) {
      prices_snapshot_promise.push(getDocs(collection(db, "products", product.docID, "prices")))
    }
    const product_and_price = []
    const snapshots = await Promise.all(prices_snapshot_promise)
    for (const priceSnapshot of snapshots) {
      priceSnapshot.forEach(priceDoc => {
        const price = priceDoc.data()
        product_and_price.push({
          product: products.find(x => x.docID === price.product),
          price: price,
          productID: price.product,
          priceID: priceDoc.id,
        })
      })
    }
    return product_and_price
  }

  function managePurchaseCard(product_data, description, features, stripeLink) {
    const buy_btn = ui_btn_info("Loading")
    buy_btn.replaceChildren("Manage")
    buy_btn.onclick = () => window.location.assign(stripeLink)
    const head = ui_card_head(product_data.name, ui_first_letter_upper_case(product_data.description))
    const card = ui_card(head, description, buy_btn, features)
    return card
  }

  /**
   * gets the users purchased products. If the user did not buy anything, or
   * the user is not signed in, then this method will return null. 
   * @returns
   */
  async function getPurchasedProducts(user) {
    if (!user) { return [] }
    const usersSubsSnapshot = await getDocs(collection(db, "customers", user.uid, "subscriptions"))
    const subs = []
    usersSubsSnapshot.forEach(x => x = subs.push(x.data()))
    return subs.map(x => ({ ...x, productID: x.items[0].plan.product }))
  }

  async function checkout_product(price_id, price_type, user) {
    if (!user) {
      user = await getUser()
    }
    const checkoutPayload = {
      price: price_id,
      success_url: window.location.origin,
      cancel_url: window.location.origin,
    }
    if (price_type === "one_time") { checkoutPayload.mode = "payment" }

    const checkout_session_ref = collection(db, "customers", user.uid, "checkout_sessions");
    const ref = await addDoc(checkout_session_ref, checkoutPayload)
    await new Promise((resolve) => {
      onSnapshot(ref, (snap) => {
        const { error, url } = snap.data();
        if (error) {
          resolve()
          alert(`An error occured: ${error.message}`);
        }
        if (url) {
          // We have a Stripe Checkout URL, let's redirect.
          resolve()
          window.location.assign(url);
        }
      });
    })



  }

  function markProductsAsPurchased(products_and_prices, purchased_products) {
    return products_and_prices.map(product_and_price => {
      const purchased_product = purchased_products.find(purchased_product => purchased_product.productID === product_and_price.productID)
      if (purchased_product) {
        return { ...product_and_price, purchased: true, }
      }
      return product_and_price
    })
  }

  function uiProductsWithPrices(product_and_price, stripe_link, user) {
    const grid = ui_grid()
    for (const { product, price, priceID, purchased } of product_and_price) {
      const dollar_amount = price.unit_amount / 100
      const product_features = Object.entries(product.metadata)
        .filter((([key]) => key.match(/feature_\d+/)))
        .sort((a, b) => a[0] > b[0])
        .map(([_, value]) => ui_first_letter_upper_case(value))

      let price_div
      if (price.type === "one_time") {
        price_div = ui_div("center price-tag", "$", dollar_amount)
      } else {
        price_div = ui_div("center price-tag", "$", dollar_amount, " / ", price.interval)
      }

      const description = ui_card_body_item("", price_div, ui_br())
      const features = ui_div("", ui_ul("card-ul-padding", ...product_features.map(x => ui_li("", x))))

      if (purchased) {
        grid.append(ui_grid_item(managePurchaseCard(product, description, features, stripe_link)))
      } else {

        const buy_btn = ui_btn("Buy Now")
        buy_btn.onclick = async () => {
          buy_btn.disabled = true
          buy_btn.innerText = "Heading to checkout..."
          try {
            await checkout_product(priceID, price.type, user)
          } catch (error) {
            if (error === "user abort") {
              // the user simply quit. So we dont have to do anything.
            } else {
              console.error(error)
            }
          } finally {
            buy_btn.innerText = "Buy Now"
            buy_btn.disabled = false
          }
        }

        const head = ui_card_head(product.name, ui_first_letter_upper_case(product.description))
        const card = ui_card(head, description, buy_btn, features)
        grid.append(ui_grid_item(card))
      }
    }
    return grid
  }

  return async () => {
    const root_node = document.getElementById("user-profile-asd")
    root_node.replaceChildren(loading)
    const product_and_price = await productAndPriceData()
    await new Promise(() => {
      onAuthStateChanged(auth, async (user) => {
        let p_a_p = product_and_price
        let stripe_link = ""
        if (user) {
          let purchased_products
          [purchased_products, stripe_link] = await Promise.all([getPurchasedProducts(user), getStripePortalLink()])
          p_a_p = markProductsAsPurchased(p_a_p, purchased_products)
          root_node.replaceChildren(profile(user))
        }else{
          root_node.replaceChildren()
        }

        const signInButton = ui_button("ui-button-link", "Have an account? Sign in")
        signInButton.onclick = getUser

        root_node.append(
          uiProductsWithPrices(p_a_p, stripe_link, user),
          ui_div("center",
            user ? "" : signInButton,
          ),
        )
      })
    })
  }
})()
person
navigate_before Table of Contents expand_more navigate_next
Loading Page hourglass_empty