Implement validation and proper type detection

zio/stable
Paul Frazee 2023-01-03 13:08:56 -06:00
parent 1acef14a1c
commit b9b0965000
7 changed files with 114 additions and 30 deletions

View File

@ -33,6 +33,7 @@ export class FeedItemModel {
// data // data
post: PostView post: PostView
postRecord?: AppBskyFeedPost.Record
reply?: FeedViewPost['reply'] reply?: FeedViewPost['reply']
replyParent?: FeedItemModel replyParent?: FeedItemModel
reason?: FeedViewPost['reason'] reason?: FeedViewPost['reason']
@ -44,6 +45,22 @@ export class FeedItemModel {
) { ) {
this._reactKey = reactKey this._reactKey = reactKey
this.post = v.post this.post = v.post
if (AppBskyFeedPost.isRecord(this.post.record)) {
const valid = AppBskyFeedPost.validateRecord(this.post.record)
if (valid.success) {
this.postRecord = this.post.record
} else {
rootStore.log.warn(
'Received an invalid app.bsky.feed.post record',
valid.error,
)
}
} else {
rootStore.log.warn(
'app.bsky.feed.getTimeline or app.bsky.feed.getAuthorFeed served an unexpected record type',
this.post.record,
)
}
this.reply = v.reply this.reply = v.reply
if (v.reply?.parent) { if (v.reply?.parent) {
this.replyParent = new FeedItemModel(rootStore, '', { this.replyParent = new FeedItemModel(rootStore, '', {

View File

@ -2,11 +2,16 @@ import {makeAutoObservable, runInAction} from 'mobx'
import { import {
AppBskyNotificationList as ListNotifications, AppBskyNotificationList as ListNotifications,
AppBskyActorRef as ActorRef, AppBskyActorRef as ActorRef,
AppBskyFeedPost,
AppBskyFeedRepost,
AppBskyFeedTrend,
AppBskyFeedVote,
AppBskyGraphAssertion,
AppBskyGraphFollow,
APP_BSKY_GRAPH, APP_BSKY_GRAPH,
} from '@atproto/api' } from '@atproto/api'
import {RootStoreModel} from './root-store' import {RootStoreModel} from './root-store'
import {PostThreadViewModel} from './post-thread-view' import {PostThreadViewModel} from './post-thread-view'
import {hasProp} from '../lib/type-guards'
import {cleanError} from '../../lib/strings' import {cleanError} from '../../lib/strings'
const UNGROUPABLE_REASONS = ['trend', 'assertion'] const UNGROUPABLE_REASONS = ['trend', 'assertion']
@ -19,7 +24,15 @@ export interface GroupedNotification extends ListNotifications.Notification {
additional?: ListNotifications.Notification[] additional?: ListNotifications.Notification[]
} }
export class NotificationsViewItemModel implements GroupedNotification { type SupportedRecord =
| AppBskyFeedPost.Record
| AppBskyFeedRepost.Record
| AppBskyFeedTrend.Record
| AppBskyFeedVote.Record
| AppBskyGraphAssertion.Record
| AppBskyGraphFollow.Record
export class NotificationsViewItemModel {
// ui state // ui state
_reactKey: string = '' _reactKey: string = ''
@ -34,7 +47,7 @@ export class NotificationsViewItemModel implements GroupedNotification {
} }
reason: string = '' reason: string = ''
reasonSubject?: string reasonSubject?: string
record: any = {} record?: SupportedRecord
isRead: boolean = false isRead: boolean = false
indexedAt: string = '' indexedAt: string = ''
additional?: NotificationsViewItemModel[] additional?: NotificationsViewItemModel[]
@ -58,7 +71,7 @@ export class NotificationsViewItemModel implements GroupedNotification {
this.author = v.author this.author = v.author
this.reason = v.reason this.reason = v.reason
this.reasonSubject = v.reasonSubject this.reasonSubject = v.reasonSubject
this.record = v.record this.record = this.toSupportedRecord(v.record)
this.isRead = v.isRead this.isRead = v.isRead
this.indexedAt = v.indexedAt this.indexedAt = v.indexedAt
if (v.additional?.length) { if (v.additional?.length) {
@ -116,23 +129,55 @@ export class NotificationsViewItemModel implements GroupedNotification {
get isInvite() { get isInvite() {
return ( return (
this.isAssertion && this.record.assertion === APP_BSKY_GRAPH.AssertMember this.isAssertion &&
AppBskyGraphAssertion.isRecord(this.record) &&
this.record.assertion === APP_BSKY_GRAPH.AssertMember
) )
} }
get subjectUri() { get subjectUri(): string {
if (this.reasonSubject) { if (this.reasonSubject) {
return this.reasonSubject return this.reasonSubject
} }
const record = this.record
if ( if (
hasProp(this.record, 'subject') && AppBskyFeedRepost.isRecord(record) ||
typeof this.record.subject === 'string' AppBskyFeedTrend.isRecord(record) ||
AppBskyFeedVote.isRecord(record)
) { ) {
return this.record.subject return record.subject.uri
} }
return '' return ''
} }
toSupportedRecord(v: unknown): SupportedRecord | undefined {
for (const ns of [
AppBskyFeedPost,
AppBskyFeedRepost,
AppBskyFeedTrend,
AppBskyFeedVote,
AppBskyGraphAssertion,
AppBskyGraphFollow,
]) {
if (ns.isRecord(v)) {
const valid = ns.validateRecord(v)
if (valid.success) {
return v
} else {
this.rootStore.log.warn('Received an invalid record', {
record: v,
error: valid.error,
})
return
}
}
}
this.rootStore.log.warn(
'app.bsky.notifications.list served an unsupported record type',
v,
)
}
async fetchAdditionalData() { async fetchAdditionalData() {
if (!this.needsAdditionalData) { if (!this.needsAdditionalData) {
return return
@ -140,7 +185,7 @@ export class NotificationsViewItemModel implements GroupedNotification {
let postUri let postUri
if (this.isReply || this.isMention) { if (this.isReply || this.isMention) {
postUri = this.uri postUri = this.uri
} else if (this.isUpvote || this.isRead || this.isTrend) { } else if (this.isUpvote || this.isRepost || this.isTrend) {
postUri = this.subjectUri postUri = this.subjectUri
} }
if (postUri) { if (postUri) {

View File

@ -1,5 +1,8 @@
import {makeAutoObservable, runInAction} from 'mobx' import {makeAutoObservable, runInAction} from 'mobx'
import {AppBskyFeedGetPostThread as GetPostThread} from '@atproto/api' import {
AppBskyFeedGetPostThread as GetPostThread,
AppBskyFeedPost as FeedPost,
} from '@atproto/api'
import {AtUri} from '../../third-party/uri' import {AtUri} from '../../third-party/uri'
import {RootStoreModel} from './root-store' import {RootStoreModel} from './root-store'
import * as apilib from '../lib/api' import * as apilib from '../lib/api'
@ -19,7 +22,8 @@ export class PostThreadViewPostModel {
_hasMore = false _hasMore = false
// data // data
post: GetPostThread.ThreadViewPost['post'] post: FeedPost.View
postRecord?: FeedPost.Record
parent?: PostThreadViewPostModel | GetPostThread.NotFoundPost parent?: PostThreadViewPostModel | GetPostThread.NotFoundPost
replies?: (PostThreadViewPostModel | GetPostThread.NotFoundPost)[] replies?: (PostThreadViewPostModel | GetPostThread.NotFoundPost)[]
@ -30,6 +34,22 @@ export class PostThreadViewPostModel {
) { ) {
this._reactKey = reactKey this._reactKey = reactKey
this.post = v.post this.post = v.post
if (FeedPost.isRecord(this.post.record)) {
const valid = FeedPost.validateRecord(this.post.record)
if (valid.success) {
this.postRecord = this.post.record
} else {
rootStore.log.warn(
'Received an invalid app.bsky.feed.post record',
valid.error,
)
}
} else {
rootStore.log.warn(
'app.bsky.feed.getPostThread served an unexpected record type',
this.post.record,
)
}
// replies and parent are handled via assignTreeModels // replies and parent are handled via assignTreeModels
makeAutoObservable(this, {rootStore: false}) makeAutoObservable(this, {rootStore: false})
} }
@ -278,7 +298,6 @@ export class PostThreadViewModel {
} }
private _replaceAll(res: GetPostThread.Response) { private _replaceAll(res: GetPostThread.Response) {
// TODO: validate .record
// sortThread(res.data.thread) TODO needed? // sortThread(res.data.thread) TODO needed?
const keyGen = reactKeyGenerator() const keyGen = reactKeyGenerator()
const thread = new PostThreadViewPostModel( const thread = new PostThreadViewPostModel(

View File

@ -221,14 +221,14 @@ function AdditionalPostText({
additionalPost?: PostThreadViewModel additionalPost?: PostThreadViewModel
}) { }) {
const pal = usePalette('default') const pal = usePalette('default')
if (!additionalPost) { if (!additionalPost || !additionalPost.thread?.postRecord) {
return <View /> return <View />
} }
if (additionalPost.error) { if (additionalPost.error) {
return <ErrorMessage message={additionalPost.error} /> return <ErrorMessage message={additionalPost.error} />
} }
return ( return (
<Text style={pal.textLight}>{additionalPost.thread?.post.record.text}</Text> <Text style={pal.textLight}>{additionalPost.thread?.postRecord.text}</Text>
) )
} }

View File

@ -3,7 +3,6 @@ import {observer} from 'mobx-react-lite'
import {StyleSheet, View} from 'react-native' import {StyleSheet, View} from 'react-native'
import Clipboard from '@react-native-clipboard/clipboard' import Clipboard from '@react-native-clipboard/clipboard'
import {AtUri} from '../../../third-party/uri' import {AtUri} from '../../../third-party/uri'
import {AppBskyFeedPost} from '@atproto/api'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {PostThreadViewPostModel} from '../../../state/models/post-thread-view' import {PostThreadViewPostModel} from '../../../state/models/post-thread-view'
import {Link} from '../util/Link' import {Link} from '../util/Link'
@ -18,6 +17,7 @@ import {useStores} from '../../../state'
import {PostMeta} from '../util/PostMeta' import {PostMeta} from '../util/PostMeta'
import {PostEmbeds} from '../util/PostEmbeds' import {PostEmbeds} from '../util/PostEmbeds'
import {PostCtrls} from '../util/PostCtrls' import {PostCtrls} from '../util/PostCtrls'
import {ErrorMessage} from '../util/error/ErrorMessage'
import {ComposePrompt} from '../composer/Prompt' import {ComposePrompt} from '../composer/Prompt'
import {usePalette} from '../../lib/hooks/usePalette' import {usePalette} from '../../lib/hooks/usePalette'
@ -33,7 +33,7 @@ export const PostThreadItem = observer(function PostThreadItem({
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const [deleted, setDeleted] = useState(false) const [deleted, setDeleted] = useState(false)
const record = item.post.record as unknown as AppBskyFeedPost.Record const record = item.postRecord
const hasEngagement = item.post.upvoteCount || item.post.repostCount const hasEngagement = item.post.upvoteCount || item.post.repostCount
const itemHref = useMemo(() => { const itemHref = useMemo(() => {
@ -96,6 +96,10 @@ export const PostThreadItem = observer(function PostThreadItem({
) )
} }
if (!record) {
return <ErrorMessage message="Invalid or unsupported post record" />
}
if (deleted) { if (deleted) {
return ( return (
<View style={[styles.outer, pal.view, s.p20, s.flexRow]}> <View style={[styles.outer, pal.view, s.p20, s.flexRow]}>

View File

@ -9,7 +9,6 @@ import {
import {observer} from 'mobx-react-lite' import {observer} from 'mobx-react-lite'
import Clipboard from '@react-native-clipboard/clipboard' import Clipboard from '@react-native-clipboard/clipboard'
import {AtUri} from '../../../third-party/uri' import {AtUri} from '../../../third-party/uri'
import {AppBskyFeedPost} from '@atproto/api'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {PostThreadViewModel} from '../../../state/models/post-thread-view' import {PostThreadViewModel} from '../../../state/models/post-thread-view'
import {Link} from '../util/Link' import {Link} from '../util/Link'
@ -21,6 +20,7 @@ import {Text} from '../util/text/Text'
import {RichText} from '../util/text/RichText' import {RichText} from '../util/text/RichText'
import * as Toast from '../util/Toast' import * as Toast from '../util/Toast'
import {UserAvatar} from '../util/UserAvatar' import {UserAvatar} from '../util/UserAvatar'
import {ErrorMessage} from '../util/error/ErrorMessage'
import {useStores} from '../../../state' import {useStores} from '../../../state'
import {s, colors} from '../../lib/styles' import {s, colors} from '../../lib/styles'
import {usePalette} from '../../lib/hooks/usePalette' import {usePalette} from '../../lib/hooks/usePalette'
@ -68,7 +68,7 @@ export const Post = observer(function Post({
// error // error
// = // =
if (view.hasError || !view.thread) { if (view.hasError || !view.thread || !view.thread?.postRecord) {
return ( return (
<View style={pal.view}> <View style={pal.view}>
<Text>{view.error || 'Thread not found'}</Text> <Text>{view.error || 'Thread not found'}</Text>
@ -79,7 +79,7 @@ export const Post = observer(function Post({
// loaded // loaded
// = // =
const item = view.thread const item = view.thread
const record = view.thread?.post.record as unknown as AppBskyFeedPost.Record const record = view.thread.postRecord
const itemUrip = new AtUri(item.post.uri) const itemUrip = new AtUri(item.post.uri)
const itemHref = `/profile/${item.post.author.handle}/post/${itemUrip.rkey}` const itemHref = `/profile/${item.post.author.handle}/post/${itemUrip.rkey}`

View File

@ -4,7 +4,6 @@ import {StyleSheet, View} from 'react-native'
import Clipboard from '@react-native-clipboard/clipboard' import Clipboard from '@react-native-clipboard/clipboard'
import Svg, {Circle, Line} from 'react-native-svg' import Svg, {Circle, Line} from 'react-native-svg'
import {AtUri} from '../../../third-party/uri' import {AtUri} from '../../../third-party/uri'
import {AppBskyFeedPost} from '@atproto/api'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {FeedItemModel} from '../../../state/models/feed-view' import {FeedItemModel} from '../../../state/models/feed-view'
import {Link} from '../util/Link' import {Link} from '../util/Link'
@ -34,7 +33,7 @@ export const FeedItem = observer(function ({
const theme = useTheme() const theme = useTheme()
const pal = usePalette('default') const pal = usePalette('default')
const [deleted, setDeleted] = useState(false) const [deleted, setDeleted] = useState(false)
const record = item.post.record as unknown as AppBskyFeedPost.Record const record = item.postRecord
const itemHref = useMemo(() => { const itemHref = useMemo(() => {
const urip = new AtUri(item.post.uri) const urip = new AtUri(item.post.uri)
return `/profile/${item.post.author.handle}/post/${urip.rkey}` return `/profile/${item.post.author.handle}/post/${urip.rkey}`
@ -42,22 +41,22 @@ export const FeedItem = observer(function ({
const itemTitle = `Post by ${item.post.author.handle}` const itemTitle = `Post by ${item.post.author.handle}`
const authorHref = `/profile/${item.post.author.handle}` const authorHref = `/profile/${item.post.author.handle}`
const replyAuthorDid = useMemo(() => { const replyAuthorDid = useMemo(() => {
if (!record.reply) return '' if (!record?.reply) return ''
const urip = new AtUri(record.reply.parent?.uri || record.reply.root.uri) const urip = new AtUri(record.reply.parent?.uri || record.reply.root.uri)
return urip.hostname return urip.hostname
}, [record.reply]) }, [record?.reply])
const replyHref = useMemo(() => { const replyHref = useMemo(() => {
if (!record.reply) return '' if (!record?.reply) return ''
const urip = new AtUri(record.reply.parent?.uri || record.reply.root.uri) const urip = new AtUri(record?.reply.parent?.uri || record?.reply.root.uri)
return `/profile/${urip.hostname}/post/${urip.rkey}` return `/profile/${urip.hostname}/post/${urip.rkey}`
}, [record.reply]) }, [record?.reply])
const onPressReply = () => { const onPressReply = () => {
store.shell.openComposer({ store.shell.openComposer({
replyTo: { replyTo: {
uri: item.post.uri, uri: item.post.uri,
cid: item.post.cid, cid: item.post.cid,
text: record.text as string, text: record?.text || '',
author: { author: {
handle: item.post.author.handle, handle: item.post.author.handle,
displayName: item.post.author.displayName, displayName: item.post.author.displayName,
@ -77,7 +76,7 @@ export const FeedItem = observer(function ({
.catch(e => store.log.error('Failed to toggle upvote', e)) .catch(e => store.log.error('Failed to toggle upvote', e))
} }
const onCopyPostText = () => { const onCopyPostText = () => {
Clipboard.setString(record.text) Clipboard.setString(record?.text || '')
Toast.show('Copied to clipboard') Toast.show('Copied to clipboard')
} }
const onDeletePost = () => { const onDeletePost = () => {
@ -93,7 +92,7 @@ export const FeedItem = observer(function ({
) )
} }
if (deleted) { if (!record || deleted) {
return <View /> return <View />
} }