diff --git a/common/calculate-cpmm.ts b/common/calculate-cpmm.ts index 6b14211d..4ced5b16 100644 --- a/common/calculate-cpmm.ts +++ b/common/calculate-cpmm.ts @@ -63,10 +63,8 @@ export function getCpmmLiquidityFee( bet: number, outcome: string ) { - const probBefore = getCpmmProbability(contract.pool, contract.p) - const probAfter = getCpmmProbabilityAfterBetBeforeFees(contract, outcome, bet) - const probMid = Math.sqrt(probBefore * probAfter) - const betP = outcome === 'YES' ? 1 - probMid : probMid + const prob = getCpmmProbabilityAfterBetBeforeFees(contract, outcome, bet) + const betP = outcome === 'YES' ? 1 - prob : prob const liquidityFee = LIQUIDITY_FEE * betP * bet const platformFee = PLATFORM_FEE * betP * bet diff --git a/functions/README.md b/functions/README.md index 7601b7c8..e8debc6e 100644 --- a/functions/README.md +++ b/functions/README.md @@ -24,8 +24,9 @@ Adapted from https://firebase.google.com/docs/functions/get-started 0. `$ firebase functions:config:get > .runtimeconfig.json` to cache secrets for local dev 1. [Install](https://cloud.google.com/sdk/docs/install) gcloud CLI -2. `$ brew install java` to install java if you don't already have it - 1. `$ echo 'export PATH="/usr/local/opt/openjdk/bin:$PATH"' >> ~/.zshrc` to add java to your path +2. If you don't have java (or see the error `Error: Process java -version has exited with code 1. Please make sure Java is installed and on your system PATH.`): + 1. `$ brew install java` + 2. `$ sudo ln -sfn /opt/homebrew/opt/openjdk/libexec/openjdk.jdk /Library/Java/JavaVirtualMachines/openjdk.jdk` 3. `$ gcloud auth login` to authenticate the CLI tools to Google Cloud 4. `$ gcloud config set project ` to choose the project (`$ gcloud projects list` to see options) 5. `$ mkdir firestore_export` to create a folder to store the exported database diff --git a/web/components/bet-panel.tsx b/web/components/bet-panel.tsx index bb3cacb8..8010e8de 100644 --- a/web/components/bet-panel.tsx +++ b/web/components/bet-panel.tsx @@ -14,7 +14,7 @@ import { formatWithCommas, } from 'common/util/format' import { Title } from './title' -import { firebaseLogin, User } from 'web/lib/firebase/users' +import { User } from 'web/lib/firebase/users' import { Bet } from 'common/bet' import { APIError, placeBet } from 'web/lib/firebase/api-call' import { sellShares } from 'web/lib/firebase/fn-call' @@ -36,6 +36,7 @@ import { } from 'common/calculate-cpmm' import { SellRow } from './sell-row' import { useSaveShares } from './use-save-shares' +import { SignUpPrompt } from './sign-up-prompt' export function BetPanel(props: { contract: FullContract @@ -70,14 +71,7 @@ export function BetPanel(props: { - {user === null && ( - - )} + ) @@ -183,14 +177,7 @@ export function BetPanelSwitcher(props: { /> )} - {user === null && ( - - )} + ) diff --git a/web/components/contract/contract-card.tsx b/web/components/contract/contract-card.tsx index 8fffd844..6f83ccb9 100644 --- a/web/components/contract/contract-card.tsx +++ b/web/components/contract/contract-card.tsx @@ -6,7 +6,6 @@ import { Contract, contractPath, getBinaryProbPercent, - listenForContract, } from 'web/lib/firebase/contracts' import { Col } from '../layout/col' import { @@ -26,8 +25,7 @@ import { import { getOutcomeProbability, getTopAnswer } from 'common/calculate' import { AvatarDetails, MiscDetails } from './contract-details' import { getExpectedValue, getValueFromBucket } from 'common/calculate-dpm' -import { useEffect, useState } from 'react' -import { QuickBet, QuickOutcomeView, ProbBar, getColor } from './quick-bet' +import { QuickBet, ProbBar, getColor } from './quick-bet' import { useContractWithPreload } from 'web/hooks/use-contract' export function ContractCard(props: { @@ -39,6 +37,7 @@ export function ContractCard(props: { const { showHotVolume, showCloseTime, className } = props const contract = useContractWithPreload(props.contract) ?? props.contract const { question, outcomeType } = contract + const { resolution } = contract const marketClosed = (contract.closeTime || Infinity) < Date.now() const showQuickBet = !( @@ -54,7 +53,7 @@ export function ContractCard(props: { className )} > - +

{question}

- {outcomeType === 'FREE_RESPONSE' && ( - } - truncate="long" - /> - )} + {outcomeType === 'FREE_RESPONSE' && + (resolution ? ( + + ) : ( + } + truncate="long" + /> + ))} ) : ( - + {outcomeType === 'BINARY' && ( + + )} + + {outcomeType === 'NUMERIC' && ( + + )} + + {outcomeType === 'FREE_RESPONSE' && ( + } + truncate="long" + /> + )} + )} -
) @@ -106,9 +132,8 @@ export function BinaryResolutionOrChance(props: { contract: FullContract large?: boolean className?: string - hideText?: boolean }) { - const { contract, large, className, hideText } = props + const { contract, large, className } = props const { resolution } = contract const textColor = `text-${getColor(contract)}` @@ -129,11 +154,9 @@ export function BinaryResolutionOrChance(props: { ) : ( <>
{getBinaryProbPercent(contract)}
- {!hideText && ( -
- chance -
- )} +
+ chance +
)} @@ -162,9 +185,8 @@ export function FreeResponseResolutionOrChance(props: { contract: FreeResponseContract truncate: 'short' | 'long' | 'none' className?: string - hideText?: boolean }) { - const { contract, truncate, className, hideText } = props + const { contract, truncate, className } = props const { resolution } = contract const topAnswer = getTopAnswer(contract) @@ -174,13 +196,17 @@ export function FreeResponseResolutionOrChance(props: { {resolution ? ( <> -
Resolved
- +
+ Resolved +
+ {(resolution === 'CANCEL' || resolution === 'MKT') && ( + + )} ) : ( topAnswer && ( @@ -189,7 +215,7 @@ export function FreeResponseResolutionOrChance(props: {
{formatPercent(getOutcomeProbability(contract, topAnswer.id))}
- {!hideText &&
chance
} +
chance
) @@ -201,9 +227,8 @@ export function FreeResponseResolutionOrChance(props: { export function NumericResolutionOrExpectation(props: { contract: NumericContract className?: string - hideText?: boolean }) { - const { contract, className, hideText } = props + const { contract, className } = props const { resolution } = contract const textColor = `text-${getColor(contract)}` @@ -222,9 +247,7 @@ export function NumericResolutionOrExpectation(props: {
{formatLargeNumber(getExpectedValue(contract))}
- {!hideText && ( -
expected
- )} +
expected
)} diff --git a/web/components/contract/contract-details.tsx b/web/components/contract/contract-details.tsx index 3d8789d2..42894263 100644 --- a/web/components/contract/contract-details.tsx +++ b/web/components/contract/contract-details.tsx @@ -14,6 +14,7 @@ import { UserLink } from '../user-page' import { Contract, contractMetrics, + contractPool, updateContract, } from 'web/lib/firebase/contracts' import { Col } from '../layout/col' @@ -43,10 +44,6 @@ export function MiscDetails(props: { return ( - {categories.length > 0 && ( - - )} - {showHotVolume ? ( {formatMoney(volume24Hours)} @@ -58,10 +55,14 @@ export function MiscDetails(props: { {fromNow(closeTime || 0)} ) : volume > 0 ? ( - {volumeLabel} + {contractPool(contract)} pool ) : ( )} + + {categories.length > 0 && ( + + )} ) } @@ -71,7 +72,7 @@ export function AvatarDetails(props: { contract: Contract }) { const { creatorName, creatorUsername } = contract return ( - + {tradersCount} - {contract.mechanism === 'cpmm-1' && ( - - Liquidity - {formatMoney(contract.totalLiquidity)} - - )} - - {contract.mechanism === 'dpm-2' && ( - - Pool - {formatMoney(sum(Object.values(contract.pool)))} - - )} + + Pool + {contractPool(contract)} + diff --git a/web/components/contract/quick-bet.tsx b/web/components/contract/quick-bet.tsx index 35fcad50..daca33a7 100644 --- a/web/components/contract/quick-bet.tsx +++ b/web/components/contract/quick-bet.tsx @@ -1,5 +1,9 @@ import clsx from 'clsx' -import { getOutcomeProbability, getTopAnswer } from 'common/calculate' +import { + getOutcomeProbability, + getOutcomeProbabilityAfterBet, + getTopAnswer, +} from 'common/calculate' import { getExpectedValue } from 'common/calculate-dpm' import { Contract, @@ -8,55 +12,81 @@ import { DPM, Binary, NumericContract, - FreeResponse, + FreeResponseContract, } from 'common/contract' -import { formatMoney } from 'common/util/format' +import { + formatLargeNumber, + formatMoney, + formatPercent, +} from 'common/util/format' +import { useState } from 'react' import toast from 'react-hot-toast' import { useUser } from 'web/hooks/use-user' import { useUserContractBets } from 'web/hooks/use-user-bets' import { placeBet } from 'web/lib/firebase/api-call' -import { getBinaryProb } from 'web/lib/firebase/contracts' +import { getBinaryProb, getBinaryProbPercent } from 'web/lib/firebase/contracts' import TriangleDownFillIcon from 'web/lib/icons/triangle-down-fill-icon' import TriangleFillIcon from 'web/lib/icons/triangle-fill-icon' import { Col } from '../layout/col' import { OUTCOME_TO_COLOR } from '../outcome-label' import { useSaveShares } from '../use-save-shares' -import { - BinaryResolutionOrChance, - NumericResolutionOrExpectation, - FreeResponseResolutionOrChance, -} from './contract-card' + +const BET_SIZE = 10 export function QuickBet(props: { contract: Contract }) { const { contract } = props const user = useUser() const userBets = useUserContractBets(user?.id, contract.id) - const { yesFloorShares, noFloorShares, yesShares, noShares } = useSaveShares( + const { yesFloorShares, noFloorShares } = useSaveShares( contract as FullContract, userBets ) - // TODO: For some reason, Floor Shares are inverted for non-BINARY markets + // TODO: This relies on a hack in useSaveShares, where noFloorShares includes + // all non-YES shares. Ideally, useSaveShares should group by all outcomes const hasUpShares = contract.outcomeType === 'BINARY' ? yesFloorShares : noFloorShares const hasDownShares = contract.outcomeType === 'BINARY' ? noFloorShares : yesFloorShares - const color = getColor(contract) + const [upHover, setUpHover] = useState(false) + const [downHover, setDownHover] = useState(false) + + let previewProb = undefined + try { + previewProb = upHover + ? getOutcomeProbabilityAfterBet( + contract, + quickOutcome(contract, 'UP') || '', + BET_SIZE + ) + : downHover + ? 1 - + getOutcomeProbabilityAfterBet( + contract, + quickOutcome(contract, 'DOWN') || '', + BET_SIZE + ) + : undefined + } catch (e) { + // Catch any errors from hovering on an invalid option + } + + const color = getColor(contract, previewProb) async function placeQuickBet(direction: 'UP' | 'DOWN') { const betPromise = async () => { const outcome = quickOutcome(contract, direction) return await placeBet({ - amount: 10, + amount: BET_SIZE, outcome, contractId: contract.id, }) } const shortQ = contract.question.slice(0, 20) toast.promise(betPromise(), { - loading: `${formatMoney(10)} on "${shortQ}"...`, - success: `${formatMoney(10)} on "${shortQ}"...`, + loading: `${formatMoney(BET_SIZE)} on "${shortQ}"...`, + success: `${formatMoney(BET_SIZE)} on "${shortQ}"...`, error: (err) => `${err.message}`, }) } @@ -68,7 +98,7 @@ export function QuickBet(props: { contract: Contract }) { if (contract.outcomeType === 'FREE_RESPONSE') { // TODO: Implement shorting of free response answers if (direction === 'DOWN') { - throw new Error("Can't short free response answers") + throw new Error("Can't bet against free response answers") } return getTopAnswer(contract)?.id } @@ -81,18 +111,19 @@ export function QuickBet(props: { contract: Contract }) { return ( {/* Up bet triangle */}
setUpHover(true)} + onMouseLeave={() => setUpHover(false)} onClick={() => placeQuickBet('UP')} - >
+ />
{formatMoney(10)}
@@ -101,31 +132,43 @@ export function QuickBet(props: { contract: Contract }) { ) : ( - + )}
- + {/* Down bet triangle */}
setDownHover(true)} + onMouseLeave={() => setDownHover(false)} onClick={() => placeQuickBet('DOWN')} >
{hasDownShares > 0 ? ( ) : ( - + )}
{formatMoney(10)} @@ -135,10 +178,10 @@ export function QuickBet(props: { contract: Contract }) { ) } -export function ProbBar(props: { contract: Contract }) { - const { contract } = props - const color = getColor(contract) - const prob = getProb(contract) +export function ProbBar(props: { contract: Contract; previewProb?: number }) { + const { contract, previewProb } = props + const color = getColor(contract, previewProb) + const prob = previewProb ?? getProb(contract) return ( <>
+ />
+ /> ) } -export function QuickOutcomeView(props: { contract: Contract }) { - const { contract } = props +function QuickOutcomeView(props: { + contract: Contract + previewProb?: number + caption?: 'chance' | 'expected' +}) { + const { contract, previewProb, caption } = props const { outcomeType } = contract + // If there's a preview prob, display that instead of the current prob + const override = + previewProb === undefined ? undefined : formatPercent(previewProb) + const textColor = `text-${getColor(contract, previewProb)}` + + let display: string | undefined + switch (outcomeType) { + case 'BINARY': + display = getBinaryProbPercent(contract) + break + case 'NUMERIC': + display = formatLargeNumber(getExpectedValue(contract as NumericContract)) + break + case 'FREE_RESPONSE': + const topAnswer = getTopAnswer(contract as FreeResponseContract) + display = + topAnswer && + formatPercent(getOutcomeProbability(contract, topAnswer.id)) + break + } + return ( - <> - {outcomeType === 'BINARY' && ( - - )} - - {outcomeType === 'NUMERIC' && ( - - )} - - {outcomeType === 'FREE_RESPONSE' && ( - } - truncate="long" - hideText - /> - )} - + + {override ?? display} + {caption &&
{caption}
} + + ) } @@ -215,15 +262,14 @@ function getNumericScale(contract: NumericContract) { return (ev - min) / (max - min) } -export function getColor(contract: Contract) { +export function getColor(contract: Contract, previewProb?: number) { // TODO: Not sure why eg green-400 doesn't work here; try upgrading Tailwind // TODO: Try injecting a gradient here // return 'primary' const { resolution } = contract if (resolution) { return ( - // @ts-ignore; TODO: Have better typing for contract.resolution? - OUTCOME_TO_COLOR[resolution] || + OUTCOME_TO_COLOR[resolution as 'YES' | 'NO' | 'CANCEL' | 'MKT'] ?? // If resolved to a FR answer, use 'primary' 'primary' ) @@ -233,9 +279,6 @@ export function getColor(contract: Contract) { } const marketClosed = (contract.closeTime || Infinity) < Date.now() - return marketClosed - ? 'gray-400' - : getProb(contract) >= 0.5 - ? 'primary' - : 'red-400' + const prob = previewProb ?? getProb(contract) + return marketClosed ? 'gray-400' : prob >= 0.5 ? 'primary' : 'red-400' } diff --git a/web/components/numeric-bet-panel.tsx b/web/components/numeric-bet-panel.tsx index 20478e25..f249e3c3 100644 --- a/web/components/numeric-bet-panel.tsx +++ b/web/components/numeric-bet-panel.tsx @@ -12,12 +12,13 @@ import { formatPercent, formatMoney } from 'common/util/format' import { useUser } from '../hooks/use-user' import { APIError, placeBet } from '../lib/firebase/api-call' -import { firebaseLogin, User } from '../lib/firebase/users' +import { User } from '../lib/firebase/users' import { BuyAmountInput } from './amount-input' import { BucketInput } from './bucket-input' import { Col } from './layout/col' import { Row } from './layout/row' import { Spacer } from './layout/spacer' +import { SignUpPrompt } from './sign-up-prompt' export function NumericBetPanel(props: { contract: NumericContract @@ -32,14 +33,7 @@ export function NumericBetPanel(props: { - {user === null && ( - - )} + ) } diff --git a/web/components/sign-up-prompt.tsx b/web/components/sign-up-prompt.tsx new file mode 100644 index 00000000..256a64d7 --- /dev/null +++ b/web/components/sign-up-prompt.tsx @@ -0,0 +1,16 @@ +import React from 'react' +import { useUser } from 'web/hooks/use-user' +import { firebaseLogin } from 'web/lib/firebase/users' + +export function SignUpPrompt() { + const user = useUser() + + return user === null ? ( + + ) : null +} diff --git a/web/lib/firebase/contracts.ts b/web/lib/firebase/contracts.ts index cead2c6a..81767dcb 100644 --- a/web/lib/firebase/contracts.ts +++ b/web/lib/firebase/contracts.ts @@ -13,7 +13,7 @@ import { updateDoc, limit, } from 'firebase/firestore' -import { range, sortBy } from 'lodash' +import { range, sortBy, sum } from 'lodash' import { app } from './init' import { getValues, listenForValue, listenForValues } from './utils' @@ -56,6 +56,14 @@ export function contractMetrics(contract: Contract) { return { volumeLabel, createdDate, automaticResolutionDate, resolvedDate } } +export function contractPool(contract: Contract) { + return contract.mechanism === 'cpmm-1' + ? formatMoney(contract.totalLiquidity) + : contract.mechanism === 'dpm-2' + ? formatMoney(sum(Object.values(contract.pool))) + : 'Empty pool' +} + export function getBinaryProb(contract: FullContract) { const { totalShares, pool, p, resolutionProbability, mechanism } = contract diff --git a/web/lib/icons/triangle-down-fill-icon.tsx b/web/lib/icons/triangle-down-fill-icon.tsx index 28ed5ba6..1c5b7ab4 100644 --- a/web/lib/icons/triangle-down-fill-icon.tsx +++ b/web/lib/icons/triangle-down-fill-icon.tsx @@ -8,7 +8,7 @@ export default function TriangleDownFillIcon(props: { className?: string }) { viewBox="0 0 16 16" > diff --git a/web/lib/icons/triangle-fill-icon.tsx b/web/lib/icons/triangle-fill-icon.tsx index e24c005f..0894df20 100644 --- a/web/lib/icons/triangle-fill-icon.tsx +++ b/web/lib/icons/triangle-fill-icon.tsx @@ -8,7 +8,7 @@ export default function TriangleFillIcon(props: { className?: string }) { viewBox="0 0 16 16" > diff --git a/web/pages/api/v0/_types.ts b/web/pages/api/v0/_types.ts index 355b1973..9f043c80 100644 --- a/web/pages/api/v0/_types.ts +++ b/web/pages/api/v0/_types.ts @@ -20,6 +20,7 @@ export type LiteMarket = { description: string tags: string[] url: string + outcomeType: string mechanism: string pool: number @@ -57,6 +58,7 @@ export function toLiteMarket(contract: Contract): LiteMarket { tags, slug, pool, + outcomeType, mechanism, volume7Days, volume24Hours, @@ -88,6 +90,7 @@ export function toLiteMarket(contract: Contract): LiteMarket { probability, p, totalLiquidity, + outcomeType, mechanism, volume7Days, volume24Hours,