Put canReply state on post viewer state instead of thread viewer state ()

* switch canReply from thread to post view

* tweaks & fix up tests

* update snaps

* fix more snaps

* hydrate feed items for getPosts & searchPosts

* fix another snapshot

* getPosts test

* canReply -> blockedByGate & DRY up code

* blockedByGate -> excludedByGate

* replyDisabled
This commit is contained in:
Daniel Holmgren 2023-11-27 20:14:20 -06:00 committed by GitHub
parent 95d33f7b11
commit 7edad62c12
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
31 changed files with 262 additions and 323 deletions

@ -38,7 +38,8 @@
"type": "object",
"properties": {
"repost": { "type": "string", "format": "at-uri" },
"like": { "type": "string", "format": "at-uri" }
"like": { "type": "string", "format": "at-uri" },
"replyDisabled": { "type": "boolean" }
}
},
"feedViewPost": {
@ -87,8 +88,7 @@
"type": "union",
"refs": ["#threadViewPost", "#notFoundPost", "#blockedPost"]
}
},
"viewer": { "type": "ref", "ref": "#viewerThreadState" }
}
}
},
"notFoundPost": {
@ -116,12 +116,6 @@
"viewer": { "type": "ref", "ref": "app.bsky.actor.defs#viewerState" }
}
},
"viewerThreadState": {
"type": "object",
"properties": {
"canReply": { "type": "boolean" }
}
},
"generatorView": {
"type": "object",
"required": ["uri", "cid", "did", "creator", "displayName", "indexedAt"],

@ -4892,6 +4892,9 @@ export const schemaDict = {
type: 'string',
format: 'at-uri',
},
replyDisabled: {
type: 'boolean',
},
},
},
feedViewPost: {
@ -4975,10 +4978,6 @@ export const schemaDict = {
],
},
},
viewer: {
type: 'ref',
ref: 'lex:app.bsky.feed.defs#viewerThreadState',
},
},
},
notFoundPost: {
@ -5027,14 +5026,6 @@ export const schemaDict = {
},
},
},
viewerThreadState: {
type: 'object',
properties: {
canReply: {
type: 'boolean',
},
},
},
generatorView: {
type: 'object',
required: ['uri', 'cid', 'did', 'creator', 'displayName', 'indexedAt'],

@ -48,6 +48,7 @@ export function validatePostView(v: unknown): ValidationResult {
export interface ViewerState {
repost?: string
like?: string
replyDisabled?: boolean
[k: string]: unknown
}
@ -137,7 +138,6 @@ export interface ThreadViewPost {
| BlockedPost
| { $type: string; [k: string]: unknown }
)[]
viewer?: ViewerThreadState
[k: string]: unknown
}
@ -208,23 +208,6 @@ export function validateBlockedAuthor(v: unknown): ValidationResult {
return lexicons.validate('app.bsky.feed.defs#blockedAuthor', v)
}
export interface ViewerThreadState {
canReply?: boolean
[k: string]: unknown
}
export function isViewerThreadState(v: unknown): v is ViewerThreadState {
return (
isObj(v) &&
hasProp(v, '$type') &&
v.$type === 'app.bsky.feed.defs#viewerThreadState'
)
}
export function validateViewerThreadState(v: unknown): ValidationResult {
return lexicons.validate('app.bsky.feed.defs#viewerThreadState', v)
}
export interface GeneratorView {
uri: string
cid: string

@ -107,9 +107,7 @@ const noPostBlocks = (state: HydrationState) => {
const presentation = (state: HydrationState, ctx: Context) => {
const { feedService } = ctx
const { feedItems, cursor, params } = state
const feed = feedService.views.formatFeed(feedItems, state, {
viewer: params.viewer,
})
const feed = feedService.views.formatFeed(feedItems, state, params.viewer)
return { feed, cursor }
}

@ -147,9 +147,7 @@ const noBlocksOrMutedReposts = (state: HydrationState) => {
const presentation = (state: HydrationState, ctx: Context) => {
const { feedService } = ctx
const { feedItems, cursor, params } = state
const feed = feedService.views.formatFeed(feedItems, state, {
viewer: params.viewer,
})
const feed = feedService.views.formatFeed(feedItems, state, params.viewer)
return { feed, cursor }
}

@ -113,9 +113,7 @@ const noBlocksOrMutes = (state: HydrationState) => {
const presentation = (state: HydrationState, ctx: Context) => {
const { feedService } = ctx
const { feedItems, cursor, passthrough, params } = state
const feed = feedService.views.formatFeed(feedItems, state, {
viewer: params.viewer,
})
const feed = feedService.views.formatFeed(feedItems, state, params.viewer)
return {
feed,
cursor,

@ -105,9 +105,7 @@ const noBlocksOrMutes = (state: HydrationState) => {
const presentation = (state: HydrationState, ctx: Context) => {
const { feedService } = ctx
const { feedItems, cursor, params } = state
const feed = feedService.views.formatFeed(feedItems, state, {
viewer: params.viewer,
})
const feed = feedService.views.formatFeed(feedItems, state, params.viewer)
return { feed, cursor }
}

@ -6,27 +6,21 @@ import {
NotFoundPost,
ThreadViewPost,
isNotFoundPost,
isThreadViewPost,
} from '../../../../lexicon/types/app/bsky/feed/defs'
import { Record as PostRecord } from '../../../../lexicon/types/app/bsky/feed/post'
import { Record as ThreadgateRecord } from '../../../../lexicon/types/app/bsky/feed/threadgate'
import { QueryParams } from '../../../../lexicon/types/app/bsky/feed/getPostThread'
import AppContext from '../../../../context'
import {
FeedService,
FeedRow,
FeedHydrationState,
PostInfo,
} from '../../../../services/feed'
import {
getAncestorsAndSelfQb,
getDescendentsQb,
} from '../../../../services/util/post'
import { Database } from '../../../../db'
import DatabaseSchema from '../../../../db/database-schema'
import { setRepoRev } from '../../../util'
import { ActorInfoMap, ActorService } from '../../../../services/actor'
import { violatesThreadGate } from '../../../../services/feed/util'
import { createPipeline, noRules } from '../../../../pipeline'
export default function (server: Server, ctx: AppContext) {
@ -80,21 +74,7 @@ const hydration = async (state: SkeletonState, ctx: Context) => {
} = state
const relevant = getRelevantIds(threadData)
const hydrated = await feedService.feedHydration({ ...relevant, viewer })
// check root reply interaction rules
const anchorPostUri = threadData.post.postUri
const rootUri = threadData.post.replyRoot || anchorPostUri
const anchor = hydrated.posts[anchorPostUri]
const root = hydrated.posts[rootUri]
const gate = hydrated.threadgates[rootUri]?.record
const viewerCanReply = await checkViewerCanReply(
ctx.db.db,
anchor ?? null,
viewer,
new AtUri(rootUri).host,
(root?.record ?? null) as PostRecord | null,
gate ?? null,
)
return { ...state, ...hydrated, viewerCanReply }
return { ...state, ...hydrated }
}
const presentation = (state: HydrationState, ctx: Context) => {
@ -103,16 +83,19 @@ const presentation = (state: HydrationState, ctx: Context) => {
const actors = actorService.views.profileBasicPresentation(
Object.keys(profiles),
state,
{ viewer: params.viewer },
params.viewer,
)
const thread = composeThread(
state.threadData,
actors,
state,
ctx,
params.viewer,
)
const thread = composeThread(state.threadData, actors, state, ctx)
if (isNotFoundPost(thread)) {
// @TODO technically this could be returned as a NotFoundPost based on lexicon
throw new InvalidRequestError(`Post not found: ${params.uri}`, 'NotFound')
}
if (isThreadViewPost(thread) && params.viewer) {
thread.viewer = { canReply: state.viewerCanReply }
}
return { thread }
}
@ -121,6 +104,7 @@ const composeThread = (
actors: ActorInfoMap,
state: HydrationState,
ctx: Context,
viewer: string | null,
) => {
const { feedService } = ctx
const { posts, threadgates, embeds, blocks, labels, lists } = state
@ -133,6 +117,7 @@ const composeThread = (
embeds,
labels,
lists,
viewer,
)
// replies that are invalid due to reply-gating:
@ -179,14 +164,14 @@ const composeThread = (
notFound: true,
}
} else {
parent = composeThread(threadData.parent, actors, state, ctx)
parent = composeThread(threadData.parent, actors, state, ctx, viewer)
}
}
let replies: (ThreadViewPost | NotFoundPost | BlockedPost)[] | undefined
if (threadData.replies && !badReply) {
replies = threadData.replies.flatMap((reply) => {
const thread = composeThread(reply, actors, state, ctx)
const thread = composeThread(reply, actors, state, ctx, viewer)
// e.g. don't bother including #postNotFound reply placeholders for takedowns. either way matches api contract.
const skip = []
return isNotFoundPost(thread) ? skip : thread
@ -223,6 +208,7 @@ const getRelevantIds = (
if (thread.post.replyRoot) {
// ensure root is included for checking interactions
uris.add(thread.post.replyRoot)
dids.add(new AtUri(thread.post.replyRoot).hostname)
}
return { dids, uris }
}
@ -317,28 +303,6 @@ const getChildrenData = (
}))
}
const checkViewerCanReply = async (
db: DatabaseSchema,
anchor: PostInfo | null,
viewer: string | null,
owner: string,
root: PostRecord | null,
threadgate: ThreadgateRecord | null,
) => {
if (!viewer) return false
// @TODO re-enable invalidReplyRoot check
// if (anchor?.invalidReplyRoot || anchor?.violatesThreadGate) return false
if (anchor?.violatesThreadGate) return false
const viewerViolatesThreadGate = await violatesThreadGate(
db,
viewer,
owner,
root,
threadgate,
)
return !viewerViolatesThreadGate
}
class ParentNotFoundError extends Error {
constructor(public uri: string) {
super(`Parent not found: ${uri}`)
@ -364,7 +328,4 @@ type SkeletonState = {
threadData: PostThread
}
type HydrationState = SkeletonState &
FeedHydrationState & {
viewerCanReply: boolean
}
type HydrationState = SkeletonState & FeedHydrationState

@ -1,10 +1,13 @@
import { dedupeStrs } from '@atproto/common'
import { AtUri } from '@atproto/syntax'
import { Server } from '../../../../lexicon'
import { QueryParams } from '../../../../lexicon/types/app/bsky/feed/getPosts'
import AppContext from '../../../../context'
import { Database } from '../../../../db'
import { FeedHydrationState, FeedService } from '../../../../services/feed'
import {
FeedHydrationState,
FeedRow,
FeedService,
} from '../../../../services/feed'
import { createPipeline } from '../../../../pipeline'
import { ActorService } from '../../../../services/actor'
@ -31,18 +34,18 @@ export default function (server: Server, ctx: AppContext) {
})
}
const skeleton = async (params: Params) => {
return { params, postUris: dedupeStrs(params.uris) }
const skeleton = async (params: Params, ctx: Context) => {
const deduped = dedupeStrs(params.uris)
const feedItems = await ctx.feedService.postUrisToFeedItems(deduped)
return { params, feedItems }
}
const hydration = async (state: SkeletonState, ctx: Context) => {
const { feedService } = ctx
const { params, postUris } = state
const uris = new Set<string>(postUris)
const dids = new Set<string>(postUris.map((uri) => new AtUri(uri).hostname))
const { params, feedItems } = state
const refs = feedService.feedItemRefs(feedItems)
const hydrated = await feedService.feedHydration({
uris,
dids,
...refs,
viewer: params.viewer,
})
return { ...state, ...hydrated }
@ -50,32 +53,32 @@ const hydration = async (state: SkeletonState, ctx: Context) => {
const noBlocks = (state: HydrationState) => {
const { viewer } = state.params
state.postUris = state.postUris.filter((uri) => {
const post = state.posts[uri]
if (!viewer || !post) return true
return !state.bam.block([viewer, post.creator])
state.feedItems = state.feedItems.filter((item) => {
if (!viewer) return true
return !state.bam.block([viewer, item.postAuthorDid])
})
return state
}
const presentation = (state: HydrationState, ctx: Context) => {
const { feedService, actorService } = ctx
const { postUris, profiles, params } = state
const { feedItems, profiles, params } = state
const SKIP = []
const actors = actorService.views.profileBasicPresentation(
Object.keys(profiles),
state,
{ viewer: params.viewer },
params.viewer,
)
const postViews = postUris.flatMap((uri) => {
const postViews = feedItems.flatMap((item) => {
const postView = feedService.views.formatPostView(
uri,
item.postUri,
actors,
state.posts,
state.threadgates,
state.embeds,
state.labels,
state.lists,
params.viewer,
)
return postView ?? SKIP
})
@ -92,7 +95,7 @@ type Params = QueryParams & { viewer: string | null }
type SkeletonState = {
params: Params
postUris: string[]
feedItems: FeedRow[]
}
type HydrationState = SkeletonState & FeedHydrationState

@ -146,9 +146,7 @@ const noBlocksOrMutes = (state: HydrationState): HydrationState => {
const presentation = (state: HydrationState, ctx: Context) => {
const { feedService } = ctx
const { feedItems, cursor, params } = state
const feed = feedService.views.formatFeed(feedItems, state, {
viewer: params.viewer,
})
const feed = feedService.views.formatFeed(feedItems, state, params.viewer)
return { feed, cursor }
}

@ -2,11 +2,14 @@ import AppContext from '../../../../context'
import { Server } from '../../../../lexicon'
import { InvalidRequestError } from '@atproto/xrpc-server'
import AtpAgent from '@atproto/api'
import { AtUri } from '@atproto/syntax'
import { mapDefined } from '@atproto/common'
import { QueryParams } from '../../../../lexicon/types/app/bsky/feed/searchPosts'
import { Database } from '../../../../db'
import { FeedHydrationState, FeedService } from '../../../../services/feed'
import {
FeedHydrationState,
FeedRow,
FeedService,
} from '../../../../services/feed'
import { ActorService } from '../../../../services/actor'
import { createPipeline } from '../../../../pipeline'
@ -51,9 +54,11 @@ const skeleton = async (
cursor: params.cursor,
limit: params.limit,
})
const postUris = res.data.posts.map((a) => a.uri)
const feedItems = await ctx.feedService.postUrisToFeedItems(postUris)
return {
params,
postUris: res.data.posts.map((a) => a.uri),
feedItems,
cursor: res.data.cursor,
hitsTotal: res.data.hitsTotal,
}
@ -64,12 +69,10 @@ const hydration = async (
ctx: Context,
): Promise<HydrationState> => {
const { feedService } = ctx
const { params, postUris } = state
const uris = new Set<string>(postUris)
const dids = new Set<string>(postUris.map((uri) => new AtUri(uri).hostname))
const { params, feedItems } = state
const refs = feedService.feedItemRefs(feedItems)
const hydrated = await feedService.feedHydration({
uris,
dids,
...refs,
viewer: params.viewer,
})
return { ...state, ...hydrated }
@ -77,32 +80,32 @@ const hydration = async (
const noBlocks = (state: HydrationState): HydrationState => {
const { viewer } = state.params
state.postUris = state.postUris.filter((uri) => {
const post = state.posts[uri]
if (!viewer || !post) return true
return !state.bam.block([viewer, post.creator])
state.feedItems = state.feedItems.filter((item) => {
if (!viewer) return true
return !state.bam.block([viewer, item.postAuthorDid])
})
return state
}
const presentation = (state: HydrationState, ctx: Context) => {
const { feedService, actorService } = ctx
const { postUris, profiles, params } = state
const { feedItems, profiles, params } = state
const actors = actorService.views.profileBasicPresentation(
Object.keys(profiles),
state,
{ viewer: params.viewer },
params.viewer,
)
const postViews = mapDefined(postUris, (uri) =>
const postViews = mapDefined(feedItems, (item) =>
feedService.views.formatPostView(
uri,
item.postUri,
actors,
state.posts,
state.threadgates,
state.embeds,
state.labels,
state.lists,
params.viewer,
),
)
return { posts: postViews, cursor: state.cursor, hitsTotal: state.hitsTotal }
@ -119,7 +122,7 @@ type Params = QueryParams & { viewer: string | null }
type SkeletonState = {
params: Params
postUris: string[]
feedItems: FeedRow[]
hitsTotal?: number
cursor?: string
}

@ -91,7 +91,7 @@ const presentation = (state: HydrationState, ctx: Context) => {
const actors = actorService.views.profilePresentation(
Object.keys(profileState.profiles),
profileState,
{ viewer: params.viewer },
params.viewer,
)
const creator = actors[list.creator]
if (!creator) {

@ -87,7 +87,7 @@ const presentation = (state: HydrationState, ctx: Context) => {
const actors = actorService.views.profilePresentation(
Object.keys(profileState.profiles),
profileState,
{ viewer: params.viewer },
params.viewer,
)
const lists = listInfos.map((list) =>
graphService.formatListView(list, actors),

@ -4892,6 +4892,9 @@ export const schemaDict = {
type: 'string',
format: 'at-uri',
},
replyDisabled: {
type: 'boolean',
},
},
},
feedViewPost: {
@ -4975,10 +4978,6 @@ export const schemaDict = {
],
},
},
viewer: {
type: 'ref',
ref: 'lex:app.bsky.feed.defs#viewerThreadState',
},
},
},
notFoundPost: {
@ -5027,14 +5026,6 @@ export const schemaDict = {
},
},
},
viewerThreadState: {
type: 'object',
properties: {
canReply: {
type: 'boolean',
},
},
},
generatorView: {
type: 'object',
required: ['uri', 'cid', 'did', 'creator', 'displayName', 'indexedAt'],

@ -48,6 +48,7 @@ export function validatePostView(v: unknown): ValidationResult {
export interface ViewerState {
repost?: string
like?: string
replyDisabled?: boolean
[k: string]: unknown
}
@ -137,7 +138,6 @@ export interface ThreadViewPost {
| BlockedPost
| { $type: string; [k: string]: unknown }
)[]
viewer?: ViewerThreadState
[k: string]: unknown
}
@ -208,23 +208,6 @@ export function validateBlockedAuthor(v: unknown): ValidationResult {
return lexicons.validate('app.bsky.feed.defs#blockedAuthor', v)
}
export interface ViewerThreadState {
canReply?: boolean
[k: string]: unknown
}
export function isViewerThreadState(v: unknown): v is ViewerThreadState {
return (
isObj(v) &&
hasProp(v, '$type') &&
v.$type === 'app.bsky.feed.defs#viewerThreadState'
)
}
export function validateViewerThreadState(v: unknown): ValidationResult {
return lexicons.validate('app.bsky.feed.defs#viewerThreadState', v)
}
export interface GeneratorView {
uri: string
cid: string

@ -45,10 +45,7 @@ export class ActorViews {
viewer,
...opts,
})
return this.profilePresentation(dids, hydrated, {
viewer,
...opts,
})
return this.profilePresentation(dids, hydrated, viewer)
}
async profilesBasic(
@ -62,10 +59,7 @@ export class ActorViews {
viewer,
includeSoftDeleted: opts?.includeSoftDeleted,
})
return this.profileBasicPresentation(dids, hydrated, {
viewer,
omitLabels: opts?.omitLabels,
})
return this.profileBasicPresentation(dids, hydrated, viewer, opts)
}
async profilesList(
@ -293,11 +287,8 @@ export class ActorViews {
labels: Labels
bam: BlockAndMuteState
},
opts?: {
viewer?: string | null
},
viewer: string | null,
): ProfileViewMap {
const { viewer } = opts ?? {}
const { profiles, lists, labels, bam } = state
return dids.reduce((acc, did) => {
const prof = profiles[did]
@ -357,12 +348,12 @@ export class ActorViews {
profileBasicPresentation(
dids: string[],
state: ProfileHydrationState,
viewer: string | null,
opts?: {
viewer?: string | null
omitLabels?: boolean
},
): ProfileViewMap {
const result = this.profilePresentation(dids, state, opts)
const result = this.profilePresentation(dids, state, viewer)
return Object.values(result).reduce((acc, prof) => {
const profileBasic = {
did: prof.did,

@ -44,6 +44,7 @@ import {
import { FeedViews } from './views'
import { LabelCache } from '../../label-cache'
import { threadgateToPostUri, postToThreadgateUri } from './util'
import { mapDefined } from '@atproto/common'
export * from './types'
@ -205,6 +206,11 @@ export class FeedService {
}, {} as Record<string, FeedRow>)
}
async postUrisToFeedItems(uris: string[]): Promise<FeedRow[]> {
const feedItems = await this.getFeedItems(uris)
return mapDefined(uris, (uri) => feedItems[uri])
}
feedItemRefs(items: FeedRow[]) {
const actorDids = new Set<string>()
const postUris = new Set<string>()
@ -399,7 +405,7 @@ export class FeedService {
const actorInfos = this.services.actor.views.profileBasicPresentation(
[...nestedDids],
feedState,
{ viewer },
viewer,
)
const recordEmbedViews: RecordEmbedViewRecordMap = {}
for (const uri of nestedUris) {
@ -423,6 +429,7 @@ export class FeedService {
feedState.embeds,
feedState.labels,
feedState.lists,
viewer,
)
recordEmbedViews[uri] = this.views.getRecordEmbedView(
uri,

@ -36,43 +36,72 @@ export const invalidReplyRoot = (
return parent.record.reply?.root.uri !== replyRoot
}
export const violatesThreadGate = async (
db: DatabaseSchema,
did: string,
owner: string,
root: PostRecord | null,
gate: GateRecord | null,
) => {
if (did === owner) return false
if (!gate?.allow) return false
type ParsedThreadGate = {
canReply?: boolean
allowMentions?: boolean
allowFollowing?: boolean
allowListUris?: string[]
}
const allowMentions = gate.allow.find(isMentionRule)
const allowFollowing = gate.allow.find(isFollowingRule)
export const parseThreadGate = (
replierDid: string,
ownerDid: string,
rootPost: PostRecord | null,
gate: GateRecord | null,
): ParsedThreadGate => {
if (replierDid === ownerDid) {
return { canReply: true }
}
// if gate.allow is unset then *any* reply is allowed, if it is an empty array then *no* reply is allowed
if (!gate || !gate.allow) {
return { canReply: true }
}
const allowMentions = !!gate.allow.find(isMentionRule)
const allowFollowing = !!gate.allow.find(isFollowingRule)
const allowListUris = gate.allow?.filter(isListRule).map((item) => item.list)
// check mentions first since it's quick and synchronous
if (allowMentions) {
const isMentioned = root?.facets?.some((facet) => {
return facet.features.some((item) => isMention(item) && item.did === did)
const isMentioned = rootPost?.facets?.some((facet) => {
return facet.features.some(
(item) => isMention(item) && item.did === replierDid,
)
})
if (isMentioned) {
return false
return { canReply: true, allowMentions, allowFollowing, allowListUris }
}
}
return { allowMentions, allowFollowing, allowListUris }
}
// check follows and list containment
if (!allowFollowing && !allowListUris.length) {
export const violatesThreadGate = async (
db: DatabaseSchema,
replierDid: string,
ownerDid: string,
rootPost: PostRecord | null,
gate: GateRecord | null,
) => {
const {
canReply,
allowFollowing,
allowListUris = [],
} = parseThreadGate(replierDid, ownerDid, rootPost, gate)
if (canReply) {
return false
}
if (!allowFollowing && !allowListUris?.length) {
return true
}
const { ref } = db.dynamic
const nullResult = sql<null>`${null}`
const check = await db
.selectFrom(valuesList([did]).as(sql`subject (did)`))
.selectFrom(valuesList([replierDid]).as(sql`subject (did)`))
.select([
allowFollowing
? db
.selectFrom('follow')
.where('creator', '=', owner)
.where('creator', '=', ownerDid)
.whereRef('subjectDid', '=', ref('subject.did'))
.select('creator')
.as('isFollowed')
@ -91,8 +120,7 @@ export const violatesThreadGate = async (
if (allowFollowing && check?.isFollowed) {
return false
}
if (allowListUris.length && check?.isInList) {
} else if (allowListUris.length && check?.isInList) {
return false
}

@ -5,7 +5,6 @@ import {
GeneratorView,
PostView,
} from '../../lexicon/types/app/bsky/feed/defs'
import { isListRule } from '../../lexicon/types/app/bsky/feed/threadgate'
import {
Main as EmbedImages,
isMain as isEmbedImages,
@ -22,6 +21,8 @@ import {
ViewNotFound,
ViewRecord,
} from '../../lexicon/types/app/bsky/embed/record'
import { Record as PostRecord } from '../../lexicon/types/app/bsky/feed/post'
import { isListRule } from '../../lexicon/types/app/bsky/feed/threadgate'
import {
PostEmbedViews,
FeedGenInfo,
@ -39,6 +40,8 @@ import { ImageUriBuilder } from '../../image/uri'
import { LabelCache } from '../../label-cache'
import { ActorInfoMap, ActorService } from '../actor'
import { ListInfoMap, GraphService } from '../graph'
import { AtUri } from '@atproto/syntax'
import { parseThreadGate } from './util'
export class FeedViews {
constructor(
@ -91,8 +94,8 @@ export class FeedViews {
formatFeed(
items: FeedRow[],
state: FeedHydrationState,
viewer: string | null,
opts?: {
viewer?: string | null
usePostViewUnion?: boolean
},
): FeedViewPost[] {
@ -101,7 +104,7 @@ export class FeedViews {
const actors = this.services.actor.views.profileBasicPresentation(
Object.keys(profiles),
state,
opts,
viewer,
)
const feed: FeedViewPost[] = []
for (const item of items) {
@ -114,6 +117,7 @@ export class FeedViews {
embeds,
labels,
lists,
viewer,
)
// skip over not found & blocked posts
if (!post || blocks[post.uri]?.reply) {
@ -149,6 +153,7 @@ export class FeedViews {
labels,
lists,
blocks,
viewer,
opts,
)
const replyRoot = this.formatMaybePostView(
@ -160,6 +165,7 @@ export class FeedViews {
labels,
lists,
blocks,
viewer,
opts,
)
if (replyRoot && replyParent) {
@ -182,6 +188,7 @@ export class FeedViews {
embeds: PostEmbedViews,
labels: Labels,
lists: ListInfoMap,
viewer: string | null,
): PostView | undefined {
const post = posts[uri]
const gate = threadgates[uri]
@ -207,6 +214,14 @@ export class FeedViews {
? {
repost: post.requesterRepost ?? undefined,
like: post.requesterLike ?? undefined,
replyDisabled: this.userReplyDisabled(
uri,
actors,
posts,
threadgates,
lists,
viewer,
),
}
: undefined,
labels: [...postLabels, ...postSelfLabels],
@ -217,6 +232,50 @@ export class FeedViews {
}
}
userReplyDisabled(
uri: string,
actors: ActorInfoMap,
posts: PostInfoMap,
threadgates: ThreadgateInfoMap,
lists: ListInfoMap,
viewer: string | null,
): boolean | undefined {
if (viewer === null) {
return undefined
} else if (posts[uri]?.violatesThreadGate) {
return true
}
const rootUriStr: string =
posts[uri]?.record?.['reply']?.['root']?.['uri'] ?? uri
const gate = threadgates[rootUriStr]?.record
if (!gate) {
return undefined
}
const rootPost = posts[rootUriStr]?.record as PostRecord | undefined
const ownerDid = new AtUri(rootUriStr).hostname
const {
canReply,
allowFollowing,
allowListUris = [],
} = parseThreadGate(viewer, ownerDid, rootPost ?? null, gate ?? null)
if (canReply) {
return false
}
if (allowFollowing && actors[ownerDid]?.viewer?.followedBy) {
return false
}
for (const listUri of allowListUris) {
const list = lists[listUri]
if (list?.viewerInList) {
return false
}
}
return true
}
formatMaybePostView(
uri: string,
actors: ActorInfoMap,
@ -226,6 +285,7 @@ export class FeedViews {
labels: Labels,
lists: ListInfoMap,
blocks: PostBlocksMap,
viewer: string | null,
opts?: {
usePostViewUnion?: boolean
},
@ -238,6 +298,7 @@ export class FeedViews {
embeds,
labels,
lists,
viewer,
)
if (!post) {
if (!opts?.usePostViewUnion) return

@ -91,6 +91,12 @@ export class GraphService {
.whereRef('list_block.subjectUri', '=', ref('list.uri'))
.select('list_block.uri')
.as('viewerListBlockUri'),
this.db.db
.selectFrom('list_item')
.whereRef('list_item.listUri', '=', ref('list.uri'))
.where('list_item.subjectDid', '=', viewer ?? '')
.select('list_item.uri')
.as('viewerInList'),
])
}

@ -4,6 +4,7 @@ import { List } from '../../db/tables/list'
export type ListInfo = Selectable<List> & {
viewerMuted: string | null
viewerListBlockUri: string | null
viewerInList: string | null
}
export type ListInfoMap = Record<string, ListInfo>

@ -420,7 +420,7 @@ async function validateReply(
const violatesThreadGate = await feedutil.violatesThreadGate(
db,
creator,
new AtUri(reply.root.uri).host,
new AtUri(reply.root.uri).hostname,
replyRefs.root?.record ?? null,
replyRefs.gate?.record ?? null,
)

@ -518,9 +518,6 @@ Object {
"viewer": Object {},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
},
}
`;
@ -587,9 +584,6 @@ Object {
"viewer": Object {},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
},
}
`;

@ -126,9 +126,6 @@ Object {
"uri": "record(0)",
"viewer": Object {},
},
"viewer": Object {
"canReply": true,
},
},
}
`;
@ -204,9 +201,6 @@ Object {
"viewer": Object {},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
},
}
`;
@ -288,9 +282,6 @@ Object {
"uri": "record(7)",
},
],
"viewer": Object {
"canReply": true,
},
},
}
`;

@ -126,9 +126,6 @@ Object {
"uri": "record(0)",
"viewer": Object {},
},
"viewer": Object {
"canReply": true,
},
},
}
`;
@ -204,9 +201,6 @@ Object {
"viewer": Object {},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
},
}
`;
@ -359,9 +353,6 @@ Object {
},
},
],
"viewer": Object {
"canReply": true,
},
},
}
`;

@ -292,9 +292,6 @@ Object {
},
},
],
"viewer": Object {
"canReply": true,
},
}
`;

@ -269,8 +269,5 @@ Object {
},
},
],
"viewer": Object {
"canReply": true,
},
}
`;

@ -195,9 +195,6 @@ Object {
},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
}
`;
@ -437,9 +434,6 @@ Object {
],
},
],
"viewer": Object {
"canReply": true,
},
}
`;
@ -615,9 +609,6 @@ Object {
},
},
],
"viewer": Object {
"canReply": true,
},
}
`;
@ -771,9 +762,6 @@ Object {
],
},
],
"viewer": Object {
"canReply": true,
},
}
`;
@ -826,9 +814,6 @@ Object {
"viewer": Object {},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
}
`;
@ -896,9 +881,6 @@ Object {
"viewer": Object {},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
}
`;
@ -968,9 +950,6 @@ Object {
},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
}
`;
@ -1040,9 +1019,6 @@ Object {
},
},
"replies": Array [],
"viewer": Object {
"canReply": true,
},
}
`;
@ -1243,9 +1219,6 @@ Object {
],
},
],
"viewer": Object {
"canReply": true,
},
}
`;
@ -1384,8 +1357,5 @@ Object {
"replies": Array [],
},
],
"viewer": Object {
"canReply": true,
},
}
`;

@ -29,6 +29,19 @@ describe('views with thread gating', () => {
await network.close()
})
// check that replyDisabled state is applied correctly in a simple method like getPosts
const checkReplyDisabled = async (
uri: string,
user: string,
blocked: boolean | undefined,
) => {
const res = await agent.api.app.bsky.feed.getPosts(
{ uris: [uri] },
{ headers: await network.serviceHeaders(user) },
)
expect(res.data.posts[0].viewer?.replyDisabled).toBe(blocked)
}
it('applies gate for empty rules.', async () => {
const post = await sc.post(sc.dids.carol, 'empty rules')
await pdsAgent.api.app.bsky.feed.threadgate.create(
@ -46,8 +59,9 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(thread))
expect(forSnapshot(thread.post.threadgate)).toMatchSnapshot()
expect(thread.viewer).toEqual({ canReply: false })
expect(thread.post.viewer).toEqual({ replyDisabled: true })
expect(thread.replies?.length).toEqual(0)
await checkReplyDisabled(post.ref.uriStr, sc.dids.alice, true)
})
it('applies gate for mention rule.', async () => {
@ -98,7 +112,8 @@ describe('views with thread gating', () => {
{ headers: await network.serviceHeaders(sc.dids.alice) },
)
assert(isThreadViewPost(aliceThread))
expect(aliceThread.viewer).toEqual({ canReply: false })
expect(aliceThread.post.viewer).toEqual({ replyDisabled: true })
await checkReplyDisabled(post.ref.uriStr, sc.dids.alice, true)
const {
data: { thread: danThread },
} = await agent.api.app.bsky.feed.getPostThread(
@ -107,7 +122,8 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(danThread))
expect(forSnapshot(danThread.post.threadgate)).toMatchSnapshot()
expect(danThread.viewer).toEqual({ canReply: true })
expect(danThread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(post.ref.uriStr, sc.dids.dan, false)
const [reply, ...otherReplies] = danThread.replies ?? []
assert(isThreadViewPost(reply))
expect(otherReplies.length).toEqual(0)
@ -146,7 +162,8 @@ describe('views with thread gating', () => {
{ headers: await network.serviceHeaders(sc.dids.dan) },
)
assert(isThreadViewPost(danThread))
expect(danThread.viewer).toEqual({ canReply: false })
expect(danThread.post.viewer).toEqual({ replyDisabled: true })
await checkReplyDisabled(post.ref.uriStr, sc.dids.dan, true)
const {
data: { thread: aliceThread },
} = await agent.api.app.bsky.feed.getPostThread(
@ -155,7 +172,8 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(aliceThread))
expect(forSnapshot(aliceThread.post.threadgate)).toMatchSnapshot()
expect(aliceThread.viewer).toEqual({ canReply: true })
expect(aliceThread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(post.ref.uriStr, sc.dids.alice, false)
const [reply, ...otherReplies] = aliceThread.replies ?? []
assert(isThreadViewPost(reply))
expect(otherReplies.length).toEqual(0)
@ -235,7 +253,8 @@ describe('views with thread gating', () => {
{ headers: await network.serviceHeaders(sc.dids.bob) },
)
assert(isThreadViewPost(bobThread))
expect(bobThread.viewer).toEqual({ canReply: false })
expect(bobThread.post.viewer).toEqual({ replyDisabled: true })
await checkReplyDisabled(post.ref.uriStr, sc.dids.bob, true)
const {
data: { thread: aliceThread },
} = await agent.api.app.bsky.feed.getPostThread(
@ -243,7 +262,8 @@ describe('views with thread gating', () => {
{ headers: await network.serviceHeaders(sc.dids.alice) },
)
assert(isThreadViewPost(aliceThread))
expect(aliceThread.viewer).toEqual({ canReply: true })
expect(aliceThread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(post.ref.uriStr, sc.dids.alice, false)
const {
data: { thread: danThread },
} = await agent.api.app.bsky.feed.getPostThread(
@ -252,7 +272,8 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(danThread))
expect(forSnapshot(danThread.post.threadgate)).toMatchSnapshot()
expect(danThread.viewer).toEqual({ canReply: true })
expect(danThread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(post.ref.uriStr, sc.dids.dan, false)
const [reply1, reply2, ...otherReplies] = aliceThread.replies ?? []
assert(isThreadViewPost(reply1))
assert(isThreadViewPost(reply2))
@ -292,8 +313,9 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(thread))
expect(forSnapshot(thread.post.threadgate)).toMatchSnapshot()
expect(thread.viewer).toEqual({ canReply: false })
expect(thread.post.viewer).toEqual({ replyDisabled: true })
expect(thread.replies?.length).toEqual(0)
await checkReplyDisabled(post.ref.uriStr, sc.dids.alice, true)
})
it('applies gate for multiple rules.', async () => {
@ -339,7 +361,8 @@ describe('views with thread gating', () => {
{ headers: await network.serviceHeaders(sc.dids.bob) },
)
assert(isThreadViewPost(bobThread))
expect(bobThread.viewer).toEqual({ canReply: false })
expect(bobThread.post.viewer).toEqual({ replyDisabled: true })
await checkReplyDisabled(post.ref.uriStr, sc.dids.bob, true)
const {
data: { thread: aliceThread },
} = await agent.api.app.bsky.feed.getPostThread(
@ -347,7 +370,8 @@ describe('views with thread gating', () => {
{ headers: await network.serviceHeaders(sc.dids.alice) },
)
assert(isThreadViewPost(aliceThread))
expect(aliceThread.viewer).toEqual({ canReply: true })
expect(aliceThread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(post.ref.uriStr, sc.dids.alice, false)
const {
data: { thread: danThread },
} = await agent.api.app.bsky.feed.getPostThread(
@ -356,7 +380,8 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(danThread))
expect(forSnapshot(danThread.post.threadgate)).toMatchSnapshot()
expect(danThread.viewer).toEqual({ canReply: true })
expect(danThread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(post.ref.uriStr, sc.dids.dan, false)
const [reply1, reply2, ...otherReplies] = aliceThread.replies ?? []
assert(isThreadViewPost(reply1))
assert(isThreadViewPost(reply2))
@ -387,7 +412,8 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(thread))
expect(forSnapshot(thread.post.threadgate)).toMatchSnapshot()
expect(thread.viewer).toEqual({ canReply: true })
expect(thread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(post.ref.uriStr, sc.dids.alice, false)
const [reply, ...otherReplies] = thread.replies ?? []
assert(isThreadViewPost(reply))
expect(otherReplies.length).toEqual(0)
@ -438,7 +464,8 @@ describe('views with thread gating', () => {
{ headers: await network.serviceHeaders(sc.dids.dan) },
)
assert(isThreadViewPost(danThread))
expect(danThread.viewer).toEqual({ canReply: false })
expect(danThread.post.viewer).toEqual({ replyDisabled: true })
await checkReplyDisabled(orphanedReply.ref.uriStr, sc.dids.dan, true)
const {
data: { thread: aliceThread },
} = await agent.api.app.bsky.feed.getPostThread(
@ -451,7 +478,8 @@ describe('views with thread gating', () => {
aliceThread.parent.uri === post.ref.uriStr,
)
expect(aliceThread.post.threadgate).toMatchSnapshot()
expect(aliceThread.viewer).toEqual({ canReply: true })
expect(aliceThread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(orphanedReply.ref.uriStr, sc.dids.alice, false)
const [reply, ...otherReplies] = aliceThread.replies ?? []
assert(isThreadViewPost(reply))
expect(otherReplies.length).toEqual(0)
@ -480,7 +508,8 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(thread))
expect(forSnapshot(thread.post.threadgate)).toMatchSnapshot()
expect(thread.viewer).toEqual({ canReply: true })
expect(thread.post.viewer).toEqual({ replyDisabled: false })
await checkReplyDisabled(post.ref.uriStr, sc.dids.carol, false)
const [reply, ...otherReplies] = thread.replies ?? []
assert(isThreadViewPost(reply))
expect(otherReplies.length).toEqual(0)
@ -516,10 +545,11 @@ describe('views with thread gating', () => {
{ headers: await network.serviceHeaders(sc.dids.alice) },
)
assert(isThreadViewPost(thread))
expect(thread.viewer).toEqual({ canReply: false }) // nobody can reply to this, not even alice.
expect(thread.post.viewer).toEqual({ replyDisabled: true }) // nobody can reply to this, not even alice.
expect(thread.replies).toBeUndefined()
expect(thread.parent).toBeUndefined()
expect(thread.post.threadgate).toBeUndefined()
await checkReplyDisabled(badReply.ref.uriStr, sc.dids.alice, true)
// check feed view
const {
data: { feed },
@ -552,8 +582,9 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(threadA))
expect(threadA.post.threadgate).toBeUndefined()
expect(threadA.viewer).toEqual({ canReply: true })
expect(threadA.post.viewer).toEqual({})
expect(threadA.replies?.length).toEqual(1)
await checkReplyDisabled(postA.ref.uriStr, sc.dids.alice, undefined)
const {
data: { thread: threadB },
} = await agent.api.app.bsky.feed.getPostThread(
@ -562,7 +593,8 @@ describe('views with thread gating', () => {
)
assert(isThreadViewPost(threadB))
expect(threadB.post.threadgate).toBeUndefined()
expect(threadB.viewer).toEqual({ canReply: true })
expect(threadB.post.viewer).toEqual({})
await checkReplyDisabled(postB.ref.uriStr, sc.dids.alice, undefined)
expect(threadB.replies?.length).toEqual(1)
})
})

@ -4892,6 +4892,9 @@ export const schemaDict = {
type: 'string',
format: 'at-uri',
},
replyDisabled: {
type: 'boolean',
},
},
},
feedViewPost: {
@ -4975,10 +4978,6 @@ export const schemaDict = {
],
},
},
viewer: {
type: 'ref',
ref: 'lex:app.bsky.feed.defs#viewerThreadState',
},
},
},
notFoundPost: {
@ -5027,14 +5026,6 @@ export const schemaDict = {
},
},
},
viewerThreadState: {
type: 'object',
properties: {
canReply: {
type: 'boolean',
},
},
},
generatorView: {
type: 'object',
required: ['uri', 'cid', 'did', 'creator', 'displayName', 'indexedAt'],

@ -48,6 +48,7 @@ export function validatePostView(v: unknown): ValidationResult {
export interface ViewerState {
repost?: string
like?: string
replyDisabled?: boolean
[k: string]: unknown
}
@ -137,7 +138,6 @@ export interface ThreadViewPost {
| BlockedPost
| { $type: string; [k: string]: unknown }
)[]
viewer?: ViewerThreadState
[k: string]: unknown
}
@ -208,23 +208,6 @@ export function validateBlockedAuthor(v: unknown): ValidationResult {
return lexicons.validate('app.bsky.feed.defs#blockedAuthor', v)
}
export interface ViewerThreadState {
canReply?: boolean
[k: string]: unknown
}
export function isViewerThreadState(v: unknown): v is ViewerThreadState {
return (
isObj(v) &&
hasProp(v, '$type') &&
v.$type === 'app.bsky.feed.defs#viewerThreadState'
)
}
export function validateViewerThreadState(v: unknown): ValidationResult {
return lexicons.validate('app.bsky.feed.defs#viewerThreadState', v)
}
export interface GeneratorView {
uri: string
cid: string