Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Interaction Gating #1561

Merged
merged 79 commits into from
Sep 14, 2023
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
Show all changes
79 commits
Select commit Hold shift + click to select a range
d7ee9f3
lexicons for block lists
devinivy Aug 28, 2023
a21aa90
reorg blockset functionality into graph service, impl block/mute filt…
devinivy Aug 29, 2023
1a5eae2
apply filterBlocksAndMutes() throughout appview except feeds
devinivy Aug 29, 2023
15d1220
update local feeds to pass through cleanFeedSkeleton(), offload block…
devinivy Aug 29, 2023
b194a58
impl for grabbing block/mute details by did pair
devinivy Aug 30, 2023
0dc8413
refactor getActorInfos away, use actor service
devinivy Aug 30, 2023
c6385f1
Merge remote-tracking branch 'origin/main' into block-lists-redux-redux
devinivy Aug 30, 2023
d571bfe
experiment with moving getFeedGenerators over to a pipeline
devinivy Aug 30, 2023
890c9fc
move getPostThread over to a pipeline
devinivy Aug 30, 2023
57fbaa2
move feeds over to pipelines
devinivy Aug 31, 2023
6712b2e
move suggestions and likes over to pipelines
devinivy Aug 31, 2023
b550d1d
move reposted-by, follows, followers over to pipelines, tidy author f…
devinivy Aug 31, 2023
4f4580c
remove old block/mute checks
devinivy Aug 31, 2023
9801663
unify post presentation logic
devinivy Aug 31, 2023
2a1eced
move profiles endpoints over to pipelines
devinivy Aug 31, 2023
8a72f5a
tidy
devinivy Aug 31, 2023
e5b51c2
tidy
devinivy Sep 1, 2023
541c6c0
Merge remote-tracking branch 'origin/main' into block-lists-redux-redux
devinivy Sep 1, 2023
7ab70f9
Merge remote-tracking branch 'origin/block-lists-redux-redux' into bl…
devinivy Sep 1, 2023
cdcc26b
misc fixes
devinivy Sep 1, 2023
992c25f
unify some profile hydration/presentation in appview
devinivy Sep 1, 2023
da221b0
profile detail, split hydration and presentation, misc fixes
devinivy Sep 2, 2023
a353624
unify feed hydration w/ profile hydration
devinivy Sep 4, 2023
0015729
unify hydration step for embeds, tidy application of labels
devinivy Sep 4, 2023
32f826e
setup indexing of list-blocks in bsky appview
devinivy Sep 4, 2023
ec21246
apply list-blocks, impl getListBlocks, tidy getList, tests
devinivy Sep 5, 2023
eb30fd1
tidy
devinivy Sep 5, 2023
8711f6c
update pds proxy snaps
devinivy Sep 5, 2023
a27b532
Merge remote-tracking branch 'origin/block-lists-redux-redux-pipeline…
devinivy Sep 5, 2023
0265e49
update pds proxy snaps
devinivy Sep 5, 2023
280eb5a
fix snap
devinivy Sep 5, 2023
0dfa73b
make algos return feed items, save work in getFeed
devinivy Sep 6, 2023
0259785
misc changes, tidy
devinivy Sep 6, 2023
c84bc28
tidy
devinivy Sep 6, 2023
c645ed7
Merge pull request #1547 from bluesky-social/block-lists-redux-redux-…
devinivy Sep 6, 2023
d01ae4a
Merge remote-tracking branch 'origin/block-lists-redux-redux' into bl…
devinivy Sep 6, 2023
d704510
Merge branch 'main' into block-lists-redux-redux
dholms Sep 6, 2023
f4918c8
Merge pull request #1549 from bluesky-social/block-lists-redux-redux-…
devinivy Sep 6, 2023
447d7a5
fix aturi import
devinivy Sep 6, 2023
2a8e48b
initial lexicons for interaction-gating
devinivy Sep 5, 2023
9b7743c
add interactions view to post views
devinivy Sep 6, 2023
a910034
codegen
devinivy Sep 6, 2023
1656335
model bad reply/interaction check state on posts
devinivy Sep 6, 2023
e916b5c
initial impl for checking bad reply or interaction on write
devinivy Sep 6, 2023
3eb6e63
omit invalid interactions from post thread
devinivy Sep 6, 2023
0e560ea
support not-found list in interaction view
devinivy Sep 7, 2023
aec42a2
hydrate can-reply state on threads
devinivy Sep 7, 2023
e3cf14d
present interaction views on posts
devinivy Sep 7, 2023
cea6ff0
misc fixes, update snaps
devinivy Sep 7, 2023
aa49fbb
tidy/reorg
devinivy Sep 7, 2023
04c738f
tidy
devinivy Sep 7, 2023
d7c6dce
split interaction gating into separate record in lexicon
devinivy Sep 8, 2023
9e74667
switch interaction-gating impl to use separate record type
devinivy Sep 8, 2023
76cc20d
allow checking reply gate w/ root post deletion
devinivy Sep 8, 2023
8d6e0b4
fix
devinivy Sep 8, 2023
29b5ebb
initial gating tests
devinivy Sep 8, 2023
0ab206e
tighten gated reply views, tests
devinivy Sep 8, 2023
50b1875
reply-gating list rule tests
devinivy Sep 8, 2023
f0133f2
allow custom post rkeys within window
devinivy Sep 8, 2023
642f975
Merge remote-tracking branch 'origin/main' into block-lists-redux-redux
devinivy Sep 8, 2023
0535368
hoist actors out of composeThread()
devinivy Sep 8, 2023
cb30cff
tidy
devinivy Sep 8, 2023
31a7306
Merge remote-tracking branch 'origin/block-lists-redux-redux' into in…
devinivy Sep 11, 2023
c2740a3
update thread gate lexicons, codegen
devinivy Sep 11, 2023
25fc209
lex fix
devinivy Sep 11, 2023
b94477c
rename gate to threadgate in bsky, update views
devinivy Sep 11, 2023
52b5972
lex fix
devinivy Sep 11, 2023
60bbeb6
improve terminology around reply validation
devinivy Sep 11, 2023
6a8eaba
fix down migration
devinivy Sep 11, 2023
30ca7f0
remove thread gates on actor unindexing
devinivy Sep 11, 2023
cd54186
add back .prettierignore
devinivy Sep 11, 2023
edee5a6
tidy
devinivy Sep 11, 2023
d11ecd8
run ci on all prs
dholms Sep 12, 2023
5218ab9
syntax
dholms Sep 12, 2023
51cd347
run ci on all prs
dholms Sep 12, 2023
bd6a7a6
Merge branch 'block-lists-redux-redux' into interaction-gating-init
dholms Sep 12, 2023
79cb410
format
dholms Sep 12, 2023
1f852b2
merge main
dholms Sep 14, 2023
d6bdcab
fix snap
dholms Sep 14, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 5 additions & 3 deletions packages/bsky/src/api/app/bsky/actor/getSuggestions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ export default function (server: Server, ctx: AppContext) {
const getSuggestions = createPipeline(
skeleton,
hydration,
noBlocks,
noBlocksOrMutes,
presentation,
)
server.app.bsky.actor.getSuggestions({
Expand Down Expand Up @@ -93,11 +93,13 @@ const hydration = async (state: SkeletonState, ctx: Context) => {
return { ...state, bam, actors }
}

const noBlocks = (state: HydrationState) => {
const noBlocksOrMutes = (state: HydrationState) => {
const { viewer } = state.params
if (!viewer) return state
state.suggestions = state.suggestions.filter(
(item) => !state.bam.block([viewer, item.did]),
(item) =>
!state.bam.block([viewer, item.did]) &&
!state.bam.mute([viewer, item.did]),
)
return state
}
Expand Down
47 changes: 23 additions & 24 deletions packages/bsky/src/api/app/bsky/feed/getFeed.ts
Original file line number Diff line number Diff line change
Expand Up @@ -65,22 +65,20 @@ const skeleton = async (
ctx: Context,
): Promise<SkeletonState> => {
const timerSkele = new ServerTimer('skele').start()
const { db } = ctx
const localAlgo = ctx.appCtx.algos[params.feed]
const feedParams: GetFeedParams = {
feed: params.feed,
limit: params.limit,
cursor: params.cursor,
}
const skeleton =
const { feedItems, cursor, ...passthrough } =
localAlgo !== undefined
? await localAlgo(ctx.appCtx, params, params.viewer)
: await skeletonFromFeedGen(ctx.appCtx, db, feedParams, ctx.authorization)
const { feed, cursor, ...passthrough } = skeleton
: await skeletonFromFeedGen(ctx, feedParams)
return {
params,
cursor,
feedSkele: feed,
feedItems,
timerSkele: timerSkele.stop(),
passthrough,
}
Expand All @@ -89,14 +87,13 @@ const skeleton = async (
const hydration = async (state: SkeletonState, ctx: Context) => {
const timerHydr = new ServerTimer('hydr').start()
const { feedService } = ctx
const { params, feedSkele } = state
const feedItems = await cleanFeedSkeleton(feedSkele, ctx)
const { params, feedItems } = state
const refs = feedService.feedItemRefs(feedItems)
const hydrated = await feedService.feedHydration({
...refs,
viewer: params.viewer,
})
return { ...state, ...hydrated, feedItems, timerHydr: timerHydr.stop() }
return { ...state, ...hydrated, timerHydr: timerHydr.stop() }
}

const noBlocksOrMutes = (state: HydrationState) => {
Expand Down Expand Up @@ -137,7 +134,7 @@ type Params = GetFeedParams & { viewer: string }

type SkeletonState = {
params: Params
feedSkele: SkeletonFeedPost[]
feedItems: FeedRow[]
passthrough: Record<string, unknown> // pass through additional items in feedgen response
cursor?: string
timerSkele: ServerTimer
Expand All @@ -147,11 +144,10 @@ type HydrationState = SkeletonState &
FeedHydrationState & { feedItems: FeedRow[]; timerHydr: ServerTimer }

const skeletonFromFeedGen = async (
ctx: AppContext,
db: Database,
ctx: Context,
params: GetFeedParams,
authorization?: string,
): Promise<AlgoResponse> => {
const { db, appCtx, authorization } = ctx
const { feed } = params
// Resolve and fetch feed skeleton
const found = await db.db
Expand All @@ -166,7 +162,7 @@ const skeletonFromFeedGen = async (

let resolved: DidDocument | null
try {
resolved = await ctx.idResolver.did.resolve(feedDid)
resolved = await appCtx.idResolver.did.resolve(feedDid)
} catch (err) {
if (err instanceof PoorlyFormattedDidDocumentError) {
throw new InvalidRequestError(`invalid did document: ${feedDid}`)
Expand All @@ -190,7 +186,7 @@ const skeletonFromFeedGen = async (
try {
// @TODO currently passthrough auth headers from pds
const headers: Record<string, string> = authorization
? { authorization }
? { authorization: authorization }
: {}
const result = await agent.api.app.bsky.feed.getFeedSkeleton(params, {
headers,
Expand All @@ -214,27 +210,30 @@ const skeletonFromFeedGen = async (
throw err
}

return {
...skeleton,
feed: skeleton.feed.slice(0, params.limit), // enforce limit
}
const { feed: feedSkele, ...skele } = skeleton
const feedItems = await skeletonToFeedItems(
feedSkele.slice(0, params.limit),
ctx,
)

return { ...skele, feedItems }
}

const cleanFeedSkeleton = async (
const skeletonToFeedItems = async (
skeleton: SkeletonFeedPost[],
ctx: Context,
): Promise<FeedRow[]> => {
const { feedService } = ctx
const feedItemUris = skeleton.map(getSkeleFeedItemUri)
const feedItems = await feedService.getFeedItems(feedItemUris)
const cleaned: FeedRow[] = []
const feedItemsRaw = await feedService.getFeedItems(feedItemUris)
const results: FeedRow[] = []
for (const skeleItem of skeleton) {
const feedItem = feedItems[getSkeleFeedItemUri(skeleItem)]
const feedItem = feedItemsRaw[getSkeleFeedItemUri(skeleItem)]
if (feedItem && feedItem.postUri === skeleItem.post) {
cleaned.push(feedItem)
results.push(feedItem)
}
}
return cleaned
return results
}

const getSkeleFeedItemUri = (item: SkeletonFeedPost) => {
Expand Down
4 changes: 3 additions & 1 deletion packages/bsky/src/api/app/bsky/feed/getFeedSkeleton.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { InvalidRequestError } from '@atproto/xrpc-server'
import { Server } from '../../../../lexicon'
import AppContext from '../../../../context'
import { toSkeletonItem } from '../../../../feed-gen/types'

export default function (server: Server, ctx: AppContext) {
server.app.bsky.feed.getFeedSkeleton({
Expand All @@ -19,7 +20,8 @@ export default function (server: Server, ctx: AppContext) {
return {
encoding: 'application/json',
body: {
feed: result.feed, // @TODO should we proactively filter blocks/mutes from the skeleton, or treat this similar to other custom feeds?
// @TODO should we proactively filter blocks/mutes from the skeleton, or treat this similar to other custom feeds?
feed: result.feedItems.map(toSkeletonItem),
cursor: result.cursor,
},
}
Expand Down
2 changes: 1 addition & 1 deletion packages/bsky/src/api/app/bsky/feed/getPostThread.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ export default function (server: Server, ctx: AppContext) {
const getPostThread = createPipeline(
skeleton,
hydration,
noRules,
noRules, // handled in presentation: 3p block-violating replies are turned to #blockedPost, viewer blocks turned to #notFoundPost.
presentation,
)
server.app.bsky.feed.getPostThread({
Expand Down
4 changes: 2 additions & 2 deletions packages/bsky/src/feed-gen/best-of-follows.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { InvalidRequestError } from '@atproto/xrpc-server'
import { QueryParams as SkeletonParams } from '../lexicon/types/app/bsky/feed/getFeedSkeleton'
import { AlgoHandler, AlgoResponse, toSkeletonItem } from './types'
import { AlgoHandler, AlgoResponse } from './types'
import { GenericKeyset, paginate } from '../db/pagination'
import AppContext from '../context'

Expand Down Expand Up @@ -39,7 +39,7 @@ const handler: AlgoHandler = async (
const feedItems = await builder.execute()

return {
feed: feedItems.map(toSkeletonItem),
feedItems,
cursor: keyset.packFromResult(feedItems),
}
}
Expand Down
4 changes: 2 additions & 2 deletions packages/bsky/src/feed-gen/bsky-team.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { NotEmptyArray } from '@atproto/common'
import { QueryParams as SkeletonParams } from '../lexicon/types/app/bsky/feed/getFeedSkeleton'
import AppContext from '../context'
import { paginate } from '../db/pagination'
import { AlgoHandler, AlgoResponse, toSkeletonItem } from './types'
import { AlgoHandler, AlgoResponse } from './types'
import { FeedKeyset } from '../api/app/bsky/util/feed'

const BSKY_TEAM: NotEmptyArray<string> = [
Expand Down Expand Up @@ -34,7 +34,7 @@ const handler: AlgoHandler = async (
const feedItems = await feedQb.execute()

return {
feed: feedItems.map(toSkeletonItem),
feedItems,
cursor: keyset.packFromResult(feedItems),
}
}
Expand Down
4 changes: 2 additions & 2 deletions packages/bsky/src/feed-gen/hot-classic.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import AppContext from '../context'
import { NotEmptyArray } from '@atproto/common'
import { QueryParams as SkeletonParams } from '../lexicon/types/app/bsky/feed/getFeedSkeleton'
import { paginate } from '../db/pagination'
import { AlgoHandler, AlgoResponse, toSkeletonItem } from './types'
import { AlgoHandler, AlgoResponse } from './types'
import { FeedKeyset } from '../api/app/bsky/util/feed'
import { valuesList } from '../db/util'

Expand Down Expand Up @@ -47,7 +47,7 @@ const handler: AlgoHandler = async (
const feedItems = await feedQb.execute()

return {
feed: feedItems.map(toSkeletonItem),
feedItems,
cursor: keyset.packFromResult(feedItems),
}
}
Expand Down
4 changes: 2 additions & 2 deletions packages/bsky/src/feed-gen/mutuals.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { QueryParams as SkeletonParams } from '../lexicon/types/app/bsky/feed/getFeedSkeleton'
import AppContext from '../context'
import { paginate } from '../db/pagination'
import { AlgoHandler, AlgoResponse, toSkeletonItem } from './types'
import { AlgoHandler, AlgoResponse } from './types'
import { FeedKeyset, getFeedDateThreshold } from '../api/app/bsky/util/feed'

const handler: AlgoHandler = async (
Expand Down Expand Up @@ -45,7 +45,7 @@ const handler: AlgoHandler = async (
const feedItems = await feedQb.execute()

return {
feed: feedItems.map(toSkeletonItem),
feedItems,
cursor: keyset.packFromResult(feedItems),
}
}
Expand Down
3 changes: 2 additions & 1 deletion packages/bsky/src/feed-gen/types.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
import AppContext from '../context'
import { SkeletonFeedPost } from '../lexicon/types/app/bsky/feed/defs'
import { QueryParams as SkeletonParams } from '../lexicon/types/app/bsky/feed/getFeedSkeleton'
import { FeedRow } from '../services/feed'

export type AlgoResponse = {
feed: SkeletonFeedPost[]
feedItems: FeedRow[]
cursor?: string
}

Expand Down
9 changes: 7 additions & 2 deletions packages/bsky/src/feed-gen/whats-hot.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { NotEmptyArray } from '@atproto/common'
import { InvalidRequestError } from '@atproto/xrpc-server'
import { QueryParams as SkeletonParams } from '../lexicon/types/app/bsky/feed/getFeedSkeleton'
import { AlgoHandler, AlgoResponse, toSkeletonItem } from './types'
import { AlgoHandler, AlgoResponse } from './types'
import { GenericKeyset, paginate } from '../db/pagination'
import AppContext from '../context'
import { valuesList } from '../db/util'
Expand Down Expand Up @@ -51,6 +51,11 @@ const handler: AlgoHandler = async (
sql<FeedItemType>`${'post'}`.as('type'),
'post.uri as uri',
'post.uri as postUri',
'post.creator as originatorDid',
'post.creator as postAuthorDid',
'post.replyParent as replyParent',
'post.replyRoot as replyRoot',
'post.indexedAt as sortAt',
'candidate.score',
'candidate.cid',
])
Expand All @@ -61,7 +66,7 @@ const handler: AlgoHandler = async (
const feedItems = await builder.execute()

return {
feed: feedItems.map(toSkeletonItem),
feedItems,
cursor: keyset.packFromResult(feedItems),
}
}
Expand Down
4 changes: 2 additions & 2 deletions packages/bsky/src/feed-gen/with-friends.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import AppContext from '../context'
import { QueryParams as SkeletonParams } from '../lexicon/types/app/bsky/feed/getFeedSkeleton'
import { paginate } from '../db/pagination'
import { AlgoHandler, AlgoResponse, toSkeletonItem } from './types'
import { AlgoHandler, AlgoResponse } from './types'
import { FeedKeyset, getFeedDateThreshold } from '../api/app/bsky/util/feed'

const handler: AlgoHandler = async (
Expand Down Expand Up @@ -31,7 +31,7 @@ const handler: AlgoHandler = async (
const feedItems = await postsQb.execute()

return {
feed: feedItems.map(toSkeletonItem),
feedItems,
cursor: keyset.packFromResult(feedItems),
}
}
Expand Down