mirror of
https://github.com/supabase/supabase.git
synced 2026-07-04 02:54:28 +08:00
210 lines
5.5 KiB
TypeScript
210 lines
5.5 KiB
TypeScript
import { v4 as _uuidV4 } from 'uuid'
|
|
import { post } from 'lib/common/fetch'
|
|
import { PASSWORD_STRENGTH, DEFAULT_MINIMUM_PASSWORD_STRENGTH, API_URL } from 'lib/constants'
|
|
|
|
export const tryParseJson = (jsonString: any) => {
|
|
try {
|
|
let parsed = JSON.parse(jsonString)
|
|
return parsed
|
|
} catch (error) {
|
|
return undefined
|
|
}
|
|
}
|
|
|
|
export const minifyJSON = (prettifiedJSON: string) => {
|
|
try {
|
|
const res = JSON.stringify(JSON.parse(prettifiedJSON))
|
|
if (!isNaN(Number(res))) return Number(res)
|
|
else return res
|
|
} catch (err) {
|
|
throw err
|
|
}
|
|
}
|
|
|
|
export const prettifyJSON = (minifiedJSON: string) => {
|
|
try {
|
|
if (minifiedJSON && minifiedJSON.length > 0) {
|
|
return JSON.stringify(JSON.parse(minifiedJSON), undefined, 2)
|
|
} else {
|
|
return minifiedJSON
|
|
}
|
|
} catch (err) {
|
|
// dont need to throw error, just return text value
|
|
// Users have to fix format if they want to save
|
|
return minifiedJSON
|
|
}
|
|
}
|
|
|
|
export const uuidv4 = () => {
|
|
return _uuidV4()
|
|
}
|
|
|
|
export const timeout = (ms: number) => {
|
|
return new Promise((resolve) => setTimeout(resolve, ms))
|
|
}
|
|
|
|
export const getURL = () => {
|
|
const url =
|
|
process?.env?.NEXT_PUBLIC_SITE_URL && process.env.NEXT_PUBLIC_SITE_URL !== ''
|
|
? process.env.NEXT_PUBLIC_SITE_URL
|
|
: process?.env?.VERCEL_URL && process.env.VERCEL_URL !== ''
|
|
? process.env.VERCEL_URL
|
|
: 'https://app.supabase.com'
|
|
return url.includes('http') ? url : `https://${url}`
|
|
}
|
|
|
|
/**
|
|
* Generates a random string using alpha characters
|
|
*/
|
|
export const makeRandomString = (length: number) => {
|
|
var result = ''
|
|
var characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'
|
|
var charactersLength = characters.length
|
|
for (var i = 0; i < length; i++) {
|
|
result += characters.charAt(Math.floor(Math.random() * charactersLength))
|
|
}
|
|
return result.toString()
|
|
}
|
|
|
|
/**
|
|
* Get a subset of fields from an object
|
|
* @param {object} model
|
|
* @param {array} fields a list of properties to pluck. eg: ['first_name', 'last_name']
|
|
*/
|
|
export const pluckObjectFields = (model: any, fields: any[]) => {
|
|
let o: any = {}
|
|
fields.forEach((field) => {
|
|
o[field] = model[field]
|
|
})
|
|
return o
|
|
}
|
|
|
|
/**
|
|
* Trims down a JSON Schema only to the fields that a user wants.
|
|
* @param {object} jsonSchema
|
|
* @param {array} fields a list of properties to pluck. eg: ['first_name', 'last_name']
|
|
*/
|
|
export const pluckJsonSchemaFields = (jsonSchema: any, fields: any) => {
|
|
let schema: any = {
|
|
type: 'object',
|
|
required: [],
|
|
properties: {},
|
|
}
|
|
fields.forEach((field: any) => {
|
|
if (jsonSchema.properties[field]) {
|
|
schema.properties[field] = jsonSchema.properties[field]
|
|
if (jsonSchema.required.includes(field)) schema.required.push(field)
|
|
}
|
|
})
|
|
return schema
|
|
}
|
|
|
|
/**
|
|
* Before return to frontend, we should filter sensitive project props
|
|
*/
|
|
export const filterSensitiveProjectProps = (project: any) => {
|
|
project.db_user_supabase = undefined
|
|
project.db_pass_supabase = undefined
|
|
|
|
return project
|
|
}
|
|
|
|
/**
|
|
* Returns undefine if the string isn't parse-able
|
|
*/
|
|
export const tryParseInt = (str: string) => {
|
|
try {
|
|
return parseInt(str, 10)
|
|
} catch (error) {
|
|
return undefined
|
|
}
|
|
}
|
|
|
|
// Used as checker for memoised components
|
|
export const propsAreEqual = (prevProps: any, nextProps: any) => {
|
|
try {
|
|
Object.keys(prevProps).forEach((key) => {
|
|
if (typeof prevProps[key] !== 'function') {
|
|
if (prevProps[key] !== nextProps[key]) {
|
|
throw new Error()
|
|
}
|
|
}
|
|
})
|
|
return true
|
|
} catch (e) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
export const formatBytes = (bytes: any, decimals = 2) => {
|
|
if (bytes === 0 || bytes === undefined) return '0 bytes'
|
|
|
|
const k = 1000
|
|
const dm = decimals < 0 ? 0 : decimals
|
|
const sizes = ['bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']
|
|
|
|
const i = Math.floor(Math.log(bytes) / Math.log(k))
|
|
|
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i]
|
|
}
|
|
|
|
export const snakeToCamel = (str: string) =>
|
|
str.replace(/([-_][a-z])/g, (group: string) =>
|
|
group.toUpperCase().replace('-', '').replace('_', '')
|
|
)
|
|
|
|
export const copyToClipboard = (str: string, callback = () => {}) => {
|
|
const focused = window.document.hasFocus()
|
|
if (focused) {
|
|
window.navigator?.clipboard?.writeText(str).then(callback)
|
|
} else {
|
|
console.warn('Unable to copy to clipboard')
|
|
}
|
|
}
|
|
|
|
export async function passwordStrength(value: string) {
|
|
let message = ''
|
|
let warning = ''
|
|
let strength = 0
|
|
|
|
if (value && value !== '') {
|
|
const response = await post(`${API_URL}/profile/password-check`, { password: value })
|
|
if (!response.error) {
|
|
const { result } = response
|
|
const score = (PASSWORD_STRENGTH as any)[result.score]
|
|
const suggestions = result.feedback?.suggestions ? result.feedback.suggestions.join(' ') : ''
|
|
|
|
// set message :string
|
|
message = `${score} ${suggestions}`
|
|
|
|
// set strength :number
|
|
strength = result.score
|
|
|
|
// warning message for anything below 4 strength :string
|
|
if (result.score < DEFAULT_MINIMUM_PASSWORD_STRENGTH) {
|
|
warning = `${
|
|
result?.feedback?.warning ? result?.feedback?.warning + '.' : ''
|
|
} You need a stronger password.`
|
|
}
|
|
}
|
|
}
|
|
|
|
return {
|
|
message,
|
|
warning,
|
|
strength,
|
|
}
|
|
}
|
|
|
|
export const detectBrowser = () => {
|
|
if (!navigator) return undefined
|
|
|
|
if (navigator.userAgent.indexOf('Chrome') !== -1) {
|
|
return 'Chrome'
|
|
} else if (navigator.userAgent.indexOf('Firefox') !== -1) {
|
|
return 'Firefox'
|
|
} else if (navigator.userAgent.indexOf('Safari') !== -1) {
|
|
return 'Safari'
|
|
}
|
|
}
|