From b2b38d89249bcc1209b08de3d1d6dc6e1e030f80 Mon Sep 17 00:00:00 2001
From: ekzyis <27162016+ekzyis@users.noreply.github.com>
Date: Mon, 2 Oct 2023 01:03:52 +0200
Subject: [PATCH] Images v2 (#513)
---
.env.sample | 6 +-
api/resolvers/imgproxy/index.js | 69 ------
api/resolvers/item.js | 11 +-
api/typeDefs/item.js | 1 +
components/comment.js | 2 +-
components/form.js | 2 +-
components/image.js | 221 ++++++++++++++++++
components/item-full.js | 2 +-
components/modal.js | 31 ++-
components/text.js | 150 ++----------
fragments/comments.js | 1 +
fragments/items.js | 1 +
.../migration.sql | 175 ++++++++++++++
prisma/schema.prisma | 1 +
public/placeholder_click_to_load.png | Bin 0 -> 9254 bytes
public/placeholder_processing.png | Bin 0 -> 6400 bytes
scripts/imgproxy.js | 71 ++++++
styles/globals.scss | 29 +++
worker/imgproxy.js | 194 +++++++++++++++
worker/index.js | 2 +
20 files changed, 747 insertions(+), 222 deletions(-)
delete mode 100644 api/resolvers/imgproxy/index.js
create mode 100644 components/image.js
create mode 100644 prisma/migrations/20230920192620_item_imgproxy_urls/migration.sql
create mode 100644 public/placeholder_click_to_load.png
create mode 100644 public/placeholder_processing.png
create mode 100644 scripts/imgproxy.js
create mode 100644 worker/imgproxy.js
diff --git a/.env.sample b/.env.sample
index c5ea5bcf..f9aa0d4b 100644
--- a/.env.sample
+++ b/.env.sample
@@ -71,8 +71,10 @@ OPENSEARCH_PASSWORD=
# imgproxy options
IMGPROXY_ENABLE_WEBP_DETECTION=1
-IMGPROXY_MAX_ANIMATION_FRAMES=100
-IMGPROXY_MAX_SRC_RESOLUTION=200
+IMGPROXY_ENABLE_AVIF_DETECTION=1
+IMGPROXY_MAX_ANIMATION_FRAMES=2000
+IMGPROXY_MAX_SRC_RESOLUTION=50
+IMGPROXY_MAX_ANIMATION_FRAME_RESOLUTION=200
IMGPROXY_READ_TIMEOUT=10
IMGPROXY_WRITE_TIMEOUT=10
IMGPROXY_DOWNLOAD_TIMEOUT=9
diff --git a/api/resolvers/imgproxy/index.js b/api/resolvers/imgproxy/index.js
deleted file mode 100644
index 39d7a0b6..00000000
--- a/api/resolvers/imgproxy/index.js
+++ /dev/null
@@ -1,69 +0,0 @@
-import { createHmac } from 'node:crypto'
-import { extractUrls } from '../../../lib/md'
-
-const imgProxyEnabled = process.env.NODE_ENV === 'production' ||
- (process.env.NEXT_PUBLIC_IMGPROXY_URL && process.env.IMGPROXY_SALT && process.env.IMGPROXY_KEY)
-
-if (!imgProxyEnabled) {
- console.warn('IMGPROXY_* env vars not set, imgproxy calls are no-ops now')
-}
-
-const IMGPROXY_URL = process.env.NEXT_PUBLIC_IMGPROXY_URL
-const IMGPROXY_SALT = process.env.IMGPROXY_SALT
-const IMGPROXY_KEY = process.env.IMGPROXY_KEY
-
-const hexDecode = (hex) => Buffer.from(hex, 'hex')
-
-const sign = (target) => {
- // https://github.com/imgproxy/imgproxy/blob/master/examples/signature.js
- const hmac = createHmac('sha256', hexDecode(IMGPROXY_KEY))
- hmac.update(hexDecode(IMGPROXY_SALT))
- hmac.update(target)
- return hmac.digest('base64url')
-}
-
-const createImageProxyUrl = url => {
- const processingOptions = '/rs:fit:600:500:0/g:no'
- const b64Url = Buffer.from(url, 'utf-8').toString('base64url')
- const target = `${processingOptions}/${b64Url}`
- const signature = sign(target)
- return `${IMGPROXY_URL}${signature}${target}`
-}
-
-async function fetchWithTimeout (resource, { timeout = 1000, ...options } = {}) {
- const controller = new AbortController()
- const id = setTimeout(() => controller.abort(), timeout)
-
- const response = await fetch(resource, {
- ...options,
- signal: controller.signal
- })
- clearTimeout(id)
-
- return response
-}
-
-const isImageURL = async url => {
- // https://stackoverflow.com/a/68118683
- try {
- const res = await fetchWithTimeout(url, { method: 'HEAD' })
- const buf = await res.blob()
- return buf.type.startsWith('image/')
- } catch (err) {
- console.log(url, err)
- return false
- }
-}
-
-export const proxyImages = async text => {
- if (!imgProxyEnabled) return text
-
- const urls = extractUrls(text)
- for (const url of urls) {
- if (url.startsWith(IMGPROXY_URL)) continue
- if (!(await isImageURL(url))) continue
- const proxyUrl = createImageProxyUrl(url)
- text = text.replaceAll(url, proxyUrl)
- }
- return text
-}
diff --git a/api/resolvers/item.js b/api/resolvers/item.js
index e895fc0a..eb3460bf 100644
--- a/api/resolvers/item.js
+++ b/api/resolvers/item.js
@@ -14,7 +14,6 @@ import { parse } from 'tldts'
import uu from 'url-unshort'
import { advSchema, amountSchema, bountySchema, commentSchema, discussionSchema, jobSchema, linkSchema, pollSchema, ssValidate } from '../../lib/validate'
import { sendUserNotification } from '../webPush'
-import { proxyImages } from './imgproxy'
import { defaultCommentSort } from '../../lib/item'
import { notifyItemParents, notifyUserSubscribers, notifyZapped } from '../../lib/push-notifications'
@@ -1019,13 +1018,9 @@ export const updateItem = async (parent, { sub: subName, forward, options, ...it
throw new GraphQLError('item can no longer be editted', { extensions: { code: 'BAD_INPUT' } })
}
- if (item.text) {
- item.text = await proxyImages(item.text)
- }
if (item.url && typeof item.maxBid === 'undefined') {
item.url = ensureProtocol(item.url)
item.url = removeTracking(item.url)
- item.url = await proxyImages(item.url)
}
// only update item with the boost delta ... this is a bit of hack given the way
// boost used to work
@@ -1063,13 +1058,9 @@ export const createItem = async (parent, { forward, options, ...item }, { me, mo
item.userId = me ? Number(me.id) : ANON_USER_ID
const fwdUsers = await getForwardUsers(models, forward)
- if (item.text) {
- item.text = await proxyImages(item.text)
- }
if (item.url && typeof item.maxBid === 'undefined') {
item.url = ensureProtocol(item.url)
item.url = removeTracking(item.url)
- item.url = await proxyImages(item.url)
}
const enforceFee = me ? undefined : (item.parentId ? ANON_COMMENT_FEE : (ANON_POST_FEE + (item.boost || 0)))
@@ -1113,7 +1104,7 @@ export const SELECT =
"Item"."subName", "Item".status, "Item"."uploadId", "Item"."pollCost", "Item".boost, "Item".msats,
"Item".ncomments, "Item"."commentMsats", "Item"."lastCommentAt", "Item"."weightedVotes",
"Item"."weightedDownVotes", "Item".freebie, "Item"."otsHash", "Item"."bountyPaidTo",
- ltree2text("Item"."path") AS "path", "Item"."weightedComments"`
+ ltree2text("Item"."path") AS "path", "Item"."weightedComments", "Item"."imgproxyUrls"`
async function topOrderByWeightedSats (me, models) {
return `ORDER BY ${await orderByNumerator(me, models)} DESC NULLS LAST, "Item".id DESC`
diff --git a/api/typeDefs/item.js b/api/typeDefs/item.js
index 6a6fc51c..a28099dd 100644
--- a/api/typeDefs/item.js
+++ b/api/typeDefs/item.js
@@ -114,6 +114,7 @@ export default gql`
otsHash: String
parentOtsHash: String
forwards: [ItemForward]
+ imgproxyUrls: JSONObject
}
input ItemForwardInput {
diff --git a/components/comment.js b/components/comment.js
index d92eab6c..563e11c1 100644
--- a/components/comment.js
+++ b/components/comment.js
@@ -208,7 +208,7 @@ export default function Comment ({
)
: (
-
+
{truncate ? truncateString(item.text) : item.searchText || item.text}
diff --git a/components/form.js b/components/form.js
index c94bbf0c..77dd3dad 100644
--- a/components/form.js
+++ b/components/form.js
@@ -167,7 +167,7 @@ export function MarkdownInput ({ label, topLevel, groupClassName, onChange, setH
: (
- {meta.value}
+ {meta.value}
)}
diff --git a/components/image.js b/components/image.js
new file mode 100644
index 00000000..52c87e72
--- /dev/null
+++ b/components/image.js
@@ -0,0 +1,221 @@
+import styles from './text.module.css'
+import { useState, useRef, useEffect, useMemo, useCallback } from 'react'
+import { extractUrls } from '../lib/md'
+import { IMGPROXY_URL_REGEXP, IMG_URL_REGEXP } from '../lib/url'
+import FileMissing from '../svgs/file-warning-line.svg'
+import { useShowModal } from './modal'
+import { useMe } from './me'
+import { Dropdown } from 'react-bootstrap'
+
+export function decodeOriginalUrl (imgproxyUrl) {
+ const parts = imgproxyUrl.split('/')
+ // base64url is not a known encoding in browsers
+ // so we need to replace the invalid chars
+ const b64Url = parts[parts.length - 1].replace(/-/g, '+').replace(/_/, '/')
+ const originalUrl = Buffer.from(b64Url, 'base64').toString('utf-8')
+ return originalUrl
+}
+
+export const IMG_CACHE_STATES = {
+ LOADING: 'IS_LOADING',
+ LOADED: 'IS_LOADED',
+ ERROR: 'IS_ERROR'
+}
+
+// this is the image at public/placeholder_click_to_load.png as a data URI so we don't have to rely on network to render it
+const IMAGE_CLICK_TO_LOAD_DATA_URI = ''
+
+const IMAGE_PROCESSING_DATA_URI = ''
+
+export function useImgUrlCache (text, imgproxyUrls) {
+ const ref = useRef({})
+ const [imgUrlCache, setImgUrlCache] = useState({})
+ const me = useMe()
+
+ const updateCache = (url, state) => setImgUrlCache((prev) => ({ ...prev, [url]: state }))
+
+ useEffect(() => {
+ const urls = extractUrls(text)
+
+ urls.forEach((url) => {
+ if (IMG_URL_REGEXP.test(url) || !!imgproxyUrls?.[url]) {
+ // it's probably an image if the regexp matches or if we processed the URL as an image in the worker
+ updateCache(url, IMG_CACHE_STATES.LOADED)
+ } else {
+ // don't use image detection by trying to load as an image if user opted-out of loading external images automatically
+ if (me?.clickToLoadImg) return
+ // make sure it's not a false negative by trying to load URL as
+ const img = new window.Image()
+ ref.current[url] = img
+
+ updateCache(url, IMG_CACHE_STATES.LOADING)
+
+ const callback = (state) => {
+ updateCache(url, state)
+ delete ref.current[url]
+ }
+ img.onload = () => callback(IMG_CACHE_STATES.LOADED)
+ img.onerror = () => callback(IMG_CACHE_STATES.ERROR)
+ img.src = url
+ }
+ })
+
+ return () => {
+ Object.values(ref.current).forEach((img) => {
+ img.onload = null
+ img.onerror = null
+ img.src = ''
+ })
+ }
+ }, [text])
+
+ return imgUrlCache
+}
+
+export function ZoomableImage ({ src, topLevel, srcSet: srcSetObj, ...props }) {
+ const me = useMe()
+ const showModal = useShowModal()
+ const [originalUrlConsent, setOriginalUrlConsent] = useState(!me ? true : !me.clickToLoadImg)
+ // if there is no srcset obj, image is still processing (srcSetObj === undefined) or it wasn't detected as an image by the worker (srcSetObj === null).
+ // we handle both cases the same as imgproxy errors.
+ const [imgproxyErr, setImgproxyErr] = useState(!srcSetObj)
+ const [originalErr, setOriginalErr] = useState()
+
+ // backwards compatibility:
+ // src may already be imgproxy url since we used to replace image urls with imgproxy urls
+ const originalUrl = IMGPROXY_URL_REGEXP.test(src) ? decodeOriginalUrl(src) : src
+
+ // we will fallback to the original error if there was an error with our image proxy
+ const loadOriginalUrl = !!imgproxyErr
+
+ const srcSet = useMemo(() => {
+ if (!srcSetObj) return undefined
+ // srcSetObj shape: { [widthDescriptor]: , ... }
+ return Object.entries(srcSetObj).reduce((acc, [wDescriptor, url], i, arr) => {
+ return acc + `${url} ${wDescriptor}` + (i < arr.length - 1 ? ', ' : '')
+ }, '')
+ }, [srcSetObj])
+ const sizes = `${(topLevel ? 100 : 66)}vw`
+
+ // get source url in best resolution
+ const bestResSrc = useMemo(() => {
+ if (!srcSetObj) return undefined
+ return Object.entries(srcSetObj).reduce((acc, [wDescriptor, url]) => {
+ const w = Number(wDescriptor.replace(/w$/, ''))
+ return w > acc.w ? { w, url } : acc
+ }, { w: 0, url: undefined }).url
+ }, [srcSetObj])
+
+ const onError = useCallback((err) => {
+ if (!imgproxyErr) {
+ // first error is imgproxy error since that was loaded
+ console.error('imgproxy image error:', err)
+ setImgproxyErr(true)
+ } else {
+ // second error is error from original url
+ console.error('original image error:', err)
+ setOriginalErr(true)
+ }
+ }, [setImgproxyErr, setOriginalErr, imgproxyErr, originalUrl])
+
+ const handleClick = useCallback(() => showModal(close => (
+
+
+
+ ), {
+ fullScreen: true,
+ overflow: (
+
+ {loadOriginalUrl ? 'open in new tab' : 'open original'}
+ )
+ }), [showModal, loadOriginalUrl, originalUrl, bestResSrc, onError, props])
+
+ if (!src) return null
+
+ if ((srcSetObj === undefined) && originalUrlConsent && !originalErr) {
+ // image is still processing and user is okay with loading original url automatically
+ return (
+ setOriginalErr(true)}
+ {...props}
+ />
+ )
+ }
+
+ if ((srcSetObj === undefined) && !originalUrlConsent && !originalErr) {
+ // image is still processing and user is not okay with loading original url automatically
+ const { host } = new URL(originalUrl)
+ return (
+
+
setOriginalUrlConsent(true)}
+ />
+
click to load original from
+
{host}
+
+ )
+ }
+
+ if (originalErr) {
+ // we already tried original URL: degrade to tag
+ return (
+ <>
+
+
+ failed to load image
+
+ {originalUrl}
+ >
+ )
+ }
+
+ if (imgproxyErr && !originalUrlConsent) {
+ // respect privacy setting that external images should not be loaded automatically
+ const { host } = new URL(originalUrl)
+ return (
+
+
+
+ image proxy error
+
+
setOriginalUrlConsent(true)}
+ />
+
from {host}
+
+ )
+ }
+
+ return (
+
+ )
+}
diff --git a/components/item-full.js b/components/item-full.js
index d93493a3..2379735e 100644
--- a/components/item-full.js
+++ b/components/item-full.js
@@ -165,7 +165,7 @@ function TopLevelItem ({ item, noReply, ...props }) {
}
function ItemText ({ item }) {
- return {item.searchText || item.text}
+ return {item.searchText || item.text}
}
export default function ItemFull ({ item, bio, rank, ...props }) {
diff --git a/components/modal.js b/components/modal.js
index 4569f4a3..7ebe0613 100644
--- a/components/modal.js
+++ b/components/modal.js
@@ -1,6 +1,8 @@
-import { createContext, useCallback, useContext, useMemo, useState } from 'react'
+import { createContext, useCallback, useContext, useEffect, useMemo, useState } from 'react'
import Modal from 'react-bootstrap/Modal'
import BackArrow from '../svgs/arrow-left-line.svg'
+import { useRouter } from 'next/router'
+import ActionDropdown from './action-dropdown'
export const ShowModalContext = createContext(() => null)
@@ -37,19 +39,38 @@ export default function useModal () {
const onClose = useCallback(() => {
setModalContent(null)
setModalStack([])
- }, [])
+ modalOptions?.onClose?.()
+ }, [modalOptions?.onClose])
+
+ const router = useRouter()
+ useEffect(() => {
+ router.events.on('routeChangeStart', onClose)
+ return () => router.events.off('routeChangeStart', onClose)
+ }, [router, onClose])
const modal = useMemo(() => {
if (modalContent === null) {
return null
}
+ const className = modalOptions?.fullScreen ? 'fullscreen' : ''
return (
-
+
+ {modalOptions?.overflow &&
+
+
+ {modalOptions.overflow}
+
+
}
{modalStack.length > 0 ?
: null}
-
X
+
X
-
+
{modalContent}
diff --git a/components/text.js b/components/text.js
index 38ddbeb6..88c19b3b 100644
--- a/components/text.js
+++ b/components/text.js
@@ -8,16 +8,14 @@ import sub from '../lib/remark-sub'
import remarkDirective from 'remark-directive'
import { visit } from 'unist-util-visit'
import reactStringReplace from 'react-string-replace'
-import React, { useRef, useEffect, useState, memo } from 'react'
+import React, { useState, memo } from 'react'
import GithubSlugger from 'github-slugger'
import LinkIcon from '../svgs/link.svg'
import Thumb from '../svgs/thumb-up-fill.svg'
import { toString } from 'mdast-util-to-string'
import copy from 'clipboard-copy'
-import { IMGPROXY_URL_REGEXP, IMG_URL_REGEXP } from '../lib/url'
-import { extractUrls } from '../lib/md'
-import FileMissing from '../svgs/file-warning-line.svg'
-import { useMe } from './me'
+import { useImgUrlCache, IMG_CACHE_STATES, ZoomableImage, decodeOriginalUrl } from './image'
+import { IMGPROXY_URL_REGEXP } from '../lib/url'
function searchHighlighter () {
return (tree) => {
@@ -36,15 +34,6 @@ function searchHighlighter () {
}
}
-function decodeOriginalUrl (imgProxyUrl) {
- const parts = imgProxyUrl.split('/')
- // base64url is not a known encoding in browsers
- // so we need to replace the invalid chars
- const b64Url = parts[parts.length - 1].replace(/-/g, '+').replace(/_/, '/')
- const originalUrl = Buffer.from(b64Url, 'base64').toString('utf-8')
- return originalUrl
-}
-
function Heading ({ h, slugger, noFragments, topLevel, children, node, ...props }) {
const [copied, setCopied] = useState(false)
const [id] = useState(noFragments ? undefined : slugger.slug(toString(node).replace(/[^\w\-\s]+/gi, '')))
@@ -73,54 +62,14 @@ function Heading ({ h, slugger, noFragments, topLevel, children, node, ...props
)
}
-const CACHE_STATES = {
- IS_LOADING: 'IS_LOADING',
- IS_LOADED: 'IS_LOADED',
- IS_ERROR: 'IS_ERROR'
-}
-
// this is one of the slowest components to render
-export default memo(function Text ({ topLevel, noFragments, nofollow, fetchOnlyImgProxy, children }) {
+export default memo(function Text ({ topLevel, noFragments, nofollow, imgproxyUrls, children }) {
// all the reactStringReplace calls are to facilitate search highlighting
const slugger = new GithubSlugger()
- fetchOnlyImgProxy ??= true
const HeadingWrapper = (props) => Heading({ topLevel, slugger, noFragments, ...props })
- const imgCache = useRef({})
- const [urlCache, setUrlCache] = useState({})
-
- useEffect(() => {
- const imgRegexp = fetchOnlyImgProxy ? IMGPROXY_URL_REGEXP : IMG_URL_REGEXP
- const urls = extractUrls(children)
-
- urls.forEach((url) => {
- if (imgRegexp.test(url)) {
- setUrlCache((prev) => ({ ...prev, [url]: CACHE_STATES.IS_LOADED }))
- } else if (!fetchOnlyImgProxy) {
- const img = new window.Image()
- imgCache.current[url] = img
-
- setUrlCache((prev) => ({ ...prev, [url]: CACHE_STATES.IS_LOADING }))
-
- const callback = (state) => {
- setUrlCache((prev) => ({ ...prev, [url]: state }))
- delete imgCache.current[url]
- }
- img.onload = () => callback(CACHE_STATES.IS_LOADED)
- img.onerror = () => callback(CACHE_STATES.IS_ERROR)
- img.src = url
- }
- })
-
- return () => {
- Object.values(imgCache.current).forEach((img) => {
- img.onload = null
- img.onerror = null
- img.src = ''
- })
- }
- }, [children])
+ const imgUrlCache = useImgUrlCache(children, imgproxyUrls)
return (
@@ -159,8 +108,12 @@ export default memo(function Text ({ topLevel, noFragments, nofollow, fetchOnlyI
return <>{children}>
}
- if (urlCache[href] === CACHE_STATES.IS_LOADED) {
- return
+ if (imgUrlCache[href] === IMG_CACHE_STATES.LOADED) {
+ const url = IMGPROXY_URL_REGEXP.test(href) ? decodeOriginalUrl(href) : href
+ // if `srcSet` is undefined, it means the image was not processed by worker yet
+ // if `srcSet` is null, image was processed but this specific url was not detected as an image by the worker
+ const srcSet = imgproxyUrls ? (imgproxyUrls[url] || null) : undefined
+ return
}
// map: fix any highlighted links
@@ -183,8 +136,12 @@ export default memo(function Text ({ topLevel, noFragments, nofollow, fetchOnlyI
)
},
- img: ({ node, ...props }) => {
- return
+ img: ({ node, src, ...props }) => {
+ const url = IMGPROXY_URL_REGEXP.test(src) ? decodeOriginalUrl(src) : src
+ // if `srcSet` is undefined, it means the image was not processed by worker yet
+ // if `srcSet` is null, image was processed but this specific url was not detected as an image by the worker
+ const srcSet = imgproxyUrls ? (imgproxyUrls[url] || null) : undefined
+ return
}
}}
remarkPlugins={[gfm, mention, sub, remarkDirective, searchHighlighter]}
@@ -194,76 +151,3 @@ export default memo(function Text ({ topLevel, noFragments, nofollow, fetchOnlyI
)
})
-
-function ClickToLoad ({ children }) {
- const [clicked, setClicked] = useState(false)
- return clicked ? children : setClicked(true)}>click to load image
-}
-
-export function ZoomableImage ({ src, topLevel, useClickToLoad, ...props }) {
- const me = useMe()
- const [err, setErr] = useState()
- const [imgSrc, setImgSrc] = useState(src)
- const [isImgProxy, setIsImgProxy] = useState(IMGPROXY_URL_REGEXP.test(src))
- const defaultMediaStyle = {
- maxHeight: topLevel ? '75vh' : '25vh',
- cursor: 'zoom-in'
- }
- useClickToLoad ??= true
-
- // if image changes we need to update state
- const [mediaStyle, setMediaStyle] = useState(defaultMediaStyle)
- useEffect(() => {
- setMediaStyle(defaultMediaStyle)
- setErr(null)
- }, [src])
-
- if (!src) return null
- if (err) {
- if (!isImgProxy) {
- return (
-
-
- image error
-
- )
- }
- try {
- const originalUrl = decodeOriginalUrl(src)
- setImgSrc(originalUrl)
- setErr(null)
- } catch (err) {
- console.error(err)
- setErr(err)
- }
- // always set to false since imgproxy returned error
- setIsImgProxy(false)
- }
-
- const img = (
- {
- if (mediaStyle.cursor === 'zoom-in') {
- setMediaStyle({
- width: '100%',
- cursor: 'zoom-out'
- })
- } else {
- setMediaStyle(defaultMediaStyle)
- }
- }}
- onError={() => setErr(true)}
- {...props}
- />
- )
-
- return (
- (!me || !me.clickToLoadImg || isImgProxy || !useClickToLoad)
- ? img
- : {img}
-
- )
-}
diff --git a/fragments/comments.js b/fragments/comments.js
index b33fe9c6..8e00f89e 100644
--- a/fragments/comments.js
+++ b/fragments/comments.js
@@ -30,6 +30,7 @@ export const COMMENT_FIELDS = gql`
mine
otsHash
ncomments
+ imgproxyUrls
}
`
diff --git a/fragments/items.js b/fragments/items.js
index 5345ca10..bb5d4f02 100644
--- a/fragments/items.js
+++ b/fragments/items.js
@@ -45,6 +45,7 @@ export const ITEM_FIELDS = gql`
status
uploadId
mine
+ imgproxyUrls
}`
export const ITEM_FULL_FIELDS = gql`
diff --git a/prisma/migrations/20230920192620_item_imgproxy_urls/migration.sql b/prisma/migrations/20230920192620_item_imgproxy_urls/migration.sql
new file mode 100644
index 00000000..38ca2f36
--- /dev/null
+++ b/prisma/migrations/20230920192620_item_imgproxy_urls/migration.sql
@@ -0,0 +1,175 @@
+-- AlterTable
+ALTER TABLE "Item" ADD COLUMN "imgproxyUrls" JSONB;
+
+-- schedule imgproxy job
+CREATE OR REPLACE FUNCTION create_item(
+ jitem JSONB, forward JSONB, poll_options JSONB, spam_within INTERVAL)
+RETURNS "Item"
+LANGUAGE plpgsql
+AS $$
+DECLARE
+ user_msats BIGINT;
+ cost_msats BIGINT;
+ freebie BOOLEAN;
+ item "Item";
+ med_votes FLOAT;
+ select_clause TEXT;
+BEGIN
+ PERFORM ASSERT_SERIALIZED();
+
+ -- access fields with appropriate types
+ item := jsonb_populate_record(NULL::"Item", jitem);
+
+ SELECT msats INTO user_msats FROM users WHERE id = item."userId";
+
+ IF item."maxBid" IS NOT NULL THEN
+ cost_msats := 1000000;
+ ELSE
+ cost_msats := 1000 * POWER(10, item_spam(item."parentId", item."userId", spam_within));
+ END IF;
+ -- it's only a freebie if it's a 1 sat cost, they have < 1 sat, and boost = 0
+ freebie := (cost_msats <= 1000) AND (user_msats < 1000) AND (item.boost = 0);
+
+ IF NOT freebie AND cost_msats > user_msats THEN
+ RAISE EXCEPTION 'SN_INSUFFICIENT_FUNDS';
+ END IF;
+
+ -- get this user's median item score
+ SELECT COALESCE(
+ percentile_cont(0.5) WITHIN GROUP(
+ ORDER BY "weightedVotes" - "weightedDownVotes"), 0)
+ INTO med_votes FROM "Item" WHERE "userId" = item."userId";
+
+ -- if their median votes are positive, start at 0
+ -- if the median votes are negative, start their post with that many down votes
+ -- basically: if their median post is bad, presume this post is too
+ -- addendum: if they're an anon poster, always start at 0
+ IF med_votes >= 0 OR item."userId" = 27 THEN
+ med_votes := 0;
+ ELSE
+ med_votes := ABS(med_votes);
+ END IF;
+
+ -- there's no great way to set default column values when using json_populate_record
+ -- so we need to only select fields with non-null values that way when func input
+ -- does not include a value, the default value is used instead of null
+ SELECT string_agg(quote_ident(key), ',') INTO select_clause
+ FROM jsonb_object_keys(jsonb_strip_nulls(jitem)) k(key);
+ -- insert the item
+ EXECUTE format($fmt$
+ INSERT INTO "Item" (%s, "weightedDownVotes")
+ SELECT %1$s, %L
+ FROM jsonb_populate_record(NULL::"Item", %L) RETURNING *
+ $fmt$, select_clause, med_votes, jitem) INTO item;
+
+ INSERT INTO "ItemForward" ("itemId", "userId", "pct")
+ SELECT item.id, "userId", "pct" FROM jsonb_populate_recordset(NULL::"ItemForward", forward);
+
+ -- Automatically subscribe forward recipients to the new post
+ INSERT INTO "ThreadSubscription" ("itemId", "userId")
+ SELECT item.id, "userId" FROM jsonb_populate_recordset(NULL::"ItemForward", forward);
+
+ INSERT INTO "PollOption" ("itemId", "option")
+ SELECT item.id, "option" FROM jsonb_array_elements_text(poll_options) o("option");
+
+ IF NOT freebie THEN
+ UPDATE users SET msats = msats - cost_msats WHERE id = item."userId";
+
+ INSERT INTO "ItemAct" (msats, "itemId", "userId", act)
+ VALUES (cost_msats, item.id, item."userId", 'FEE');
+ END IF;
+
+ -- if this item has boost
+ IF item.boost > 0 THEN
+ PERFORM item_act(item.id, item."userId", 'BOOST', item.boost);
+ END IF;
+
+ -- if this is a job
+ IF item."maxBid" IS NOT NULL THEN
+ PERFORM run_auction(item.id);
+ END IF;
+
+ -- if this is a bio
+ IF item.bio THEN
+ UPDATE users SET "bioId" = item.id WHERE id = item."userId";
+ END IF;
+
+ -- schedule imgproxy job
+ INSERT INTO pgboss.job (name, data, retrylimit, retrybackoff, startafter)
+ VALUES ('imgproxy', jsonb_build_object('id', item.id), 21, true, now() + interval '5 seconds');
+
+ RETURN item;
+END;
+$$;
+
+-- schedule imgproxy job
+CREATE OR REPLACE FUNCTION update_item(
+ jitem JSONB, forward JSONB, poll_options JSONB)
+RETURNS "Item"
+LANGUAGE plpgsql
+AS $$
+DECLARE
+ user_msats INTEGER;
+ item "Item";
+ select_clause TEXT;
+BEGIN
+ PERFORM ASSERT_SERIALIZED();
+
+ item := jsonb_populate_record(NULL::"Item", jitem);
+
+ IF item.boost > 0 THEN
+ UPDATE "Item" SET boost = boost + item.boost WHERE id = item.id;
+ PERFORM item_act(item.id, item."userId", 'BOOST', item.boost);
+ END IF;
+
+ IF item.status IS NOT NULL THEN
+ UPDATE "Item" SET "statusUpdatedAt" = now_utc()
+ WHERE id = item.id AND status <> item.status;
+ END IF;
+
+ SELECT string_agg(quote_ident(key), ',') INTO select_clause
+ FROM jsonb_object_keys(jsonb_strip_nulls(jitem)) k(key)
+ WHERE key <> 'boost';
+
+ EXECUTE format($fmt$
+ UPDATE "Item" SET (%s) = (
+ SELECT %1$s
+ FROM jsonb_populate_record(NULL::"Item", %L)
+ ) WHERE id = %L RETURNING *
+ $fmt$, select_clause, jitem, item.id) INTO item;
+
+ -- Delete any old thread subs if the user is no longer a fwd recipient
+ DELETE FROM "ThreadSubscription"
+ WHERE "itemId" = item.id
+ -- they aren't in the new forward list
+ AND NOT EXISTS (SELECT 1 FROM jsonb_populate_recordset(NULL::"ItemForward", forward) as nf WHERE "ThreadSubscription"."userId" = nf."userId")
+ -- and they are in the old forward list
+ AND EXISTS (SELECT 1 FROM "ItemForward" WHERE "ItemForward"."itemId" = item.id AND "ItemForward"."userId" = "ThreadSubscription"."userId" );
+
+ -- Automatically subscribe any new forward recipients to the post
+ INSERT INTO "ThreadSubscription" ("itemId", "userId")
+ SELECT item.id, "userId" FROM jsonb_populate_recordset(NULL::"ItemForward", forward)
+ EXCEPT
+ SELECT item.id, "userId" FROM "ItemForward" WHERE "itemId" = item.id;
+
+ -- Delete all old forward entries, to recreate in next command
+ DELETE FROM "ItemForward" WHERE "itemId" = item.id;
+
+ INSERT INTO "ItemForward" ("itemId", "userId", "pct")
+ SELECT item.id, "userId", "pct" FROM jsonb_populate_recordset(NULL::"ItemForward", forward);
+
+ INSERT INTO "PollOption" ("itemId", "option")
+ SELECT item.id, "option" FROM jsonb_array_elements_text(poll_options) o("option");
+
+ -- if this is a job
+ IF item."maxBid" IS NOT NULL THEN
+ PERFORM run_auction(item.id);
+ END IF;
+
+ -- schedule imgproxy job
+ INSERT INTO pgboss.job (name, data, retrylimit, retrybackoff, startafter)
+ VALUES ('imgproxy', jsonb_build_object('id', item.id), 21, true, now() + interval '5 seconds');
+
+ RETURN item;
+END;
+$$;
\ No newline at end of file
diff --git a/prisma/schema.prisma b/prisma/schema.prisma
index 3f9f9d0e..59970053 100644
--- a/prisma/schema.prisma
+++ b/prisma/schema.prisma
@@ -264,6 +264,7 @@ model Item {
deletedAt DateTime?
otsFile Bytes?
otsHash String?
+ imgproxyUrls Json?
bounty Int?
rootId Int?
bountyPaidTo Int[]
diff --git a/public/placeholder_click_to_load.png b/public/placeholder_click_to_load.png
new file mode 100644
index 0000000000000000000000000000000000000000..1a297a9aead4d26ae018405be6cc213cadd14192
GIT binary patch
literal 9254
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