60f68b178d
* Fold type, fold page, query for fold contracts * Tsconfig: target esnext, nounused locals: false * Store tags in field on contract. Script to update contract tags * Show tags on fold page * Load all fold comments server-side to serve better feed * Fix the annoying firebase already initialized error! * Add links to /edit and /leaderboards for fold * Page with list of folds * UI for creating a fold * Create a fold * Edit fold page
72 lines
2.1 KiB
TypeScript
72 lines
2.1 KiB
TypeScript
import { collection, doc, query, updateDoc, where } from 'firebase/firestore'
|
|
import { Fold } from '../../../common/fold'
|
|
import { Contract, contractCollection } from './contracts'
|
|
import { db } from './init'
|
|
import { getValues } from './utils'
|
|
|
|
const foldCollection = collection(db, 'folds')
|
|
|
|
export function foldPath(fold: Fold, subpath?: 'edit' | 'leaderboards') {
|
|
return `/fold/${fold.slug}${subpath ? `/${subpath}` : ''}`
|
|
}
|
|
|
|
export function updateFold(fold: Fold, updates: Partial<Fold>) {
|
|
return updateDoc(doc(foldCollection, fold.id), updates)
|
|
}
|
|
|
|
export async function listAllFolds() {
|
|
return getValues<Fold>(foldCollection)
|
|
}
|
|
|
|
export async function getFoldBySlug(slug: string) {
|
|
const q = query(foldCollection, where('slug', '==', slug))
|
|
const folds = await getValues<Fold>(q)
|
|
|
|
return folds.length === 0 ? null : folds[0]
|
|
}
|
|
|
|
export async function getFoldContracts(fold: Fold) {
|
|
const {
|
|
tags,
|
|
contractIds,
|
|
excludedContractIds,
|
|
creatorIds,
|
|
excludedCreatorIds,
|
|
} = fold
|
|
|
|
const [tagsContracts, includedContracts] = await Promise.all([
|
|
// TODO: if tags.length > 10, execute multiple parallel queries
|
|
tags.length > 0
|
|
? getValues<Contract>(
|
|
query(contractCollection, where('tags', 'array-contains-any', tags))
|
|
)
|
|
: [],
|
|
|
|
// TODO: if contractIds.length > 10, execute multiple parallel queries
|
|
contractIds.length > 0
|
|
? getValues<Contract>(
|
|
query(contractCollection, where('id', 'in', contractIds))
|
|
)
|
|
: [],
|
|
])
|
|
|
|
const excludedContractsSet = new Set(excludedContractIds)
|
|
|
|
const creatorSet = creatorIds ? new Set(creatorIds) : undefined
|
|
const excludedCreatorSet = excludedCreatorIds
|
|
? new Set(excludedCreatorIds)
|
|
: undefined
|
|
|
|
const approvedContracts = tagsContracts.filter((contract) => {
|
|
const { id, creatorId } = contract
|
|
|
|
if (excludedContractsSet.has(id)) return false
|
|
if (creatorSet && !creatorSet.has(creatorId)) return false
|
|
if (excludedCreatorSet && excludedCreatorSet.has(creatorId)) return false
|
|
|
|
return true
|
|
})
|
|
|
|
return [...approvedContracts, ...includedContracts]
|
|
}
|