This repository was archived by the owner on Apr 18, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 23
/
Copy pathforum.ts
361 lines (303 loc) · 10.7 KB
/
forum.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
import { BigNumber } from 'ethers'
import { File } from 'web3.storage'
import type { Web3Storage } from 'web3.storage'
import type { Event, BigNumberish } from 'ethers'
import type { Forum as ForumContract } from '../../../contract/typechain'
export type { Forum as ForumContract } from '../../../contract/typechain'
export interface ForumAPIOptions {
readonlyContract: ForumContract,
authorizedContract?: ForumContract,
storage?: Web3Storage,
}
export class NoAuthorizedAccountError extends Error {
constructor() {
super(`This operation is only allowed when logged in. Log in with your wallet and try again.`)
}
}
/**
* Provides a high-level API for interacting with the forum smart contract and Web3.Storage.
*/
export class ForumAPI {
#readonlyContract: ForumContract
#authorizedContract: ForumContract | undefined
#storage: Web3Storage | undefined
#itemContentCache: Map<CIDString, ItemContent> = new Map()
constructor(opts: ForumAPIOptions) {
this.#readonlyContract = opts.readonlyContract
this.#authorizedContract = opts.authorizedContract
this.#storage = opts.storage
}
get canVote(): boolean {
return this.#authorizedContract != null
}
get canPost(): boolean {
return this.canVote && this.#storage != null
}
#getAuthorizedContract() {
if (!this.#authorizedContract) {
throw new NoAuthorizedAccountError()
}
return this.#authorizedContract
}
/**
* Adds the given post content object to Web3.Storage, then adds a
* new post record to the blockchain.
* @param post content of the post.
* @returns the unique id of the new post.
* @throws if content storage fails.
*/
async addPost(post: PostContent): Promise<ItemId> {
const cid = await this.#storeItemContent(post)
const tx = await this.#getAuthorizedContract().addPost(cid)
const receipt = await tx.wait()
const id = idFromEvents(receipt.events)
if (!id) {
throw new Error('unable to determine post id')
}
return id
}
/**
* Retrieves an item by its id by fetching the content CID from the blockchain,
* then fetching content from IPFS using the Web3.Storage client.
*
* @param itemId the id of a post or comment to fetch.
* @returns the item with the given id.
* @throws if no item exists with the given ID, or if the item content fails to load
*/
async getItem(itemId: ItemId, opts?: { includeScore?: boolean }): Promise<Item> {
const itemStruct = await this.#readonlyContract.getItem(itemId)
const item = await this.#hydrateItem(itemStruct)
const { includeScore } = opts || {}
if (includeScore) {
const score = await this.#readonlyContract.getItemScore(itemId)
item.score = score.toNumber()
}
return item
}
async getRecentPosts(opts: {limit?: number, includeScore?: boolean} = {}): Promise<Item[]> {
const limit = opts.limit || 20
const { includeScore } = opts
// page through the event log, querying in batches of a few block at a time
// until we have enough items, or until we hit block zero.
//
// FIXME: scanning back to block zero is fine for local dev, but a real deployment
// should only scan back until the block the contract was deployed.
const currentBlock = await this.#readonlyContract.provider.getBlockNumber()
if (currentBlock === 0) {
return []
}
const blocksPerPage = 20
let fromBlock = Math.max(0, currentBlock - blocksPerPage)
let toBlock = currentBlock
let ids: ItemId[] = []
while (ids.length < limit && toBlock > 0) {
const someIds = await this.#getPostIdsFromEvents(fromBlock, toBlock)
// return in reverse-chronological order
someIds.reverse()
ids = ids.concat(...someIds)
ids = ids.filter(id => !!id)
toBlock = fromBlock
fromBlock = Math.max(0, fromBlock - blocksPerPage)
}
ids = ids.slice(0, limit)
const promises = ids.map(id => this.getItem(id, { includeScore }))
return Promise.all(promises)
}
async #getPostIdsFromEvents(fromBlock: string | number, toBlock: string | number): Promise<ItemId[]> {
// filter NewItem events - we want posts, which have parent id == 0
const filter = this.#readonlyContract.filters.NewItem(null, BigNumber.from(0), null)
const events = await this.#readonlyContract.queryFilter(filter, fromBlock, toBlock)
return allIdsFromEvents(events)
}
/**
* Adds a comment to an existing post.
* @param comment content of the comment. `comment.postId` must be a valid id for an existing post
* @returns the unique id of the new comment
* @throws if comment.postId is missing, if no post exists with that id, or if the content storage fails
*/
async addComment(comment: CommentContent): Promise<ItemId> {
const cid = await this.#storeItemContent(comment)
const tx = await this.#getAuthorizedContract().addComment(comment.parentId, cid)
const receipt = await tx.wait()
const id = idFromEvents(receipt.events)
if (!id) {
throw new Error('unable to determine comment id')
}
return id
}
/**
* Retrieves a comment by id.
* @param commentId the unique id of the comment to fetch
* @returns the comment with the given id
* @throws if no comment exists with the given id, or if the content fails to load.
*/
async getComment(commentId: ItemId): Promise<Item> {
// Get comment info from the contract
const itemStruct = await this.#readonlyContract.getItem(commentId)
return this.#hydrateItem(itemStruct)
}
async #hydrateItem(itemStruct: ItemStruct): Promise<Item> {
const { contentCID } = itemStruct
if (this.#itemContentCache.has(contentCID)) {
const content = this.#itemContentCache.get(contentCID)!
return { ...itemStruct, content }
}
// use contentCID to fetch item content
console.log(`fectching content for item ${itemStruct.id}. CID: ${contentCID}`)
const content = await this.#getJson(contentCID) as ItemContent // TODO: validate!
this.#itemContentCache.set(contentCID, content)
return { ...itemStruct, content }
}
/**
* Applies the given vote to an item.
*
* @param itemId - the unique id of a post or comment
* @param vote - an upvote, downvote, or retraction
*/
async voteForItem(itemId: ItemId, vote: VoteValue): Promise<void> {
const tx = await this.#getAuthorizedContract().voteForItem(itemId, vote)
await tx.wait()
}
/**
* Get the total number of votes for a post or comment. May be negative.
* @param itemId - unique id of post or comment
* @returns - the total number of votes for the given post or comment, as an ethers BigNumber
*/
async getVotes(itemId: ItemId): Promise<BigNumber> {
return this.#readonlyContract.getItemScore(itemId)
}
/**
* Stores an ItemContent object as JSON with web3.storage.
*
* @param c - an ItemContent object
* @returns - a promise that resoves to the CID of the content, encoded as a string
*/
async #storeItemContent(c: ItemContent): Promise<CIDString> {
const cid = await this.#storeAsJson(c)
this.#itemContentCache.set(cid, c)
return cid
}
/**
* Stores the given value as a JSON file with web3.storage.
*
* @param o - any JS value that can be converted to JSON
* @param filename - optional filename. Ignored if wrapWithDirectory == false.
* @param wrapWithDirectory - if true, wraps content with an IPFS directory listing. Defaults to false.
* @returns - a promise that resoves to the CID of the JSON file, encoded as a string.
* If wrapWithDirectory == true, the CID will be of the directory root, and the file is accessible at `${cid}/${filename}`
*/
async #storeAsJson(o: any, filename: string = 'file.json', wrapWithDirectory: boolean = false): Promise<CIDString> {
if (!this.#storage) {
throw new NoStorageClientError()
}
const str = JSON.stringify(o)
const file = new File([str], filename, { type: 'application/json' })
const cid = await this.#storage.put([file], { wrapWithDirectory })
console.log('stored json object. cid: ', cid, o)
return cid
}
/**
* Fetches a JSON file from web3.storage and parses it.
*
* @param cid - CID of JSON file object to fetch
* @returns - a promise that resolves to the parsed JSON object
*/
async #getJson(cid: string): Promise<any> {
const url = `https://${cid}.ipfs.dweb.link`
const res = await fetch(url)
if (!res || !res.ok) {
throw new Error(`Error getting cid ${cid}: [${res?.status}] ${res?.statusText}`)
}
return res.json()
}
}
//#region helpers
/**
* Tries to extract the post or comment from a `NewItem` event containing an `id` arg.
*
* @param events - an array of ethers `Event`s to search.
* @returns the value of `id` from the first `NewItem` event, or undefined if no event was found
*/
function idFromEvents(events: Event[] | undefined): ItemId | undefined {
if (!events) {
console.warn('cant get id from events: no events')
return
}
let id: string | undefined
for (const event of events) {
if (event.event !== 'NewItem' || !event.args) {
console.log('ignoring unexpected event', event.event)
continue
}
if (!('id' in event.args)) {
continue
}
id = event.args.id.toString()
}
return id
}
function allIdsFromEvents(events: Event[]): ItemId[] {
return events.filter(e => e.event === 'NewItem' && e.args && e.args.id != null)
.map(e => e.args!['id'])
}
//#endregion helpers
//#region types
export class NoStorageClientError extends Error {
static CODE = 'NO_STORAGE_CLIENT'
constructor() {
super('No storage client - make sure you have an API token configured for your account.')
}
}
export type ItemId = BigNumberish
export type Address = string
export type CIDString = string
export type IPFSPath = string
export enum ItemKind {
POST = 0,
COMMENT,
}
export interface Item {
kind: ItemKind,
id: ItemId,
parentId: ItemId,
author: Address,
childIds: ItemId[],
contentCID: CIDString,
createdAtBlock: BigNumber,
content: ItemContent,
score?: number,
}
export interface Post extends Item {
kind: ItemKind.POST
content: PostContent
}
export interface Comment extends Item {
kind: ItemKind.COMMENT
content: CommentContent
}
export type ItemContent = PostContent | CommentContent
export interface PostContent {
itemKind: 'POST',
title: string,
body: string,
}
export interface CommentContent {
itemKind: 'COMMENT'
parentId: ItemId,
body: string,
}
export const Upvote = 1
export const Downvote = -1
export const NoVote = 0
export type VoteValue = 1 | 0 | -1
/// the types below are for struct types defined in the contract
type ItemStruct = {
kind: number
id: BigNumber
parentId: BigNumber
author: string
createdAtBlock: BigNumber
childIds: ItemId[]
contentCID: string
}
//#endregion types