@basictech/react 0.7.0-beta.0 → 0.7.0-beta.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/AuthContext.tsx","../src/sync/index.ts","../src/sync/syncProtocol.js","../src/config.ts","../src/db.ts","../package.json","../src/index.ts"],"sourcesContent":["// @ts-nocheck\n\nimport React, { createContext, useContext, useEffect, useState, useRef } from 'react'\nimport { jwtDecode } from 'jwt-decode'\n\nimport { BasicSync } from './sync'\nimport { get, add, update, deleteRecord } from './db'\nimport { validateSchema, compareSchemas } from '@basictech/schema'\n\nimport { log } from './config'\nimport {version as currentVersion} from '../package.json'\n\ntype BasicSyncType = {\n basic_schema: any;\n connect: (options: { access_token: string }) => void;\n debugeroo: () => void;\n collection: (name: string) => {\n ref: {\n toArray: () => Promise<any[]>;\n count: () => Promise<number>;\n };\n };\n [key: string]: any; // For other potential methods and properties\n};\n\n\nenum DBStatus {\n LOADING = \"LOADING\",\n OFFLINE = \"OFFLINE\",\n CONNECTING = \"CONNECTING\",\n ONLINE = \"ONLINE\",\n SYNCING = \"SYNCING\",\n ERROR = \"ERROR\"\n}\n\ntype User = {\n name?: string,\n email?: string,\n id?: string,\n primaryEmailAddress?: {\n emailAddress: string\n },\n fullName?: string\n}\ntype Token = {\n access_token: string,\n token_type: string,\n expires_in: number,\n refresh: string,\n}\n\nexport const BasicContext = createContext<{\n unicorn: string,\n isAuthReady: boolean,\n isSignedIn: boolean,\n user: User | null,\n signout: () => void,\n signin: () => void,\n getToken: () => Promise<string>,\n getSignInLink: () => string,\n db: any,\n dbStatus: DBStatus\n}>({\n unicorn: \"🦄\",\n isAuthReady: false,\n isSignedIn: false,\n user: null,\n signout: () => { },\n signin: () => { },\n getToken: () => new Promise(() => { }),\n getSignInLink: () => \"\",\n db: {},\n dbStatus: DBStatus.LOADING\n});\n\nconst EmptyDB: BasicSyncType = {\n isOpen: false,\n collection: () => {\n return {\n ref: {\n toArray: () => [],\n count: () => 0\n }\n }\n }\n}\n\nasync function getSchemaStatus(schema: any) {\n const projectId = schema.project_id\n let status = ''\n const valid = validateSchema(schema)\n\n if (!valid.valid) {\n console.warn('BasicDB Error: your local schema is invalid. Please fix errors and try again - sync is disabled')\n return { \n valid: false, \n status: 'invalid',\n latest: null\n }\n }\n\n const latestSchema = await fetch(`https://api.basic.tech/project/${projectId}/schema`)\n .then(res => res.json())\n .then(data => data.data[0].schema)\n .catch(err => {\n return { \n valid: false, \n status: 'error',\n latest: null\n }\n })\n\n console.log('latestSchema', latestSchema)\n\n if (!latestSchema.version) {\n return { \n valid: false, \n status: 'error',\n latest: null\n }\n }\n\n if (latestSchema.version > schema.version) {\n // error_code: schema_behind\n console.warn('BasicDB Error: your local schema version is behind the latest. Found version:', schema.version, 'but expected', latestSchema.version, \" - sync is disabled\")\n return { \n valid: false, \n status: 'behind', \n latest: latestSchema\n }\n } else if (latestSchema.version < schema.version) {\n // error_code: schema_ahead\n console.warn('BasicDB Error: your local schema version is ahead of the latest. Found version:', schema.version, 'but expected', latestSchema.version, \" - sync is disabled\")\n return { \n valid: false, \n status: 'ahead', \n latest: latestSchema\n }\n } else if (latestSchema.version === schema.version) {\n const changes = compareSchemas(schema, latestSchema)\n if (changes.valid) {\n return { \n valid: true,\n status: 'current',\n latest: latestSchema\n }\n } else {\n // error_code: schema_conflict\n console.warn('BasicDB Error: your local schema is conflicting with the latest. Your version:', schema.version, 'does not match origin version', latestSchema.version, \" - sync is disabled\")\n return { \n valid: false, \n status: 'conflict',\n latest: latestSchema\n }\n }\n } else { \n return { \n valid: false, \n status: 'error',\n latest: null\n }\n }\n}\n\n\nfunction getSyncStatus(statusCode: number): string {\n switch (statusCode) {\n case -1:\n return \"ERROR\";\n case 0:\n return \"OFFLINE\";\n case 1:\n return \"CONNECTING\";\n case 2:\n return \"ONLINE\";\n case 3:\n return \"SYNCING\";\n case 4:\n return \"ERROR_WILL_RETRY\";\n default:\n return \"UNKNOWN\";\n }\n}\n\ntype ErrorObject = {\n code: string;\n title: string;\n message: string;\n}\n\nasync function checkForNewVersion(): Promise<{ hasNewVersion: boolean, latestVersion: string | null, currentVersion: string | null }> {\n try {\n\n const isBeta = currentVersion.includes('beta')\n\n const response = await fetch(`https://registry.npmjs.org/@basictech/react/${isBeta ? 'beta' : 'latest'}`);\n if (!response.ok) {\n throw new Error('Failed to fetch version from npm');\n }\n\n const data = await response.json();\n const latestVersion = data.version;\n\n if (latestVersion !== currentVersion) {\n console.warn('[basic] New version available:', latestVersion, `\\nrun \"npm install @basictech/react@${latestVersion}\" to update`);\n }\n if (isBeta) {\n log('thank you for being on basictech/react beta :)')\n }\n \n return {\n hasNewVersion: currentVersion !== latestVersion,\n latestVersion,\n currentVersion\n };\n } catch (error) {\n log('Error checking for new version:', error);\n return {\n hasNewVersion: false,\n latestVersion: null, \n currentVersion: null\n };\n }\n}\n\nexport function BasicProvider({ children, project_id, schema, debug = false }: { children: React.ReactNode, project_id?: string, schema?: any, debug?: boolean }) {\n const [isAuthReady, setIsAuthReady] = useState(false)\n const [isSignedIn, setIsSignedIn] = useState<boolean>(false)\n const [token, setToken] = useState<Token | null>(null)\n const [user, setUser] = useState<User>({})\n const [shouldConnect, setShouldConnect] = useState<boolean>(false)\n const [isReady, setIsReady] = useState<boolean>(false)\n\n const [dbStatus, setDbStatus] = useState<DBStatus>(DBStatus.OFFLINE)\n const [error, setError] = useState<ErrorObject | null>(null)\n\n const syncRef = useRef<BasicSync | null>(null);\n\n useEffect(() => {\n function initDb(options: { shouldConnect: boolean }) {\n if (!syncRef.current) {\n log('Initializing Basic DB')\n syncRef.current = new BasicSync('basicdb', { schema: schema });\n \n syncRef.current.syncable.on('statusChanged', (status: number, url: string) => {\n setDbStatus(getSyncStatus(status))\n })\n \n syncRef.current.syncable.getStatus().then((status) => {\n setDbStatus(getSyncStatus(status))\n })\n\n if (options.shouldConnect) { \n setShouldConnect(true)\n } else { \n log('Sync is disabled')\n }\n\n setIsReady(true)\n\n // log('db is open', syncRef.current.isOpen())\n // syncRef.current.open()\n // .then(() => {\n // log(\"is open now:\", syncRef.current.isOpen())\n // })\n }\n }\n\n async function checkSchema() {\n const valid = validateSchema(schema)\n if (!valid.valid) {\n log('Basic Schema is invalid!', valid.errors)\n console.group('Schema Errors')\n let errorMessage = ''\n valid.errors.forEach((error, index) => {\n log(`${index + 1}:`, error.message, ` - at ${error.instancePath}`)\n errorMessage += `${index + 1}: ${error.message} - at ${error.instancePath}\\n`\n })\n console.groupEnd('Schema Errors')\n setError({\n code: 'schema_invalid',\n title: 'Basic Schema is invalid!',\n message: errorMessage\n })\n setIsReady(true)\n return null\n }\n\n\n let schemaStatus = { valid: false }\n if (schema.version !== 0) {\n schemaStatus = await getSchemaStatus(schema)\n log('schemaStatus', schemaStatus)\n }else { \n log(\"schema not published - at version 0\")\n }\n\n if (schemaStatus.valid) {\n initDb({ shouldConnect: true })\n } else {\n log('Schema is invalid!', schemaStatus)\n initDb({ shouldConnect: false })\n }\n \n checkForNewVersion()\n }\n\n if (schema) {\n checkSchema()\n } else {\n setIsReady(true)\n }\n }, []);\n\n\n useEffect(() => {\n if (token && syncRef.current && isSignedIn && shouldConnect) {\n connectToDb()\n }\n }, [isSignedIn, shouldConnect])\n\n useEffect(() => {\n localStorage.setItem('basic_debug', debug ? 'true' : 'false')\n\n try {\n if (window.location.search.includes('code')) {\n let code = window.location?.search?.split('code=')[1].split('&')[0]\n\n const state = localStorage.getItem('basic_auth_state')\n if (!state || state !== window.location.search.split('state=')[1].split('&')[0]) {\n log('error: auth state does not match')\n setIsAuthReady(true)\n\n localStorage.removeItem('basic_auth_state')\n window.history.pushState({}, document.title, \"/\");\n return\n }\n\n localStorage.removeItem('basic_auth_state')\n\n fetchToken(code) \n } else { \n let cookie_token = getCookie('basic_token')\n if (cookie_token !== '') {\n setToken(JSON.parse(cookie_token))\n } else { \n setIsAuthReady(true)\n }\n }\n\n\n } catch (e) {\n log('error getting cookie', e)\n }\n }, [])\n\n useEffect(() => {\n async function fetchUser(acc_token: string) {\n console.info('fetching user')\n const user = await fetch('https://api.basic.tech/auth/userInfo', {\n method: 'GET',\n headers: {\n 'Authorization': `Bearer ${acc_token}`\n }\n })\n .then(response => response.json())\n .catch(error => log('Error:', error))\n\n if (user.error) {\n log('error fetching user', user.error)\n // refreshToken()\n return\n } else {\n // log('user', user)\n document.cookie = `basic_token=${JSON.stringify(token)}; Secure; SameSite=Strict`;\n \n if (window.location.search.includes('code')) {\n window.history.pushState({}, document.title, \"/\");\n }\n \n setUser(user)\n setIsSignedIn(true)\n\n setIsAuthReady(true)\n }\n }\n\n async function checkToken() {\n if (!token) {\n log('error: no user token found')\n\n setIsAuthReady(true)\n return\n }\n\n const decoded = jwtDecode(token?.access_token)\n const isExpired = decoded.exp && decoded.exp < Date.now() / 1000\n\n if (isExpired) {\n log('token is expired - refreshing ...')\n const newToken = await fetchToken(token?.refresh)\n fetchUser(newToken.access_token)\n } else {\n fetchUser(token.access_token)\n }\n }\n\n if (token) {\n checkToken()\n } \n }, [token])\n\n const connectToDb = async () => {\n const tok = await getToken()\n if (!tok) {\n log('no token found')\n return\n }\n\n log('connecting to db...')\n\n // TODO: handle if signed out after connect() is already called\n\n syncRef.current.connect({ access_token: tok })\n .catch((e) => {\n log('error connecting to db', e)\n })\n }\n\n const getSignInLink = () => {\n log('getting sign in link...')\n\n const randomState = Math.random().toString(36).substring(6);\n localStorage.setItem('basic_auth_state', randomState)\n\n let baseUrl = \"https://api.basic.tech/auth/authorize\"\n baseUrl += `?client_id=${project_id}`\n baseUrl += `&redirect_uri=${encodeURIComponent(window.location.href)}`\n baseUrl += `&response_type=code`\n baseUrl += `&scope=profile`\n baseUrl += `&state=${randomState}`\n\n return baseUrl;\n }\n\n const signin = () => {\n log('signing in: ', getSignInLink())\n const signInLink = getSignInLink()\n //todo: change to the other thing?\n window.location.href = signInLink;\n }\n\n const signout = () => {\n log('signing out!')\n setUser({})\n setIsSignedIn(false)\n setToken(null)\n document.cookie = `basic_token=; Secure; SameSite=Strict`;\n localStorage.removeItem('basic_auth_state')\n\n // if (syncRef.current) {\n // // WIP - BUG - sometimes connects even after signout\n // syncRef.current.disconnect()\n\n \n // }\n if (syncRef.current) {\n (async () => {\n try {\n await syncRef.current.close()\n await syncRef.current.delete({disableAutoOpen: false})\n syncRef.current = null\n window?.location?.reload()\n } catch (error) {\n console.error('Error during database cleanup:', error)\n }\n })()\n }\n }\n\n const getToken = async (): Promise<string> => {\n log('getting token...')\n\n if (!token) {\n log('no token found')\n throw new Error('no token found')\n }\n\n const decoded = jwtDecode(token?.access_token)\n const isExpired = decoded.exp && decoded.exp < Date.now() / 1000\n\n if (isExpired) {\n log('token is expired - refreshing ...')\n const newToken = await fetchToken(token?.refresh)\n return newToken?.access_token || ''\n }\n\n return token?.access_token || ''\n }\n\n function getCookie(name: string) {\n let cookieValue = '';\n if (document.cookie && document.cookie !== '') {\n const cookies = document.cookie.split(';');\n for (let i = 0; i < cookies.length; i++) {\n const cookie = cookies[i].trim();\n if (cookie.substring(0, name.length + 1) === (name + '=')) {\n cookieValue = decodeURIComponent(cookie.substring(name.length + 1));\n break;\n }\n }\n }\n return cookieValue;\n }\n\n const fetchToken = async (code: string) => {\n const token = await fetch('https://api.basic.tech/auth/token', {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json'\n },\n body: JSON.stringify({ code: code })\n })\n .then(response => response.json())\n .catch(error => log('Error:', error))\n\n if (token.error) {\n log('error fetching token', token.error)\n return\n } else {\n // log('token', token)\n setToken(token)\n }\n return token\n }\n\n\n const db_ = (tableName: string) => {\n const checkSignIn = () => {\n if (!isSignedIn) {\n throw new Error('cannot use db. user not logged in.')\n }\n }\n\n return {\n get: async () => {\n checkSignIn()\n const tok = await getToken()\n return get({ projectId: project_id, accountId: user.id, tableName: tableName, token: tok })\n },\n add: async (value: any) => {\n checkSignIn()\n const tok = await getToken()\n return add({ projectId: project_id, accountId: user.id, tableName: tableName, value: value, token: tok })\n },\n update: async (id: string, value: any) => {\n checkSignIn()\n const tok = await getToken()\n return update({ projectId: project_id, accountId: user.id, tableName: tableName, id: id, value: value, token: tok })\n },\n delete: async (id: string) => {\n checkSignIn()\n const tok = await getToken()\n return deleteRecord({ projectId: project_id, accountId: user.id, tableName: tableName, id: id, token: tok })\n }\n\n }\n\n }\n\n const noDb = ({ \n collection: () => {\n throw new Error('no basicdb found - initialization failed. double check your schema.')\n }\n })\n\n return (\n <BasicContext.Provider value={{\n unicorn: \"🦄\",\n isAuthReady,\n isSignedIn,\n user,\n signout,\n signin,\n getToken,\n getSignInLink,\n db: syncRef.current ? syncRef.current : noDb,\n dbStatus\n }}>\n \n {error && <ErrorDisplay error={error} />}\n {isReady && children}\n </BasicContext.Provider>\n )\n}\n\nfunction ErrorDisplay({ error }: { error: ErrorObject }) {\n return <div style={{ \n position: 'absolute',\n top: 20, \n left: 20,\n color: 'black',\n backgroundColor: '#f8d7da',\n border: '1px solid #f5c6cb',\n borderRadius: '4px',\n padding: '20px',\n maxWidth: '400px',\n margin: '20px auto',\n boxShadow: '0 2px 4px rgba(0, 0, 0, 0.1)',\n fontFamily: 'monospace', \n }}>\n <h3 style={{fontSize: '0.8rem', opacity: 0.8}}>code: {error.code}</h3>\n <h1 style={{fontSize: '1.2rem', lineHeight: '1.5'}}>{error.title}</h1>\n <p>{error.message}</p>\n </div>\n}\n\n/*\npossible errors: \n- projectid missing / invalid\n- schema missing / invalid\n*/\n\nexport function useBasic() {\n return useContext(BasicContext);\n}\n","\"use client\"\n\nimport { v7 as uuidv7 } from 'uuid';\nimport { Dexie, PromiseExtended } from 'dexie';\nimport 'dexie-syncable';\nimport 'dexie-observable';\n\nimport { syncProtocol } from './syncProtocol'\nimport { log } from '../config'\n\nimport { validateSchema, validateData } from '@basictech/schema'\nsyncProtocol()\n\n\n// const DexieSyncStatus = {\n// \"-1\": \"ERROR\",\n// \"0\": \"OFFLINE\",\n// \"1\": \"CONNECTING\",\n// \"2\": \"ONLINE\",\n// \"3\": \"SYNCING\",\n// \"4\": \"ERROR_WILL_RETRY\"\n// }\n\n\n// const SERVER_URL = \"https://pds.basic.id\"\n\n\nexport class BasicSync extends Dexie {\n basic_schema: any\n\n constructor(name: string, options: any) {\n super(name, options);\n\n // --- INIT SCHEMA --- // \n\n //todo: handle versions?\n\n // TODO: validate schema\n this.basic_schema = options.schema\n this.version(1).stores(this._convertSchemaToDxSchema(this.basic_schema))\n\n this.version(2).stores({})\n // this.verssion\n\n\n // create an alias for toArray\n // @ts-ignore\n this.Collection.prototype.get = this.Collection.prototype.toArray\n\n\n // --- SYNC --- // \n\n // this.syncable.on(\"statusChanged\", (status, url) => { \n // console.log(\"statusChanged\", status, url)\n // })\n\n }\n\n async connect({ access_token }: { access_token: string }) {\n const WS_URL = `wss://pds.basic.id/ws`\n\n log('Connecting to', WS_URL)\n\n await this.updateSyncNodes();\n \n log('Starting connection...')\n return this.syncable.connect(\"websocket\", WS_URL, { authToken: access_token, schema: this.basic_schema });\n }\n\n async disconnect() {\n const WS_URL = `wss://pds.basic.id/ws`\n\n return this.syncable.disconnect(WS_URL) \n }\n\n private async updateSyncNodes() {\n try {\n const syncNodes = await this.table('_syncNodes').toArray();\n const localSyncNodes = syncNodes.filter(node => node.type === 'local');\n log('Local sync nodes:', localSyncNodes);\n\n if (localSyncNodes.length > 1) {\n\n \n const largestNodeId = Math.max(...localSyncNodes.map(node => node.id));\n // Check if the largest node is already the master\n const largestNode = localSyncNodes.find(node => node.id === largestNodeId);\n if (largestNode && largestNode.isMaster === 1) {\n log('Largest node is already the master. No changes needed.');\n return; // Exit the function early as no changes are needed\n }\n\n\n log('Largest node id:', largestNodeId);\n log('HEISENBUG: More than one local sync node found.')\n\n for (const node of localSyncNodes) {\n log(`Local sync node keys:`, node.id, node.isMaster);\n await this.table('_syncNodes').update(node.id, { isMaster: node.id === largestNodeId ? 1 : 0 });\n\n log(`HEISENBUG: Setting ${node.id} to ${node.id === largestNodeId ? 'master' : '0'}`);\n }\n\n // add delay to ensure sync nodes are updated // i dont think this helps?\n await new Promise(resolve => setTimeout(resolve, 1000));\n\n if (typeof window !== 'undefined') {\n window.location.reload();\n }\n }\n\n log('Sync nodes updated');\n } catch (error) {\n console.error('Error updating _syncNodes table:', error);\n }\n }\n\n handleStatusChange(fn: any) {\n this.syncable.on(\"statusChanged\", fn)\n }\n\n\n _convertSchemaToDxSchema(schema: any) {\n const stores = Object.entries(schema.tables).map(([key, table]: any) => {\n\n const indexedFields = Object.entries(table.fields).filter(([key, field]: any) => field.indexed).map(([key, field]: any) => `,${key}`).join('')\n return {\n [key]: 'id' + indexedFields\n }\n })\n\n return Object.assign({}, ...stores)\n }\n\n debugeroo() {\n // console.log(\"debugeroo\", this.syncable)\n\n // this.syncable.list().then(x => console.log(x))\n\n // this.syncable\n return this.syncable\n }\n\n collection(name: string) {\n // TODO: check against schema\n\n return {\n\n /**\n * Returns the underlying Dexie table\n * @type {Dexie.Table}\n */\n ref: this.table(name),\n\n // --- WRITE ---- // \n add: (data: any) => {\n // log(\"Adding data to\", name, data)\n\n const valid = validateData(this.basic_schema, name, data)\n if (!valid.valid) {\n log('Invalid data', valid)\n return Promise.reject({ ... valid })\n }\n\n return this.table(name).add({\n id: uuidv7(),\n ...data\n })\n\n },\n\n put: (data: any) => {\n const valid = validateData(this.basic_schema, name, data)\n if (!valid.valid) {\n log('Invalid data', valid)\n return Promise.reject({ ... valid })\n }\n\n return this.table(name).put({\n id: uuidv7(),\n ...data\n })\n },\n\n update: (id: string, data: any) => {\n const valid = validateData(this.basic_schema, name, data, false)\n if (!valid.valid) {\n log('Invalid data', valid)\n return Promise.reject({ ... valid })\n }\n\n return this.table(name).update(id, data)\n },\n\n delete: (id: string) => {\n return this.table(name).delete(id)\n },\n\n\n // --- READ ---- // \n\n get: async (id: string) => {\n return this.table(name).get(id) \n },\n\n getAll: async () => {\n return this.table(name).toArray();\n },\n\n // --- QUERY ---- // \n // TODO: lots to do here. simplifing creating querie, filtering/ordering/limit, and execute\n\n query: () => this.table(name),\n\n filter: (fn: any) => this.table(name).filter(fn).toArray(),\n\n }\n }\n}\n\nclass QueryMethod { \n\n}\n","\"use client\"\nimport { Dexie } from \"dexie\";\nimport { log } from \"../config\";\n\nexport const syncProtocol = function () {\n log(\"Initializing syncProtocol\");\n // Constants:\n var RECONNECT_DELAY = 5000; // Reconnect delay in case of errors such as network down.\n\n Dexie.Syncable.registerSyncProtocol(\"websocket\", {\n sync: function (\n context,\n url,\n options,\n baseRevision,\n syncedRevision,\n changes,\n partial,\n applyRemoteChanges,\n onChangesAccepted,\n onSuccess,\n onError,\n ) {\n // The following vars are needed because we must know which callback to ack when server sends it's ack to us.\n var requestId = 0;\n var acceptCallbacks = {};\n\n // Connect the WebSocket to given url:\n log(\"Connecting to\", url)\n var ws = new WebSocket(url);\n\n // console.log(\"ws OPTIONS\", options);\n\n // sendChanges() method:\n function sendChanges(changes, baseRevision, partial, onChangesAccepted) {\n log(\"sendChanges\", changes.length, baseRevision);\n ++requestId;\n acceptCallbacks[requestId.toString()] = onChangesAccepted;\n\n // In this example, the server expects the following JSON format of the request:\n // {\n // type: \"changes\"\n // baseRevision: baseRevision,\n // changes: changes,\n // partial: partial,\n // requestId: id\n // }\n // To make the sample simplified, we assume the server has the exact same specification of how changes are structured.\n // In real world, you would have to pre-process the changes array to fit the server specification.\n // However, this example shows how to deal with the WebSocket to fullfill the API.\n\n ws.send(\n JSON.stringify({\n type: \"changes\",\n changes: changes,\n partial: partial,\n baseRevision: baseRevision,\n requestId: requestId,\n }),\n );\n }\n\n\n\n // When WebSocket opens, send our changes to the server.\n ws.onopen = function (event) {\n // Initiate this socket connection by sending our clientIdentity. If we dont have a clientIdentity yet,\n // server will call back with a new client identity that we should use in future WebSocket connections.\n \n // send the schema to the server\n log(\"Opening socket - sending clientIdentity\", context.clientIdentity);\n ws.send(\n JSON.stringify({\n type: \"clientIdentity\",\n clientIdentity: context.clientIdentity || null,\n authToken: options.authToken,\n schema: options.schema\n }),\n );\n\n };\n\n // If network down or other error, tell the framework to reconnect again in some time:\n ws.onerror = function (event) {\n ws.close();\n log(\"ws.onerror\", event);\n onError(event?.message, RECONNECT_DELAY);\n };\n\n // If socket is closed (network disconnected), inform framework and make it reconnect\n ws.onclose = function (event) {\n // console.log('🙅 ws.onclose', event)\n onError(\"Socket closed: \" + event.reason, RECONNECT_DELAY);\n };\n\n // isFirstRound: Will need to call onSuccess() only when we are in sync the first time.\n // onSuccess() will unblock Dexie to be used by application code.\n // If for example app code writes: db.friends.where('shoeSize').above(40).toArray(callback), the execution of that query\n // will not run until we have called onSuccess(). This is because we want application code to get results that are as\n // accurate as possible. Specifically when connected the first time and the entire DB is being synced down to the browser,\n // it is important that queries starts running first when db is in sync.\n var isFirstRound = true;\n // When message arrive from the server, deal with the message accordingly:\n ws.onmessage = function (event) {\n try {\n // Assume we have a server that should send JSON messages of the following format:\n // {\n // type: \"clientIdentity\", \"changes\", \"ack\" or \"error\"\n // clientIdentity: unique value for our database client node to persist in the context. (Only applicable if type=\"clientIdentity\")\n // message: Error message (Only applicable if type=\"error\")\n // requestId: ID of change request that is acked by the server (Only applicable if type=\"ack\" or \"error\")\n // changes: changes from server (Only applicable if type=\"changes\")\n // lastRevision: last revision of changes sent (applicable if type=\"changes\")\n // partial: true if server has additionalChanges to send. False if these changes were the last known. (applicable if type=\"changes\")\n // }\n var requestFromServer = JSON.parse(event.data);\n log(\"requestFromServer\", requestFromServer, { acceptCallback, isFirstRound });\n\n if (requestFromServer.type == \"clientIdentity\") {\n context.clientIdentity = requestFromServer.clientIdentity;\n context.save();\n\n sendChanges(changes, baseRevision, partial, onChangesAccepted);\n\n ws.send(\n JSON.stringify({\n type: \"subscribe\",\n syncedRevision: syncedRevision,\n }),\n );\n } else if (requestFromServer.type == \"changes\") {\n applyRemoteChanges(\n requestFromServer.changes,\n requestFromServer.currentRevision,\n requestFromServer.partial,\n );\n if (isFirstRound && !requestFromServer.partial) {\n // Since this is the first sync round and server sais we've got all changes - now is the time to call onsuccess()\n onSuccess({\n // Specify a react function that will react on additional client changes\n react: function (\n changes,\n baseRevision,\n partial,\n onChangesAccepted,\n ) {\n sendChanges(\n changes,\n baseRevision,\n partial,\n onChangesAccepted,\n );\n },\n // Specify a disconnect function that will close our socket so that we dont continue to monitor changes.\n disconnect: function () {\n ws.close();\n },\n });\n isFirstRound = false;\n }\n } else if (requestFromServer.type == \"ack\") {\n var requestId = requestFromServer.requestId;\n var acceptCallback = acceptCallbacks[requestId.toString()];\n acceptCallback(); // Tell framework that server has acknowledged the changes sent.\n delete acceptCallbacks[requestId.toString()];\n } else if (requestFromServer.type == \"error\") {\n var requestId = requestFromServer.requestId;\n ws.close();\n onError(requestFromServer.message, Infinity); // Don't reconnect - an error in application level means we have done something wrong.\n } else {\n log(\"unknown message\", requestFromServer);\n ws.close();\n onError(\"unknown message\", Infinity);\n }\n } catch (e) {\n ws.close();\n log(\"caught error\", e)\n onError(e, Infinity); // Something went crazy. Server sends invalid format or our code is buggy. Dont reconnect - it would continue failing.\n }\n };\n },\n });\n};\n","\nexport const SERVER_URL = \"https://api.basic.tech\"\n// export const SERVER_URL = \"http://localhost:3003\"\n\n\nexport const log = (...args: any[]) => {\n try { \n if (localStorage.getItem('basic_debug') === 'true') {\n console.log('[basic]', ...args)\n }\n } catch (e) {\n // console.log('error logging', e)\n }\n}\n\n// export const log = (message: string, ...args: any[]) => {\n// try {\n// if (process.env.NODE_ENV === 'development') {\n// const stack = new Error().stack;\n// const caller = stack?.split('\\n')[2]?.trim();\n// console.log(`[basic] ${message}`, ...args);\n// // console.log(`[stack] ${caller}`);\n// }\n// } catch (e) {\n// console.error('Error in logWithStack:', e);\n// }\n// }\n","//@ts-nocheck\n\nconst baseUrl = 'https://api.basic.tech';\n// const baseUrl = 'http://localhost:3000';\n\n\nasync function get({ projectId, accountId, tableName, token }) {\n const url = `${baseUrl}/project/${projectId}/db/${accountId}/${tableName}`;\n const response = await fetch(url, {\n headers: {\n 'Authorization': `Bearer ${token}`\n }\n });\n return response.json();\n}\n\nasync function add({ projectId, accountId, tableName, value, token }) {\n const url = `${baseUrl}/project/${projectId}/db/${accountId}/${tableName}`;\n const response = await fetch(url, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': `Bearer ${token}`\n },\n body: JSON.stringify({\"value\": value})\n });\n return response.json();\n}\n\nasync function update({ projectId, accountId, tableName, id, value, token }) {\n const url = `${baseUrl}/project/${projectId}/db/${accountId}/${tableName}/${id}`;\n const response = await fetch(url, {\n method: 'PATCH',\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': `Bearer ${token}`\n },\n body: JSON.stringify({id: id, value: value})\n });\n return response.json();\n}\n\nasync function deleteRecord({ projectId, accountId, tableName, id, token }) {\n const url = `${baseUrl}/project/${projectId}/db/${accountId}/${tableName}/${id}`;\n const response = await fetch(url, {\n method: 'DELETE',\n headers: {\n 'Authorization': `Bearer ${token}`\n }\n });\n return response.json();\n}\n\nexport { get, add, update, deleteRecord };\n\n","{\n \"name\": \"@basictech/react\",\n \"version\": \"0.6.0\",\n \"description\": \"\",\n \"main\": \"dist/index.js\",\n \"module\": \"dist/index.mjs\",\n \"types\": \"dist/index.d.ts\",\n \"private\": false,\n \"publishConfig\": {\n \"access\": \"public\"\n },\n \"scripts\": {\n \"build\": \"tsup\",\n \"dev\": \"tsup --watch\",\n \"test\": \"echo \\\"Error: no test specified\\\" && exit 1\"\n },\n \"author\": \"\",\n \"license\": \"ISC\",\n \"dependencies\": {\n \"ajv\": \"^8.17.1\",\n \"dexie\": \"^4.0.8\",\n \"dexie-observable\": \"^4.0.1-beta.13\",\n \"dexie-react-hooks\": \"^1.1.7\",\n \"dexie-syncable\": \"^4.0.1-beta.13\",\n \"jwt-decode\": \"^4.0.0\",\n \"uuid\": \"^10.0.0\",\n \"@basictech/schema\": \"0.6.0\"\n },\n \"devDependencies\": {\n \"@repo/typescript-config\": \"*\",\n \"tsup\": \"^7.2.0\",\n \"typescript\": \"^5.0.0\"\n },\n \"peerDependencies\": {\n \"react\": \"^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0\"\n }\n}\n","import { useState } from \"react\";\nimport { useBasic, BasicProvider } from \"./AuthContext\";\nimport { useLiveQuery as useQuery } from \"dexie-react-hooks\";\n\n// const useQuery = (queryable: any) => {\n// const [loading, setLoading] = useState(true)\n// const [error, setError] = useState<Error | null>(null)\n\n// const result = useLiveQuery(async () => {\n// try {\n// setLoading(true)\n// setError(null)\n \n// // if (typeof queryable === 'function') {\n// // return await queryable()\n// // }\n// return queryable\n \n// } catch (err) {\n// setError(err instanceof Error ? err : new Error('Unknown error'))\n// return undefined\n// } finally {\n// setLoading(false)\n// }\n// }, [queryable])\n\n// return {\n// data: result,\n// loading,\n// error\n// }\n// }\n\n\n\nexport {\n useBasic, BasicProvider, useQuery\n}\n"],"mappings":"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","names":["Dexie","changes","baseRevision","partial","onChangesAccepted","requestId","Dexie","key","validateSchema","validateSchema","error","user","baseUrl","token"]}
1
+ {"version":3,"sources":["../src/AuthContext.tsx","../src/sync/index.ts","../src/sync/syncProtocol.js","../src/config.ts","../src/db_ts.ts","../package.json","../src/updater/versionUpdater.ts","../src/updater/updateMigrations.ts","../src/utils/storage.ts","../src/utils/network.ts","../src/utils/schema.ts","../src/index.ts"],"sourcesContent":["import React, { createContext, useContext, useEffect, useState, useRef } from 'react'\nimport { jwtDecode } from 'jwt-decode'\n\nimport { BasicSync } from './sync'\nimport { BasicDBSDK } from './db_ts'\n\nimport { log } from './config'\nimport { version as currentVersion } from '../package.json'\nimport { createVersionUpdater } from './updater/versionUpdater'\nimport { getMigrations } from './updater/updateMigrations'\nimport { BasicStorage, LocalStorageAdapter, STORAGE_KEYS, getCookie, setCookie, clearCookie } from './utils/storage'\nimport { isDevelopment, checkForNewVersion, cleanOAuthParamsFromUrl, getSyncStatus } from './utils/network'\nimport { getSchemaStatus, validateAndCheckSchema } from './utils/schema'\n\nexport type { BasicStorage, LocalStorageAdapter } from './utils/storage'\n\n\ntype BasicSyncType = {\n basic_schema: any;\n connect: (options: { access_token: string }) => void;\n debugeroo: () => void;\n collection: (name: string) => {\n ref: {\n toArray: () => Promise<any[]>;\n count: () => Promise<number>;\n };\n };\n [key: string]: any;\n};\n\n\nenum DBStatus {\n LOADING = \"LOADING\",\n OFFLINE = \"OFFLINE\",\n CONNECTING = \"CONNECTING\",\n ONLINE = \"ONLINE\",\n SYNCING = \"SYNCING\",\n ERROR = \"ERROR\"\n}\n\ntype User = {\n name?: string,\n email?: string,\n id?: string,\n primaryEmailAddress?: {\n emailAddress: string\n },\n fullName?: string\n}\ntype Token = {\n access_token: string,\n token_type: string,\n expires_in: number,\n refresh_token: string,\n}\n\nexport const BasicContext = createContext<{\n unicorn: string,\n isAuthReady: boolean,\n isSignedIn: boolean,\n user: User | null,\n signout: () => Promise<void>,\n signin: () => Promise<void>,\n signinWithCode: (code: string, state?: string) => Promise<{ success: boolean, error?: string }>,\n getToken: () => Promise<string>,\n getSignInLink: (redirectUri?: string) => Promise<string>,\n db: any,\n remoteDb: any,\n dbStatus: DBStatus\n}>({\n unicorn: \"🦄\",\n isAuthReady: false,\n isSignedIn: false,\n user: null,\n signout: () => Promise.resolve(),\n signin: () => Promise.resolve(),\n signinWithCode: () => new Promise(() => { }),\n getToken: () => new Promise(() => { }),\n getSignInLink: () => Promise.resolve(\"\"),\n db: {},\n remoteDb: {},\n dbStatus: DBStatus.LOADING\n});\n\ntype ErrorObject = {\n code: string;\n title: string;\n message: string;\n}\n\nexport function BasicProvider({\n children,\n project_id,\n schema,\n debug = false,\n storage\n}: {\n children: React.ReactNode,\n project_id?: string,\n schema?: any,\n debug?: boolean,\n storage?: BasicStorage\n}) {\n const [isAuthReady, setIsAuthReady] = useState(false)\n const [isSignedIn, setIsSignedIn] = useState<boolean>(false)\n const [token, setToken] = useState<Token | null>(null)\n const [user, setUser] = useState<User>({})\n \n const [shouldConnect, setShouldConnect] = useState<boolean>(false)\n const [isReady, setIsReady] = useState<boolean>(false)\n\n const [dbStatus, setDbStatus] = useState<DBStatus>(DBStatus.OFFLINE)\n const [error, setError] = useState<ErrorObject | null>(null)\n const [isOnline, setIsOnline] = useState<boolean>(navigator.onLine)\n const [pendingRefresh, setPendingRefresh] = useState<boolean>(false)\n\n const syncRef = useRef<BasicSync | null>(null);\n const remoteDbRef = useRef<BasicDBSDK<any> | null>(null);\n const storageAdapter = storage || new LocalStorageAdapter();\n\n const isDevMode = () => isDevelopment(debug)\n\n const cleanOAuthParams = () => cleanOAuthParamsFromUrl()\n\n useEffect(() => {\n const handleOnline = () => {\n log('Network came back online')\n setIsOnline(true)\n if (pendingRefresh) {\n log('Retrying pending token refresh')\n setPendingRefresh(false)\n if (token) {\n const refreshToken = token.refresh_token || localStorage.getItem('basic_refresh_token')\n if (refreshToken) {\n fetchToken(refreshToken).catch(error => {\n log('Retry refresh failed:', error)\n })\n }\n }\n }\n }\n\n const handleOffline = () => {\n log('Network went offline')\n setIsOnline(false)\n }\n\n window.addEventListener('online', handleOnline)\n window.addEventListener('offline', handleOffline)\n\n return () => {\n window.removeEventListener('online', handleOnline)\n window.removeEventListener('offline', handleOffline)\n }\n }, [pendingRefresh, token])\n\n useEffect(() => {\n function initDb(options: { shouldConnect: boolean }) {\n if (!syncRef.current) {\n log('Initializing Basic DB')\n syncRef.current = new BasicSync('basicdb', { schema: schema });\n\n syncRef.current.syncable.on('statusChanged', (status: number, url: string) => {\n setDbStatus(getSyncStatus(status) as DBStatus)\n })\n\n // syncRef.current.syncable.getStatus().then((status: number) => {\n // setDbStatus(getSyncStatus(status) as DBStatus)\n // })\n\n if (options.shouldConnect) {\n setShouldConnect(true)\n } else {\n log('Sync is disabled')\n }\n\n setIsReady(true)\n }\n }\n\n async function checkSchema() {\n const result = await validateAndCheckSchema(schema)\n\n if (!result.isValid) {\n let errorMessage = ''\n if (result.errors) {\n result.errors.forEach((error, index) => {\n errorMessage += `${index + 1}: ${error.message} - at ${error.instancePath}\\n`\n })\n }\n setError({\n code: 'schema_invalid',\n title: 'Basic Schema is invalid!',\n message: errorMessage\n })\n setIsReady(true)\n return null\n }\n\n if (result.schemaStatus.valid) {\n initDb({ shouldConnect: true })\n } else {\n log('Schema is invalid!', result.schemaStatus)\n initDb({ shouldConnect: false })\n }\n\n checkForNewVersion()\n }\n\n if (schema) {\n checkSchema()\n } else {\n setIsReady(true)\n }\n }, []);\n\n useEffect(() => {\n async function connectToDb() {\n if (token && syncRef.current && isSignedIn && shouldConnect) {\n const tok = await getToken()\n if (!tok) {\n log('no token found')\n return\n }\n\n log('connecting to db...')\n\n syncRef.current?.connect({ access_token: tok })\n .catch((e) => {\n log('error connecting to db', e)\n })\n }\n }\n connectToDb()\n\n }, [isSignedIn, shouldConnect])\n\n // Initialize remote database when we have a token\n useEffect(() => {\n if (project_id && schema && token?.access_token && !remoteDbRef.current) {\n log('Initializing Remote DB SDK')\n \n remoteDbRef.current = new BasicDBSDK({\n project_id: project_id,\n schema: schema,\n getToken: () => getToken(),\n baseUrl: 'https://api.basic.tech'\n });\n }\n }, [token, project_id, schema])\n\n useEffect(() => {\n const initializeAuth = async () => {\n await storageAdapter.set(STORAGE_KEYS.DEBUG, debug ? 'true' : 'false')\n\n try {\n const versionUpdater = createVersionUpdater(storageAdapter, currentVersion, getMigrations())\n const updateResult = await versionUpdater.checkAndUpdate()\n\n if (updateResult.updated) {\n log(`App updated from ${updateResult.fromVersion} to ${updateResult.toVersion}`)\n } else {\n log(`App version ${updateResult.toVersion} is current`)\n }\n } catch (error) {\n log('Version update failed:', error)\n }\n\n try {\n if (window.location.search.includes('code')) {\n let code = window.location?.search?.split('code=')[1]?.split('&')[0]\n if (!code) return\n\n const state = await storageAdapter.get(STORAGE_KEYS.AUTH_STATE)\n const urlState = window.location.search.split('state=')[1]?.split('&')[0]\n if (!state || state !== urlState) {\n log('error: auth state does not match')\n setIsAuthReady(true)\n\n await storageAdapter.remove(STORAGE_KEYS.AUTH_STATE)\n cleanOAuthParams()\n return\n }\n\n await storageAdapter.remove(STORAGE_KEYS.AUTH_STATE)\n cleanOAuthParams()\n\n fetchToken(code).catch((error) => {\n log('Error fetching token:', error)\n })\n } else {\n const refreshToken = await storageAdapter.get(STORAGE_KEYS.REFRESH_TOKEN)\n if (refreshToken) {\n log('Found refresh token in storage, attempting to refresh access token')\n fetchToken(refreshToken).catch((error) => {\n log('Error fetching refresh token:', error)\n })\n } else {\n let cookie_token = getCookie('basic_token')\n if (cookie_token !== '') {\n const tokenData = JSON.parse(cookie_token)\n setToken(tokenData)\n if (tokenData.refresh_token) {\n await storageAdapter.set(STORAGE_KEYS.REFRESH_TOKEN, tokenData.refresh_token)\n }\n } else {\n const cachedUserInfo = await storageAdapter.get(STORAGE_KEYS.USER_INFO)\n if (cachedUserInfo) {\n try {\n const userData = JSON.parse(cachedUserInfo)\n setUser(userData)\n setIsSignedIn(true)\n log('Loaded cached user info for offline mode')\n } catch (error) {\n log('Error parsing cached user info:', error)\n }\n }\n setIsAuthReady(true)\n }\n }\n }\n\n } catch (e) {\n log('error getting token', e)\n }\n }\n\n initializeAuth()\n }, [])\n\n useEffect(() => {\n async function fetchUser(acc_token: string) {\n console.info('fetching user')\n const user = await fetch('https://api.basic.tech/auth/userInfo', {\n method: 'GET',\n headers: {\n 'Authorization': `Bearer ${acc_token}`\n }\n })\n .then(response => response.json())\n .catch(error => log('Error:', error))\n\n if (user.error) {\n log('error fetching user', user.error)\n return\n } else {\n if (token?.refresh_token) {\n await storageAdapter.set(STORAGE_KEYS.REFRESH_TOKEN, token.refresh_token)\n }\n\n await storageAdapter.set(STORAGE_KEYS.USER_INFO, JSON.stringify(user))\n log('Cached user info in storage')\n\n setCookie('basic_access_token', token?.access_token || '', { httpOnly: false });\n setCookie('basic_token', JSON.stringify(token));\n\n setUser(user)\n setIsSignedIn(true)\n\n setIsAuthReady(true)\n }\n }\n\n async function checkToken() {\n if (!token) {\n log('error: no user token found')\n\n setIsAuthReady(true)\n return\n }\n\n const decoded = jwtDecode(token?.access_token)\n const isExpired = decoded.exp && decoded.exp < Date.now() / 1000\n\n if (isExpired) {\n log('token is expired - refreshing ...')\n try {\n const newToken = await fetchToken(token?.refresh_token || '')\n fetchUser(newToken?.access_token || '')\n } catch (error) {\n log('Failed to refresh token in checkToken:', error)\n\n if ((error as Error).message.includes('offline') || (error as Error).message.includes('Network')) {\n log('Network issue - continuing with expired token until online')\n fetchUser(token?.access_token || '')\n } else {\n setIsAuthReady(true)\n }\n }\n } else {\n fetchUser(token?.access_token || '')\n }\n }\n\n if (token) {\n checkToken()\n }\n }, [token])\n\n const getSignInLink = async (redirectUri?: string) => {\n try {\n log('getting sign in link...')\n\n if (!project_id) {\n throw new Error('Project ID is required to generate sign-in link')\n }\n\n const randomState = Math.random().toString(36).substring(6);\n await storageAdapter.set(STORAGE_KEYS.AUTH_STATE, randomState)\n\n const redirectUrl = redirectUri || window.location.href\n\n if (!redirectUrl || (!redirectUrl.startsWith('http://') && !redirectUrl.startsWith('https://'))) {\n throw new Error('Invalid redirect URI provided')\n }\n\n let baseUrl = \"https://api.basic.tech/auth/authorize\"\n baseUrl += `?client_id=${project_id}`\n baseUrl += `&redirect_uri=${encodeURIComponent(redirectUrl)}`\n baseUrl += `&response_type=code`\n baseUrl += `&scope=profile`\n baseUrl += `&state=${randomState}`\n\n log('Generated sign-in link successfully')\n return baseUrl;\n\n } catch (error) {\n log('Error generating sign-in link:', error)\n throw error\n }\n }\n\n const signin = async () => {\n try {\n log('signing in...')\n\n if (!project_id) {\n log('Error: project_id is required for sign-in')\n throw new Error('Project ID is required for authentication')\n }\n\n const signInLink = await getSignInLink()\n log('Generated sign-in link:', signInLink)\n\n if (!signInLink || !signInLink.startsWith('https://')) {\n log('Error: Invalid sign-in link generated')\n throw new Error('Failed to generate valid sign-in URL')\n }\n\n window.location.href = signInLink\n\n } catch (error) {\n log('Error during sign-in:', error)\n\n if (isDevMode()) {\n setError({\n code: 'signin_error',\n title: 'Sign-in Failed',\n message: (error as Error).message || 'An error occurred during sign-in. Please try again.'\n })\n }\n\n throw error\n }\n }\n\n const signinWithCode = async (code: string, state?: string): Promise<{ success: boolean, error?: string }> => {\n try {\n log('signinWithCode called with code:', code)\n\n if (!code || typeof code !== 'string') {\n return { success: false, error: 'Invalid authorization code' }\n }\n\n if (state) {\n const storedState = await storageAdapter.get(STORAGE_KEYS.AUTH_STATE)\n if (storedState && storedState !== state) {\n log('State parameter mismatch:', { provided: state, stored: storedState })\n return { success: false, error: 'State parameter mismatch' }\n }\n }\n\n await storageAdapter.remove(STORAGE_KEYS.AUTH_STATE)\n cleanOAuthParams()\n\n const token = await fetchToken(code)\n\n if (token) {\n log('signinWithCode successful')\n return { success: true }\n } else {\n return { success: false, error: 'Failed to exchange code for token' }\n }\n } catch (error) {\n log('signinWithCode error:', error)\n return {\n success: false,\n error: (error as Error).message || 'Authentication failed'\n }\n }\n }\n\n const signout = async () => {\n log('signing out!')\n setUser({})\n setIsSignedIn(false)\n setToken(null)\n\n clearCookie('basic_token');\n clearCookie('basic_access_token');\n await storageAdapter.remove(STORAGE_KEYS.AUTH_STATE)\n await storageAdapter.remove(STORAGE_KEYS.REFRESH_TOKEN)\n await storageAdapter.remove(STORAGE_KEYS.USER_INFO)\n if (syncRef.current) {\n (async () => {\n try {\n await syncRef.current?.close()\n await syncRef.current?.delete({ disableAutoOpen: false })\n syncRef.current = null\n window?.location?.reload()\n } catch (error) {\n console.error('Error during database cleanup:', error)\n }\n })()\n }\n }\n\n const getToken = async (): Promise<string> => {\n log('getting token...')\n\n if (!token) {\n // Try to recover from storage refresh token\n const refreshToken = await storageAdapter.get(STORAGE_KEYS.REFRESH_TOKEN)\n if (refreshToken) {\n log('No token in memory, attempting to refresh from storage')\n try {\n const newToken = await fetchToken(refreshToken)\n if (newToken?.access_token) {\n return newToken.access_token\n }\n } catch (error) {\n log('Failed to refresh token from storage:', error)\n\n if ((error as Error).message.includes('offline') || (error as Error).message.includes('Network')) {\n log('Network issue - continuing with potentially expired token')\n const lastToken = localStorage.getItem('basic_access_token')\n if (lastToken) {\n return lastToken\n }\n throw new Error('Network offline - authentication will be retried when online')\n }\n\n throw new Error('Authentication expired. Please sign in again.')\n }\n }\n log('no token found')\n throw new Error('no token found')\n }\n\n const decoded = jwtDecode(token?.access_token)\n const isExpired = decoded.exp && decoded.exp < Date.now() / 1000\n\n if (isExpired) {\n log('token is expired - refreshing ...')\n const refreshToken = token?.refresh_token || await storageAdapter.get(STORAGE_KEYS.REFRESH_TOKEN)\n if (refreshToken) {\n try {\n const newToken = await fetchToken(refreshToken)\n return newToken?.access_token || ''\n } catch (error) {\n log('Failed to refresh expired token:', error)\n\n if ((error as Error).message.includes('offline') || (error as Error).message.includes('Network')) {\n log('Network issue - using expired token until network is restored')\n return token.access_token\n }\n\n throw new Error('Authentication expired. Please sign in again.')\n }\n } else {\n throw new Error('no refresh token available')\n }\n }\n\n return token?.access_token || ''\n }\n\n const fetchToken = async (code: string) => {\n try {\n if (!isOnline) {\n log('Network is offline, marking refresh as pending')\n setPendingRefresh(true)\n throw new Error('Network offline - refresh will be retried when online')\n }\n\n const token = await fetch('https://api.basic.tech/auth/token', {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json'\n },\n body: JSON.stringify({ code: code })\n })\n .then(response => response.json())\n .catch(error => {\n log('Network error fetching token:', error)\n if (!isOnline) {\n setPendingRefresh(true)\n throw new Error('Network offline - refresh will be retried when online')\n }\n throw new Error('Network error during token refresh')\n })\n\n if (token.error) {\n log('error fetching token', token.error)\n\n if (token.error.includes('network') || token.error.includes('timeout')) {\n setPendingRefresh(true)\n throw new Error('Network issue - refresh will be retried when online')\n }\n\n await storageAdapter.remove(STORAGE_KEYS.REFRESH_TOKEN)\n await storageAdapter.remove(STORAGE_KEYS.USER_INFO)\n clearCookie('basic_token');\n clearCookie('basic_access_token');\n\n setUser({})\n setIsSignedIn(false)\n setToken(null)\n setIsAuthReady(true)\n\n throw new Error(`Token refresh failed: ${token.error}`)\n } else {\n setToken(token)\n setPendingRefresh(false)\n\n if (token.refresh_token) {\n await storageAdapter.set(STORAGE_KEYS.REFRESH_TOKEN, token.refresh_token)\n log('Updated refresh token in storage')\n }\n\n setCookie('basic_access_token', token.access_token, { httpOnly: false });\n log('Updated access token in cookie')\n }\n return token\n } catch (error) {\n log('Token refresh error:', error)\n\n if (!(error as Error).message.includes('offline') && !(error as Error).message.includes('Network')) {\n await storageAdapter.remove(STORAGE_KEYS.REFRESH_TOKEN)\n await storageAdapter.remove(STORAGE_KEYS.USER_INFO)\n clearCookie('basic_token');\n clearCookie('basic_access_token');\n\n setUser({})\n setIsSignedIn(false)\n setToken(null)\n setIsAuthReady(true)\n }\n\n throw error\n }\n }\n\n const noDb = ({\n collection: () => {\n throw new Error('no basicdb found - initialization failed. double check your schema.')\n }\n })\n\n return (\n <BasicContext.Provider value={{\n unicorn: \"🦄\",\n isAuthReady,\n isSignedIn,\n user,\n signout,\n signin,\n signinWithCode,\n getToken,\n getSignInLink,\n db: syncRef.current ? syncRef.current : noDb,\n remoteDb: remoteDbRef.current ? remoteDbRef.current : noDb,\n dbStatus\n }}>\n\n {error && isDevMode() && <ErrorDisplay error={error} />}\n {isReady && children}\n </BasicContext.Provider>\n )\n}\n\nfunction ErrorDisplay({ error }: { error: ErrorObject }) {\n return <div style={{\n position: 'absolute',\n top: 20,\n left: 20,\n color: 'black',\n backgroundColor: '#f8d7da',\n border: '1px solid #f5c6cb',\n borderRadius: '4px',\n padding: '20px',\n maxWidth: '400px',\n margin: '20px auto',\n boxShadow: '0 2px 4px rgba(0, 0, 0, 0.1)',\n fontFamily: 'monospace',\n }}>\n <h3 style={{ fontSize: '0.8rem', opacity: 0.8 }}>code: {error.code}</h3>\n <h1 style={{ fontSize: '1.2rem', lineHeight: '1.5' }}>{error.title}</h1>\n <p>{error.message}</p>\n </div>\n}\n\n\nexport function useBasic() {\n return useContext(BasicContext);\n}\n","\"use client\"\n\nimport { v7 as uuidv7 } from 'uuid';\nimport { Dexie, PromiseExtended } from 'dexie';\nimport 'dexie-syncable';\nimport 'dexie-observable';\n\nimport { syncProtocol } from './syncProtocol'\nimport { log } from '../config'\n\nimport { validateSchema, validateData } from '@basictech/schema'\nsyncProtocol()\n\n\n// const DexieSyncStatus = {\n// \"-1\": \"ERROR\",\n// \"0\": \"OFFLINE\",\n// \"1\": \"CONNECTING\",\n// \"2\": \"ONLINE\",\n// \"3\": \"SYNCING\",\n// \"4\": \"ERROR_WILL_RETRY\"\n// }\n\n\n// const SERVER_URL = \"https://pds.basic.id\"\n\n\nexport class BasicSync extends Dexie {\n basic_schema: any\n\n constructor(name: string, options: any) {\n super(name, options);\n\n // --- INIT SCHEMA --- // \n\n //todo: handle versions?\n\n // TODO: validate schema\n this.basic_schema = options.schema\n this.version(1).stores(this._convertSchemaToDxSchema(this.basic_schema))\n\n this.version(2).stores({})\n // this.verssion\n\n\n // create an alias for toArray\n // @ts-ignore\n this.Collection.prototype.get = this.Collection.prototype.toArray\n\n\n // --- SYNC --- // \n\n // this.syncable.on(\"statusChanged\", (status, url) => { \n // console.log(\"statusChanged\", status, url)\n // })\n\n }\n\n async connect({ access_token }: { access_token: string }) {\n const WS_URL = `wss://pds.basic.id/ws`\n\n log('Connecting to', WS_URL)\n\n await this.updateSyncNodes();\n \n log('Starting connection...')\n return this.syncable.connect(\"websocket\", WS_URL, { authToken: access_token, schema: this.basic_schema });\n }\n\n async disconnect() {\n const WS_URL = `wss://pds.basic.id/ws`\n\n return this.syncable.disconnect(WS_URL) \n }\n\n private async updateSyncNodes() {\n try {\n const syncNodes = await this.table('_syncNodes').toArray();\n const localSyncNodes = syncNodes.filter(node => node.type === 'local');\n log('Local sync nodes:', localSyncNodes);\n\n if (localSyncNodes.length > 1) {\n\n \n const largestNodeId = Math.max(...localSyncNodes.map(node => node.id));\n // Check if the largest node is already the master\n const largestNode = localSyncNodes.find(node => node.id === largestNodeId);\n if (largestNode && largestNode.isMaster === 1) {\n log('Largest node is already the master. No changes needed.');\n return; // Exit the function early as no changes are needed\n }\n\n\n log('Largest node id:', largestNodeId);\n log('HEISENBUG: More than one local sync node found.')\n\n for (const node of localSyncNodes) {\n log(`Local sync node keys:`, node.id, node.isMaster);\n await this.table('_syncNodes').update(node.id, { isMaster: node.id === largestNodeId ? 1 : 0 });\n\n log(`HEISENBUG: Setting ${node.id} to ${node.id === largestNodeId ? 'master' : '0'}`);\n }\n\n // add delay to ensure sync nodes are updated // i dont think this helps?\n await new Promise(resolve => setTimeout(resolve, 1000));\n\n if (typeof window !== 'undefined') {\n window.location.reload();\n }\n }\n\n log('Sync nodes updated');\n } catch (error) {\n console.error('Error updating _syncNodes table:', error);\n }\n }\n\n handleStatusChange(fn: any) {\n this.syncable.on(\"statusChanged\", fn)\n }\n\n\n _convertSchemaToDxSchema(schema: any) {\n const stores = Object.entries(schema.tables).map(([key, table]: any) => {\n\n const indexedFields = Object.entries(table.fields).filter(([key, field]: any) => field.indexed).map(([key, field]: any) => `,${key}`).join('')\n return {\n [key]: 'id' + indexedFields\n }\n })\n\n return Object.assign({}, ...stores)\n }\n\n debugeroo() {\n // console.log(\"debugeroo\", this.syncable)\n\n // this.syncable.list().then(x => console.log(x))\n\n // this.syncable\n return this.syncable\n }\n\n collection(name: string) {\n // TODO: check against schema\n\n return {\n\n /**\n * Returns the underlying Dexie table\n * @type {Dexie.Table}\n */\n ref: this.table(name),\n\n // --- WRITE ---- // \n add: (data: any) => {\n // log(\"Adding data to\", name, data)\n\n const valid = validateData(this.basic_schema, name, data)\n if (!valid.valid) {\n log('Invalid data', valid)\n return Promise.reject({ ... valid })\n }\n\n return this.table(name).add({\n id: uuidv7(),\n ...data\n })\n\n },\n\n put: (data: any) => {\n const valid = validateData(this.basic_schema, name, data)\n if (!valid.valid) {\n log('Invalid data', valid)\n return Promise.reject({ ... valid })\n }\n\n return this.table(name).put({\n id: uuidv7(),\n ...data\n })\n },\n\n update: (id: string, data: any) => {\n const valid = validateData(this.basic_schema, name, data, false)\n if (!valid.valid) {\n log('Invalid data', valid)\n return Promise.reject({ ... valid })\n }\n\n return this.table(name).update(id, data)\n },\n\n delete: (id: string) => {\n return this.table(name).delete(id)\n },\n\n\n // --- READ ---- // \n\n get: async (id: string) => {\n return this.table(name).get(id) \n },\n\n getAll: async () => {\n return this.table(name).toArray();\n },\n\n // --- QUERY ---- // \n // TODO: lots to do here. simplifing creating querie, filtering/ordering/limit, and execute\n\n query: () => this.table(name),\n\n filter: (fn: any) => this.table(name).filter(fn).toArray(),\n\n }\n }\n}\n\nclass QueryMethod { \n\n}\n","\"use client\"\nimport { Dexie } from \"dexie\";\nimport { log } from \"../config\";\n\nexport const syncProtocol = function () {\n log(\"Initializing syncProtocol\");\n // Constants:\n var RECONNECT_DELAY = 5000; // Reconnect delay in case of errors such as network down.\n\n Dexie.Syncable.registerSyncProtocol(\"websocket\", {\n sync: function (\n context,\n url,\n options,\n baseRevision,\n syncedRevision,\n changes,\n partial,\n applyRemoteChanges,\n onChangesAccepted,\n onSuccess,\n onError,\n ) {\n // The following vars are needed because we must know which callback to ack when server sends it's ack to us.\n var requestId = 0;\n var acceptCallbacks = {};\n\n // Connect the WebSocket to given url:\n log(\"Connecting to\", url)\n var ws = new WebSocket(url);\n\n // console.log(\"ws OPTIONS\", options);\n\n // sendChanges() method:\n function sendChanges(changes, baseRevision, partial, onChangesAccepted) {\n log(\"sendChanges\", changes.length, baseRevision);\n ++requestId;\n acceptCallbacks[requestId.toString()] = onChangesAccepted;\n\n // In this example, the server expects the following JSON format of the request:\n // {\n // type: \"changes\"\n // baseRevision: baseRevision,\n // changes: changes,\n // partial: partial,\n // requestId: id\n // }\n // To make the sample simplified, we assume the server has the exact same specification of how changes are structured.\n // In real world, you would have to pre-process the changes array to fit the server specification.\n // However, this example shows how to deal with the WebSocket to fullfill the API.\n\n ws.send(\n JSON.stringify({\n type: \"changes\",\n changes: changes,\n partial: partial,\n baseRevision: baseRevision,\n requestId: requestId,\n }),\n );\n }\n\n\n\n // When WebSocket opens, send our changes to the server.\n ws.onopen = function (event) {\n // Initiate this socket connection by sending our clientIdentity. If we dont have a clientIdentity yet,\n // server will call back with a new client identity that we should use in future WebSocket connections.\n \n // send the schema to the server\n log(\"Opening socket - sending clientIdentity\", context.clientIdentity);\n ws.send(\n JSON.stringify({\n type: \"clientIdentity\",\n clientIdentity: context.clientIdentity || null,\n authToken: options.authToken,\n schema: options.schema\n }),\n );\n\n };\n\n // If network down or other error, tell the framework to reconnect again in some time:\n ws.onerror = function (event) {\n ws.close();\n log(\"ws.onerror\", event);\n onError(event?.message, RECONNECT_DELAY);\n };\n\n // If socket is closed (network disconnected), inform framework and make it reconnect\n ws.onclose = function (event) {\n // console.log('🙅 ws.onclose', event)\n onError(\"Socket closed: \" + event.reason, RECONNECT_DELAY);\n };\n\n // isFirstRound: Will need to call onSuccess() only when we are in sync the first time.\n // onSuccess() will unblock Dexie to be used by application code.\n // If for example app code writes: db.friends.where('shoeSize').above(40).toArray(callback), the execution of that query\n // will not run until we have called onSuccess(). This is because we want application code to get results that are as\n // accurate as possible. Specifically when connected the first time and the entire DB is being synced down to the browser,\n // it is important that queries starts running first when db is in sync.\n var isFirstRound = true;\n // When message arrive from the server, deal with the message accordingly:\n ws.onmessage = function (event) {\n try {\n // Assume we have a server that should send JSON messages of the following format:\n // {\n // type: \"clientIdentity\", \"changes\", \"ack\" or \"error\"\n // clientIdentity: unique value for our database client node to persist in the context. (Only applicable if type=\"clientIdentity\")\n // message: Error message (Only applicable if type=\"error\")\n // requestId: ID of change request that is acked by the server (Only applicable if type=\"ack\" or \"error\")\n // changes: changes from server (Only applicable if type=\"changes\")\n // lastRevision: last revision of changes sent (applicable if type=\"changes\")\n // partial: true if server has additionalChanges to send. False if these changes were the last known. (applicable if type=\"changes\")\n // }\n var requestFromServer = JSON.parse(event.data);\n log(\"requestFromServer\", requestFromServer, { acceptCallback, isFirstRound });\n\n if (requestFromServer.type == \"clientIdentity\") {\n context.clientIdentity = requestFromServer.clientIdentity;\n context.save();\n\n sendChanges(changes, baseRevision, partial, onChangesAccepted);\n\n ws.send(\n JSON.stringify({\n type: \"subscribe\",\n syncedRevision: syncedRevision,\n }),\n );\n } else if (requestFromServer.type == \"changes\") {\n applyRemoteChanges(\n requestFromServer.changes,\n requestFromServer.currentRevision,\n requestFromServer.partial,\n );\n if (isFirstRound && !requestFromServer.partial) {\n // Since this is the first sync round and server sais we've got all changes - now is the time to call onsuccess()\n onSuccess({\n // Specify a react function that will react on additional client changes\n react: function (\n changes,\n baseRevision,\n partial,\n onChangesAccepted,\n ) {\n sendChanges(\n changes,\n baseRevision,\n partial,\n onChangesAccepted,\n );\n },\n // Specify a disconnect function that will close our socket so that we dont continue to monitor changes.\n disconnect: function () {\n ws.close();\n },\n });\n isFirstRound = false;\n }\n } else if (requestFromServer.type == \"ack\") {\n var requestId = requestFromServer.requestId;\n var acceptCallback = acceptCallbacks[requestId.toString()];\n acceptCallback(); // Tell framework that server has acknowledged the changes sent.\n delete acceptCallbacks[requestId.toString()];\n } else if (requestFromServer.type == \"error\") {\n var requestId = requestFromServer.requestId;\n ws.close();\n onError(requestFromServer.message, Infinity); // Don't reconnect - an error in application level means we have done something wrong.\n } else {\n log(\"unknown message\", requestFromServer);\n ws.close();\n onError(\"unknown message\", Infinity);\n }\n } catch (e) {\n ws.close();\n log(\"caught error\", e)\n onError(e, Infinity); // Something went crazy. Server sends invalid format or our code is buggy. Dont reconnect - it would continue failing.\n }\n };\n },\n });\n};\n","\nexport const SERVER_URL = \"https://api.basic.tech\"\n// export const SERVER_URL = \"http://localhost:3003\"\n\n\nexport const log = (...args: any[]) => {\n try { \n if (localStorage.getItem('basic_debug') === 'true') {\n console.log('[basic]', ...args)\n }\n } catch (e) {\n // console.log('error logging', e)\n }\n}\n\n// export const log = (message: string, ...args: any[]) => {\n// try {\n// if (process.env.NODE_ENV === 'development') {\n// const stack = new Error().stack;\n// const caller = stack?.split('\\n')[2]?.trim();\n// console.log(`[basic] ${message}`, ...args);\n// // console.log(`[stack] ${caller}`);\n// }\n// } catch (e) {\n// console.error('Error in logWithStack:', e);\n// }\n// }\n","// --- Type Definitions ---\ntype FieldType = \"string\" | \"boolean\" | \"number\" | \"json\";\n\ninterface SchemaField {\n type: FieldType;\n indexed?: boolean;\n}\n\ninterface TableSchema {\n fields: Record<string, SchemaField>;\n [key: string]: any; // Allow additional properties\n}\n\nexport interface DBSchema {\n project_id: string;\n version: number;\n tables: Record<string, TableSchema>;\n}\n\n// Direction for ordering results\ntype OrderDirection = \"asc\" | \"desc\";\n\n// Filter operators\ntype FilterOperator = \"eq\" | \"neq\" | \"gt\" | \"gte\" | \"lt\" | \"lte\" | \"like\" | \"ilike\" | \"in\" | \"is\";\n\n// Filter value type\ntype FilterValue = string | number | boolean | null | string[];\n\n// Single operator filter condition\ninterface OperatorFilter {\n eq?: FilterValue;\n neq?: FilterValue;\n gt?: number | string;\n gte?: number | string;\n lt?: number | string;\n lte?: number | string;\n like?: string;\n ilike?: string;\n in?: string[] | number[];\n is?: boolean | null;\n not?: OperatorFilter; // For negation\n}\n\n// Combined type for possible filter inputs\ntype FilterCondition = \n | FilterValue // Simple value = equality filter\n | OperatorFilter; // Complex operator-based filter\n\n// Query options for table operations\ninterface QueryOptions {\n id?: string;\n order?: string;\n limit?: number;\n offset?: number;\n filters?: Record<string, FilterCondition>;\n}\n\ntype FieldTypeToTS<T extends FieldType> =\n T extends \"string\" ? string :\n T extends \"boolean\" ? boolean :\n T extends \"number\" ? number :\n T extends \"json\" ? Record<string, any> : never;\n\nexport type TableData<T extends TableSchema> = {\n id: string;\n created_at: string;\n} & {\n [K in keyof T[\"fields\"]]: T[\"fields\"][K] extends { type: FieldType } ? FieldTypeToTS<T[\"fields\"][K][\"type\"]> : never;\n};\n\n// --- Schema Helper ---\n// Constraint for the input schema to ensure it has the basic shape.\n// T_SpecificSchema itself will be more specific due to 'as const'.\ntype ValidSchemaInput = {\n project_id: string;\n version: number;\n tables: Record<string, TableSchema>; // This means T_SpecificSchema.tables must be assignable to this\n};\n\nexport function createSchema<T_SpecificSchema extends ValidSchemaInput>(\n schema: T_SpecificSchema\n): T_SpecificSchema {\n // Detailed runtime validation\n if (typeof schema.project_id !== 'string' || schema.project_id.trim() === '') {\n throw new Error('Invalid schema: project_id must be a non-empty string.');\n }\n if (typeof schema.version !== 'number') {\n throw new Error('Invalid schema: version must be a number.');\n }\n if (typeof schema.tables !== 'object' || schema.tables === null || Object.keys(schema.tables).length === 0) {\n throw new Error('Invalid schema: tables must be a non-empty object.');\n }\n\n for (const tableName in schema.tables) {\n const table = schema.tables[tableName];\n if (typeof table !== 'object' || table === null) {\n throw new Error(`Invalid schema: table '${tableName}' is not an object.`);\n }\n if (typeof table.fields !== 'object' || table.fields === null || Object.keys(table.fields).length === 0) {\n throw new Error(`Invalid schema for table '${tableName}': fields must be a non-empty object.`);\n }\n for (const fieldName in table.fields) {\n const field = table.fields[fieldName];\n if (typeof field !== 'object' || field === null) {\n throw new Error(`Invalid schema for table '${tableName}', field '${fieldName}': is not an object.`);\n }\n const validTypes: FieldType[] = [\"string\", \"boolean\", \"number\", \"json\"];\n if (typeof field.type !== 'string' || !validTypes.includes(field.type as FieldType)) {\n throw new Error(`Invalid schema for table '${tableName}', field '${fieldName}': type must be one of \"string\", \"boolean\", \"number\", \"json\".`);\n }\n if (field.indexed !== undefined && typeof field.indexed !== 'boolean') {\n throw new Error(`Invalid schema for table '${tableName}', field '${fieldName}': optional 'indexed' property must be a boolean.`);\n }\n }\n }\n return schema;\n}\n\ninterface SDKConfig<S extends DBSchema> {\n project_id: string;\n token?: string;\n getToken?: () => Promise<string>;\n baseUrl?: string;\n schema: S;\n}\n\n// Add custom error class at the top of the file\nclass DBError extends Error {\n constructor(\n message: string,\n public status?: number,\n public response?: any,\n public originalError?: Error\n ) {\n super(message);\n this.name = 'DBError';\n }\n}\n\n// --- Query Builder ---\nclass QueryBuilder<T> {\n private params: QueryOptions = {};\n \n constructor(\n private tableClient: TableClient<T>,\n private tableSchema?: TableSchema\n ) {}\n \n // Reserved fields that are always allowed\n private reservedFields = [\"created_at\", \"updated_at\", \"id\"];\n \n // Validate field existence in schema\n private validateField(field: string): void {\n if (this.tableSchema && !this.reservedFields.includes(field)) {\n if (!this.tableSchema.fields || !(field in this.tableSchema.fields)) {\n throw new Error(`Invalid field: \"${field}\". Field does not exist in table schema.`);\n }\n }\n }\n \n // Validate operator based on field type\n private validateOperator(field: string, operator: FilterOperator, value: FilterValue): void {\n if (!this.tableSchema || this.reservedFields.includes(field)) {\n return; // Skip validation for reserved fields\n }\n \n const fieldInfo = this.tableSchema.fields[field];\n if (!fieldInfo) return; // Already validated by validateField\n \n // Type checking based on operator and field type\n switch (operator) {\n case 'gt':\n case 'gte':\n case 'lt':\n case 'lte':\n if (fieldInfo.type !== 'number' && fieldInfo.type !== 'string') {\n throw new Error(`Operator \"${operator}\" can only be used with number or string fields. Field \"${field}\" is type \"${fieldInfo.type}\".`);\n }\n break;\n case 'like':\n case 'ilike':\n if (fieldInfo.type !== 'string') {\n throw new Error(`Operator \"${operator}\" can only be used with string fields. Field \"${field}\" is type \"${fieldInfo.type}\".`);\n }\n if (typeof value !== 'string') {\n throw new Error(`Operator \"${operator}\" requires a string value. Received: ${typeof value}`);\n }\n break;\n case 'in':\n if (!Array.isArray(value)) {\n throw new Error(`Operator \"in\" requires an array value. Received: ${typeof value}`);\n }\n break;\n case 'is':\n if (value !== null && typeof value !== 'boolean') {\n throw new Error(`Operator \"is\" requires null or boolean. Received: ${typeof value}`);\n }\n break;\n }\n }\n \n // Add ordering to query with schema validation\n order(field: string, direction: OrderDirection = \"asc\"): QueryBuilder<T> {\n // Validate field\n this.validateField(field);\n \n this.params.order = `${field}.${direction}`;\n return this;\n }\n \n // Add filtering to query\n filter(conditions: Record<string, FilterCondition>): QueryBuilder<T> {\n if (!this.params.filters) {\n this.params.filters = {};\n }\n \n // Process each filter condition\n for (const [field, condition] of Object.entries(conditions)) {\n // Validate field\n this.validateField(field);\n \n // Process based on condition type\n if (condition === null || typeof condition !== 'object') {\n // Simple equality filter (or is.null/is.true/is.false for special values)\n this.params.filters[field] = condition;\n } else {\n // Complex filter with operators\n this.params.filters[field] = condition;\n }\n }\n \n return this;\n }\n \n // Add limit to query\n limit(count: number): QueryBuilder<T> {\n this.params.limit = count;\n return this;\n }\n \n // Add offset to query for pagination\n offset(count: number): QueryBuilder<T> {\n this.params.offset = count;\n return this;\n }\n \n // Auto-execute when awaited\n then<TResult1 = T[], TResult2 = never>(\n onfulfilled?: ((value: T[]) => TResult1 | PromiseLike<TResult1>) | null, \n onrejected?: ((reason: any) => TResult2 | PromiseLike<TResult2>) | null\n ): Promise<TResult1 | TResult2> {\n return this.tableClient.executeQuery(this.params).then(onfulfilled, onrejected);\n }\n \n // Auto-execute when awaited with catch\n catch<TResult = never>(\n onrejected?: ((reason: any) => TResult | PromiseLike<TResult>) | null\n ): Promise<T[] | TResult> {\n return this.tableClient.executeQuery(this.params).catch(onrejected);\n }\n \n // Auto-execute when awaited with finally\n finally(\n onfinally?: (() => void) | null\n ): Promise<T[]> {\n return this.tableClient.executeQuery(this.params).finally(onfinally);\n }\n}\n\n// --- Table Client ---\nclass TableClient<T> {\n private tableSchema?: TableSchema;\n \n constructor(\n private baseUrl: string,\n private projectId: string,\n private token: string,\n private table: string,\n private getToken: () => Promise<string>,\n private schema?: DBSchema\n ) {\n // Store the table schema for validation\n if (schema && schema.tables && schema.tables[table]) {\n this.tableSchema = schema.tables[table];\n }\n }\n\n private async headers() {\n const token = await this.getToken();\n return {\n Authorization: `Bearer ${token}`,\n \"Content-Type\": \"application/json\"\n };\n }\n\n private async handleRequest<T>(request: Promise<Response>): Promise<T> {\n try {\n // console.log('Making request to:', this.baseUrl);\n // console.log('Headers:', await this.headers());\n // console.log('Project ID:', this.projectId);\n // console.log('Table:', this.table);\n \n const res = await request;\n \n // console.log(\"Response status:\", res.status);\n // console.log(\"Response headers:\", res.headers);\n \n // First check if the response is OK\n if (!res.ok) {\n let errorMessage = `Request failed with status ${res.status}`;\n let errorData;\n \n try {\n const json = await res.json();\n errorData = json;\n // Format the error message more clearly\n if (json.error || json.message) {\n const errorDetails = typeof json.error === 'object' ? JSON.stringify(json.error) : json.error;\n const messageDetails = typeof json.message === 'object' ? JSON.stringify(json.message) : json.message;\n errorMessage = `${res.status} ${res.statusText}: ${messageDetails || errorDetails || 'Unknown error'}`;\n }\n } catch (e) {\n console.log(\"Failed to parse error response:\", e);\n // If we can't parse JSON, use the status text\n errorMessage = `${res.status} ${res.statusText}`;\n }\n\n throw new DBError(\n errorMessage,\n res.status,\n errorData\n );\n }\n\n // If response is OK, parse and return the data\n const json = await res.json();\n return json.data;\n } catch (error) {\n console.log(\"Caught error:\", error);\n if (error instanceof Error) {\n console.log(\"Error type:\", error.constructor.name);\n console.log(\"Error stack:\", error.stack);\n }\n \n if (error instanceof DBError) {\n throw error;\n }\n \n if (error instanceof TypeError && error.message === 'Network request failed') {\n throw new DBError(\n 'Network request failed. Please check your internet connection and try again.',\n undefined,\n undefined,\n error\n );\n }\n\n throw new DBError(\n `Unexpected error: ${error instanceof Error ? error.message : 'Unknown error'}`,\n undefined,\n undefined,\n error instanceof Error ? error : undefined\n );\n }\n }\n\n // Build query string from query options\n private buildQueryParams(query?: QueryOptions): string {\n if (!query) return \"\";\n \n const params: string[] = [];\n \n // Add id filter if provided\n if (query.id) {\n params.push(`id=${query.id}`);\n }\n \n // Add filter conditions\n if (query.filters) {\n for (const [field, condition] of Object.entries(query.filters)) {\n this.addFilterParam(params, field, condition);\n }\n }\n \n // Add ordering if provided\n if (query.order) {\n params.push(`order=${query.order}`);\n }\n \n // Add limit if provided\n if (query.limit !== undefined && query.limit >= 0) {\n params.push(`limit=${query.limit}`);\n }\n \n // Add offset if provided\n if (query.offset !== undefined && query.offset >= 0) {\n params.push(`offset=${query.offset}`);\n }\n \n return params.length > 0 ? `?${params.join('&')}` : \"\";\n }\n \n // Helper method to build filter parameters\n private addFilterParam(params: string[], field: string, condition: FilterCondition, negate: boolean = false): void {\n // Handle simple values (direct equality)\n if (condition === null || typeof condition !== 'object') {\n if (condition === null) {\n params.push(`${field}=${negate ? 'not.' : ''}is.null`);\n } else if (typeof condition === 'boolean') {\n params.push(`${field}=${negate ? 'not.' : ''}is.${condition}`);\n } else if (typeof condition === 'number') {\n params.push(`${field}=${negate ? 'not.' : ''}eq.${condition}`);\n } else {\n // String values\n params.push(`${field}=${negate ? 'not.' : ''}eq.${encodeURIComponent(String(condition))}`);\n }\n return;\n }\n \n // Handle OperatorFilter object\n const operatorObj = condition as OperatorFilter;\n \n // Handle negation wrapper\n if (operatorObj.not) {\n this.addFilterParam(params, field, operatorObj.not, true);\n return;\n }\n \n // Process each operator\n for (const [op, value] of Object.entries(operatorObj)) {\n // Skip 'not' since we handled it above\n if (op === 'not') continue;\n \n const operator = op as FilterOperator;\n \n if (value === null) {\n params.push(`${field}=${negate ? 'not.' : ''}is.null`);\n } else if (operator === 'in' && Array.isArray(value)) {\n params.push(`${field}=${negate ? 'not.' : ''}in.${value.join(',')}`);\n } else if (operator === 'is') {\n if (typeof value === 'boolean') {\n params.push(`${field}=${negate ? 'not.' : ''}is.${value}`);\n } else {\n params.push(`${field}=${negate ? 'not.' : ''}is.null`);\n }\n } else {\n // All other operators\n const paramValue = typeof value === 'string' \n ? encodeURIComponent(value) \n : String(value);\n params.push(`${field}=${negate ? 'not.' : ''}${operator}.${paramValue}`);\n }\n }\n }\n\n // Internal method to execute a query with options\n async executeQuery(options?: QueryOptions): Promise<T[]> {\n const params = this.buildQueryParams(options);\n const headers = await this.headers();\n // console.log(headers);\n return this.handleRequest(\n fetch(`${this.baseUrl}/account/${this.projectId}/db/${this.table}${params}`, {\n headers\n })\n );\n }\n\n // Public method to start building a query\n getAll(): QueryBuilder<T> {\n return new QueryBuilder<T>(this, this.tableSchema);\n }\n\n // Get a specific item by ID\n async get(id: string): Promise<T> {\n const headers = await this.headers();\n return this.handleRequest(\n fetch(`${this.baseUrl}/account/${this.projectId}/db/${this.table}/${id}`, {\n headers\n })\n );\n }\n\n async create(value: Partial<T>): Promise<T> {\n const headers = await this.headers();\n return this.handleRequest(\n fetch(`${this.baseUrl}/account/${this.projectId}/db/${this.table}`, {\n method: \"POST\",\n headers,\n body: JSON.stringify({ value })\n })\n );\n }\n\n async update(id: string, value: Partial<T>): Promise<T> {\n const headers = await this.headers();\n return this.handleRequest(\n fetch(`${this.baseUrl}/account/${this.projectId}/db/${this.table}/${id}`, {\n method: \"PATCH\",\n headers,\n body: JSON.stringify({ value })\n })\n );\n }\n\n async replace(id: string, value: Partial<T>): Promise<T> {\n const headers = await this.headers();\n return this.handleRequest(\n fetch(`${this.baseUrl}/account/${this.projectId}/db/${this.table}/${id}`, {\n method: \"PUT\",\n headers,\n body: JSON.stringify({ value })\n })\n );\n }\n\n async delete(id: string): Promise<T> {\n const token = await this.getToken();\n const headers = {\n Authorization: `Bearer ${token}`\n };\n return this.handleRequest(\n fetch(`${this.baseUrl}/account/${this.projectId}/db/${this.table}/${id}`, {\n method: \"DELETE\",\n headers\n })\n );\n }\n}\n\n// --- Main SDK ---\nexport class BasicDBSDK<S extends DBSchema> {\n private projectId: string;\n private getToken: () => Promise<string>;\n private baseUrl: string;\n private schema: S;\n private tableNames: (keyof S[\"tables\"] & string)[];\n\n constructor(config: SDKConfig<S>) {\n this.projectId = config.project_id;\n \n // Handle either static token or token getter function\n if (config.getToken) {\n this.getToken = config.getToken;\n } else if (config.token) {\n this.getToken = async () => config.token!;\n } else {\n throw new Error('Either token or getToken must be provided');\n }\n \n this.baseUrl = config.baseUrl || \"https://api.basic.tech\";\n this.schema = config.schema;\n this.tableNames = Object.keys(this.schema.tables) as (keyof S[\"tables\"] & string)[];\n }\n\n // Primary method - table access\n table<K extends keyof S[\"tables\"] & string>(\n name: K\n ): TableClient<TableData<S[\"tables\"][K]>> {\n // Validate table name at runtime\n if (!this.tableNames.includes(name)) {\n throw new Error(`Table '${name}' not found in schema. Available tables: ${this.tableNames.join(', ')}`);\n }\n \n // Create a wrapped client that will get a fresh token for each request\n return new TableClient(\n this.baseUrl,\n this.projectId,\n \"\", // Empty placeholder, will be replaced in headers() method\n name,\n this.getToken,\n this.schema // Pass the entire schema to the TableClient\n );\n }\n\n\n get tables(): {\n [K in keyof S[\"tables\"]]: TableData<S[\"tables\"][K]>;\n } {\n return {} as any;\n }\n\n fields<K extends keyof S[\"tables\"] & string>(\n table: K\n ): (keyof S[\"tables\"][K][\"fields\"] & string)[] {\n const tableSchema = this.schema.tables[table];\n if (!tableSchema) {\n throw new Error(`Table '${table}' not found in schema`);\n }\n return Object.keys(tableSchema.fields) as (keyof S[\"tables\"][K][\"fields\"] & string)[];\n }\n}\n","{\n \"name\": \"@basictech/react\",\n \"version\": \"0.7.0-beta.1\",\n \"description\": \"\",\n \"main\": \"dist/index.js\",\n \"module\": \"dist/index.mjs\",\n \"types\": \"dist/index.d.ts\",\n \"private\": false,\n \"publishConfig\": {\n \"access\": \"public\"\n },\n \"scripts\": {\n \"build\": \"tsup\",\n \"dev\": \"tsup --watch\",\n \"test\": \"echo \\\"Error: no test specified\\\" && exit 1\"\n },\n \"author\": \"\",\n \"license\": \"ISC\",\n \"dependencies\": {\n \"ajv\": \"^8.17.1\",\n \"dexie\": \"^4.0.8\",\n \"dexie-observable\": \"^4.0.1-beta.13\",\n \"dexie-react-hooks\": \"^1.1.7\",\n \"dexie-syncable\": \"^4.0.1-beta.13\",\n \"jwt-decode\": \"^4.0.0\",\n \"uuid\": \"^10.0.0\",\n \"@basictech/schema\": \"0.6.0\"\n },\n \"devDependencies\": {\n \"@repo/typescript-config\": \"*\",\n \"tsup\": \"^7.2.0\",\n \"typescript\": \"^5.0.0\"\n },\n \"peerDependencies\": {\n \"react\": \"^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0\"\n }\n}\n","import { BasicStorage } from '../utils/storage'\n\nexport interface VersionInfo {\n version: string\n lastUpdated: number\n}\n\nexport interface Migration {\n fromVersion: string\n toVersion: string\n migrate: (storage: BasicStorage) => Promise<void>\n}\n\nexport class VersionUpdater {\n private storage: BasicStorage\n private currentVersion: string\n private migrations: Migration[]\n private versionKey = 'basic_app_version'\n\n constructor(storage: BasicStorage, currentVersion: string, migrations: Migration[] = []) {\n this.storage = storage\n this.currentVersion = currentVersion\n this.migrations = migrations.sort((a, b) => this.compareVersions(a.fromVersion, b.fromVersion))\n }\n\n /**\n * Check current stored version and run migrations if needed\n * Only compares major.minor versions, ignoring beta/prerelease parts\n * Example: \"0.7.0-beta.1\" and \"0.7.0\" are treated as the same version\n */\n async checkAndUpdate(): Promise<{ updated: boolean; fromVersion?: string; toVersion: string }> {\n const storedVersion = await this.getStoredVersion()\n \n if (!storedVersion) {\n // First time setup\n await this.setStoredVersion(this.currentVersion)\n return { updated: false, toVersion: this.currentVersion }\n }\n\n if (storedVersion === this.currentVersion) {\n return { updated: false, toVersion: this.currentVersion }\n }\n\n // Need to run migrations\n const migrationsToRun = this.getMigrationsToRun(storedVersion, this.currentVersion)\n \n if (migrationsToRun.length === 0) {\n // No migrations needed, just update version\n await this.setStoredVersion(this.currentVersion)\n return { updated: true, fromVersion: storedVersion, toVersion: this.currentVersion }\n }\n\n // Run migrations\n for (const migration of migrationsToRun) {\n try {\n console.log(`Running migration from ${migration.fromVersion} to ${migration.toVersion}`)\n await migration.migrate(this.storage)\n } catch (error) {\n console.error(`Migration failed from ${migration.fromVersion} to ${migration.toVersion}:`, error)\n throw new Error(`Migration failed: ${error}`)\n }\n }\n\n // Update to current version\n await this.setStoredVersion(this.currentVersion)\n return { updated: true, fromVersion: storedVersion, toVersion: this.currentVersion }\n }\n\n private async getStoredVersion(): Promise<string | null> {\n try {\n const versionData = await this.storage.get(this.versionKey)\n if (!versionData) return null\n \n const versionInfo: VersionInfo = JSON.parse(versionData)\n return versionInfo.version\n } catch (error) {\n console.warn('Failed to get stored version:', error)\n return null\n }\n }\n\n private async setStoredVersion(version: string): Promise<void> {\n const versionInfo: VersionInfo = {\n version,\n lastUpdated: Date.now()\n }\n await this.storage.set(this.versionKey, JSON.stringify(versionInfo))\n }\n\n private getMigrationsToRun(fromVersion: string, toVersion: string): Migration[] {\n return this.migrations.filter(migration => {\n // Migration should run if we're crossing the version boundary\n // i.e., stored version is less than migration.toVersion AND current version is >= migration.toVersion\n const storedLessThanMigrationTo = this.compareVersions(fromVersion, migration.toVersion) < 0\n const currentGreaterThanOrEqualMigrationTo = this.compareVersions(toVersion, migration.toVersion) >= 0\n \n console.log(`Checking migration ${migration.fromVersion} → ${migration.toVersion}:`)\n console.log(` stored ${fromVersion} < migration.to ${migration.toVersion}: ${storedLessThanMigrationTo}`)\n console.log(` current ${toVersion} >= migration.to ${migration.toVersion}: ${currentGreaterThanOrEqualMigrationTo}`)\n \n const shouldRun = storedLessThanMigrationTo && currentGreaterThanOrEqualMigrationTo\n console.log(` Should run: ${shouldRun}`)\n \n return shouldRun\n })\n }\n\n /**\n * Simple semantic version comparison (major.minor only, ignoring beta/prerelease)\n * Returns: -1 if a < b, 0 if a === b, 1 if a > b\n */\n private compareVersions(a: string, b: string): number {\n // Extract major.minor from version strings, ignoring beta/prerelease parts\n const aMajorMinor = this.extractMajorMinor(a)\n const bMajorMinor = this.extractMajorMinor(b)\n \n // Compare major version first\n if (aMajorMinor.major !== bMajorMinor.major) {\n return aMajorMinor.major - bMajorMinor.major\n }\n \n // Then compare minor version\n return aMajorMinor.minor - bMajorMinor.minor\n }\n\n /**\n * Extract major.minor from version string, ignoring beta/prerelease\n * Examples: \"0.7.0-beta.1\" -> {major: 0, minor: 7}\n * \"1.2.3\" -> {major: 1, minor: 2}\n */\n private extractMajorMinor(version: string): { major: number, minor: number } {\n // Remove beta/prerelease parts and split by dots\n const cleanVersion = version.split('-')[0]?.split('+')[0] || version\n const parts = cleanVersion.split('.').map(Number)\n \n return {\n major: parts[0] || 0,\n minor: parts[1] || 0\n }\n }\n\n /**\n * Add a migration to the updater\n */\n addMigration(migration: Migration): void {\n this.migrations.push(migration)\n this.migrations.sort((a, b) => this.compareVersions(a.fromVersion, b.fromVersion))\n }\n}\n\n/**\n * Create a simple version updater instance\n */\nexport function createVersionUpdater(\n storage: BasicStorage, \n currentVersion: string, \n migrations: Migration[] = []\n): VersionUpdater {\n return new VersionUpdater(storage, currentVersion, migrations)\n}\n","import { BasicStorage } from '../utils/storage'\nimport { Migration } from './versionUpdater'\n\n\nexport const addMigrationTimestamp: Migration = {\n fromVersion: '0.6.0', \n toVersion: '0.7.0',\n async migrate(storage: BasicStorage) {\n console.log('Running test migration')\n storage.set('test_migration', 'true')\n }\n}\n\n\n/**\n * Get all available migrations\n */\nexport function getMigrations(): Migration[] {\n return [\n addMigrationTimestamp\n ]\n}\n","// Storage utilities for Basic React package\nexport interface BasicStorage {\n get(key: string): Promise<string | null>\n set(key: string, value: string): Promise<void>\n remove(key: string): Promise<void>\n}\n\nexport class LocalStorageAdapter implements BasicStorage {\n async get(key: string): Promise<string | null> {\n return localStorage.getItem(key)\n }\n \n async set(key: string, value: string): Promise<void> {\n localStorage.setItem(key, value)\n }\n \n async remove(key: string): Promise<void> {\n localStorage.removeItem(key)\n }\n}\n\nexport const STORAGE_KEYS = {\n REFRESH_TOKEN: 'basic_refresh_token',\n USER_INFO: 'basic_user_info',\n AUTH_STATE: 'basic_auth_state',\n DEBUG: 'basic_debug'\n} as const\n\nexport function getCookie(name: string): string {\n let cookieValue = '';\n if (document.cookie && document.cookie !== '') {\n const cookies = document.cookie.split(';');\n for (let i = 0; i < cookies.length; i++) {\n const cookie = cookies[i]?.trim();\n if (cookie && cookie.substring(0, name.length + 1) === (name + '=')) {\n cookieValue = decodeURIComponent(cookie.substring(name.length + 1));\n break;\n }\n }\n }\n return cookieValue;\n}\n\nexport function setCookie(name: string, value: string, options?: { secure?: boolean, sameSite?: string, httpOnly?: boolean }): void {\n const opts = {\n secure: true,\n sameSite: 'Strict',\n httpOnly: false,\n ...options\n };\n \n let cookieString = `${name}=${value}`;\n if (opts.secure) cookieString += '; Secure';\n if (opts.sameSite) cookieString += `; SameSite=${opts.sameSite}`;\n if (opts.httpOnly) cookieString += '; HttpOnly';\n \n document.cookie = cookieString;\n}\n\nexport function clearCookie(name: string): void {\n document.cookie = `${name}=; Secure; SameSite=Strict`;\n}\n","// Network utilities for Basic React package\nimport { log } from '../config'\nimport { version as currentVersion } from '../../package.json'\n\nexport function isDevelopment(debug?: boolean): boolean {\n return (\n window.location.hostname === 'localhost' ||\n window.location.hostname === '127.0.0.1' ||\n window.location.hostname.includes('localhost') ||\n window.location.hostname.includes('127.0.0.1') ||\n window.location.hostname.includes('.local') ||\n process.env.NODE_ENV === 'development' ||\n debug === true\n )\n}\n\nexport async function checkForNewVersion(): Promise<{ \n hasNewVersion: boolean, \n latestVersion: string | null, \n currentVersion: string | null \n}> {\n try {\n const isBeta = currentVersion.includes('beta')\n\n const response = await fetch(`https://registry.npmjs.org/@basictech/react/${isBeta ? 'beta' : 'latest'}`);\n if (!response.ok) {\n throw new Error('Failed to fetch version from npm');\n }\n\n const data = await response.json();\n const latestVersion = data.version;\n\n if (latestVersion !== currentVersion) {\n console.warn('[basic] New version available:', latestVersion, `\\nrun \"npm install @basictech/react@${latestVersion}\" to update`);\n }\n if (isBeta) {\n log('thank you for being on basictech/react beta :)')\n }\n \n return {\n hasNewVersion: currentVersion !== latestVersion,\n latestVersion,\n currentVersion\n };\n } catch (error) {\n log('Error checking for new version:', error);\n return {\n hasNewVersion: false,\n latestVersion: null, \n currentVersion: null\n };\n }\n}\n\nexport function cleanOAuthParamsFromUrl(): void {\n if (window.location.search.includes('code') || window.location.search.includes('state')) {\n const url = new URL(window.location.href)\n url.searchParams.delete('code')\n url.searchParams.delete('state')\n window.history.pushState({}, document.title, url.pathname + url.search)\n log('Cleaned OAuth parameters from URL')\n }\n}\n\nexport function getSyncStatus(statusCode: number): string {\n switch (statusCode) {\n case -1:\n return \"ERROR\";\n case 0:\n return \"OFFLINE\";\n case 1:\n return \"CONNECTING\";\n case 2:\n return \"ONLINE\";\n case 3:\n return \"SYNCING\";\n case 4:\n return \"ERROR_WILL_RETRY\";\n default:\n return \"UNKNOWN\";\n }\n}\n","// Schema utilities for Basic React package\nimport { validateSchema, compareSchemas } from '@basictech/schema'\nimport { log } from '../config'\n\nexport async function getSchemaStatus(schema: any): Promise<{\n valid: boolean,\n status: string,\n latest: any\n}> {\n const projectId = schema.project_id\n const valid = validateSchema(schema)\n\n if (!valid.valid) {\n console.warn('BasicDB Error: your local schema is invalid. Please fix errors and try again - sync is disabled')\n return { \n valid: false, \n status: 'invalid',\n latest: null\n }\n }\n\n const latestSchema = await fetch(`https://api.basic.tech/project/${projectId}/schema`)\n .then(res => res.json())\n .then(data => data.data[0].schema)\n .catch(err => {\n return { \n valid: false, \n status: 'error',\n latest: null\n }\n })\n\n console.log('latestSchema', latestSchema)\n\n if (!latestSchema.version) {\n return { \n valid: false, \n status: 'error',\n latest: null\n }\n }\n\n if (latestSchema.version > schema.version) {\n // error_code: schema_behind\n console.warn('BasicDB Error: your local schema version is behind the latest. Found version:', schema.version, 'but expected', latestSchema.version, \" - sync is disabled\")\n return { \n valid: false, \n status: 'behind', \n latest: latestSchema\n }\n } else if (latestSchema.version < schema.version) {\n // error_code: schema_ahead\n console.warn('BasicDB Error: your local schema version is ahead of the latest. Found version:', schema.version, 'but expected', latestSchema.version, \" - sync is disabled\")\n return { \n valid: false, \n status: 'ahead', \n latest: latestSchema\n }\n } else if (latestSchema.version === schema.version) {\n const changes = compareSchemas(schema, latestSchema)\n if (changes.valid) {\n return { \n valid: true,\n status: 'current',\n latest: latestSchema\n }\n } else {\n // error_code: schema_conflict\n console.warn('BasicDB Error: your local schema is conflicting with the latest. Your version:', schema.version, 'does not match origin version', latestSchema.version, \" - sync is disabled\")\n return { \n valid: false, \n status: 'conflict',\n latest: latestSchema\n }\n }\n } else { \n return { \n valid: false, \n status: 'error',\n latest: null\n }\n }\n}\n\nexport async function validateAndCheckSchema(schema: any): Promise<{\n isValid: boolean,\n schemaStatus: { valid: boolean, status?: string, latest?: any },\n errors?: any[]\n}> {\n const valid = validateSchema(schema)\n if (!valid.valid) {\n log('Basic Schema is invalid!', valid.errors)\n console.group('Schema Errors')\n let errorMessage = ''\n valid.errors.forEach((error, index) => {\n log(`${index + 1}:`, error.message, ` - at ${error.instancePath}`)\n errorMessage += `${index + 1}: ${error.message} - at ${error.instancePath}\\n`\n })\n console.groupEnd()\n \n return {\n isValid: false,\n schemaStatus: { valid: false },\n errors: valid.errors\n }\n }\n\n let schemaStatus = { valid: false }\n if (schema.version !== 0) {\n schemaStatus = await getSchemaStatus(schema)\n log('schemaStatus', schemaStatus)\n } else { \n log(\"schema not published - at version 0\")\n }\n\n return {\n isValid: true,\n schemaStatus\n }\n}\n","import { useState } from \"react\";\nimport { useBasic, BasicProvider, BasicStorage, LocalStorageAdapter } from \"./AuthContext\";\nimport { useLiveQuery as useQuery } from \"dexie-react-hooks\";\nimport { BasicDBSDK } from \"./db_ts\";\n// import { createVersionUpdater, VersionUpdater, Migration } from \"./versionUpdater\";\n\n\nexport {\n useBasic, BasicProvider, useQuery, BasicDBSDK\n}\n\n// export type {\n// VersionUpdater, Migration\n// }\n"],"mappings":"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","names":["Dexie","changes","baseRevision","partial","onChangesAccepted","requestId","Dexie","key","json","version","validateSchema","validateSchema","error","user","token"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@basictech/react",
3
- "version": "0.7.0-beta.0",
3
+ "version": "0.7.0-beta.2",
4
4
  "description": "",
5
5
  "main": "dist/index.js",
6
6
  "module": "dist/index.mjs",
@@ -24,7 +24,7 @@
24
24
  "dexie-syncable": "^4.0.1-beta.13",
25
25
  "jwt-decode": "^4.0.0",
26
26
  "uuid": "^10.0.0",
27
- "@basictech/schema": "0.7.0-beta.0"
27
+ "@basictech/schema": "0.6.0"
28
28
  },
29
29
  "devDependencies": {
30
30
  "@repo/typescript-config": "*",
package/readme.md CHANGED
@@ -60,7 +60,7 @@ In your components, you can use the `useBasic` hook to access authentication and
60
60
  import { useBasic } from '@basictech/react';
61
61
 
62
62
  function MyComponent() {
63
- const { user, isSignedIn, signin, signout, db } = useBasic();
63
+ const { user, isSignedIn, signin, signout, signinWithCode, db } = useBasic();
64
64
 
65
65
  if (!isSignedIn) {
66
66
  return <button onClick={signin}>Sign In</button>;
@@ -75,6 +75,117 @@ function MyComponent() {
75
75
  }
76
76
  ```
77
77
 
78
+ ### 3. Manual OAuth Code Handling
79
+
80
+ For custom OAuth flows (mobile apps, server-side redirects, etc.), you can manually handle authorization codes:
81
+
82
+ ```typescript
83
+ import { useBasic } from '@basictech/react';
84
+
85
+ function CustomAuthComponent() {
86
+ const { signinWithCode } = useBasic();
87
+
88
+ const handleOAuthCode = async (code: string, state?: string) => {
89
+ const result = await signinWithCode(code, state);
90
+
91
+ if (result.success) {
92
+ console.log('Successfully authenticated!');
93
+ // Redirect to authenticated area
94
+ } else {
95
+ console.error('Authentication failed:', result.error);
96
+ // Handle error
97
+ }
98
+ };
99
+
100
+ // Example: Handle OAuth code from URL parameters
101
+ useEffect(() => {
102
+ const urlParams = new URLSearchParams(window.location.search);
103
+ const code = urlParams.get('code');
104
+ const state = urlParams.get('state');
105
+
106
+ if (code) {
107
+ handleOAuthCode(code, state || undefined);
108
+ }
109
+ }, []);
110
+
111
+ return <div>Processing authentication...</div>;
112
+ }
113
+ ```
114
+
115
+ #### Mobile App Integration Example:
116
+
117
+ ```typescript
118
+ // React Native or mobile app deep link handling
119
+ const handleDeepLink = (url: string) => {
120
+ const urlParams = new URLSearchParams(url.split('?')[1]);
121
+ const code = urlParams.get('code');
122
+ const state = urlParams.get('state');
123
+
124
+ if (code) {
125
+ signinWithCode(code, state || undefined);
126
+ }
127
+ };
128
+ ```
129
+
130
+ #### Server-Side OAuth Example:
131
+
132
+ ```typescript
133
+ // When OAuth happens server-side and you receive the code
134
+ const handleServerOAuth = async (serverCode: string) => {
135
+ const result = await signinWithCode(serverCode);
136
+
137
+ if (result.success) {
138
+ // Redirect to authenticated area
139
+ window.location.href = '/dashboard';
140
+ } else {
141
+ // Show error message
142
+ alert(`Authentication failed: ${result.error}`);
143
+ }
144
+ };
145
+ ```
146
+
147
+ ### 4. Custom OAuth Redirect URIs
148
+
149
+ You can specify custom redirect URIs for OAuth flows:
150
+
151
+ ```typescript
152
+ import { useBasic } from '@basictech/react';
153
+
154
+ function CustomAuthComponent() {
155
+ const { getSignInLink } = useBasic();
156
+
157
+ const handleCustomRedirect = () => {
158
+ // Use a custom redirect URI
159
+ const signInUrl = getSignInLink('https://yourapp.com/auth/callback');
160
+ window.location.href = signInUrl;
161
+ };
162
+
163
+ const handleDefaultRedirect = () => {
164
+ // Use default redirect (current page URL)
165
+ const signInUrl = getSignInLink();
166
+ window.location.href = signInUrl;
167
+ };
168
+
169
+ return (
170
+ <div>
171
+ <button onClick={handleCustomRedirect}>
172
+ Sign In (Custom Redirect)
173
+ </button>
174
+ <button onClick={handleDefaultRedirect}>
175
+ Sign In (Default Redirect)
176
+ </button>
177
+ </div>
178
+ );
179
+ }
180
+ ```
181
+
182
+ #### Use Cases for Custom Redirect URIs:
183
+
184
+ - **Mobile Apps**: Redirect to app-specific URLs
185
+ - **Multi-Domain**: Redirect to different domains based on context
186
+ - **Testing**: Use test-specific callback URLs
187
+ - **Subdomains**: Redirect to specific subdomains
188
+
78
189
  ## API Reference
79
190
 
80
191
 
@@ -128,8 +239,23 @@ Returns an object with the following properties and methods:
128
239
  - `isSignedIn`: Boolean indicating if the user is signed in
129
240
  - `signin()`: Function to initiate the sign-in process
130
241
  - `signout()`: Function to sign out the user
242
+ - `signinWithCode(code: string, state?: string)`: Function to manually handle OAuth authorization codes
243
+ - `getSignInLink(redirectUri?: string)`: Function to generate OAuth sign-in URLs
131
244
  - `db`: Object for database operations
132
245
 
246
+ #### signinWithCode Parameters:
247
+ - `code` (required): The OAuth authorization code received from the OAuth provider
248
+ - `state` (optional): The state parameter for CSRF protection validation
249
+
250
+ #### signinWithCode Returns:
251
+ - `Promise<{ success: boolean, error?: string }>`: Returns success status and optional error message
252
+
253
+ #### getSignInLink Parameters:
254
+ - `redirectUri` (optional): Custom redirect URI for OAuth flow. If not provided, defaults to current page URL
255
+
256
+ #### getSignInLink Returns:
257
+ - `string`: Complete OAuth authorization URL
258
+
133
259
 
134
260
 
135
261
  db methods: