forked from storacha/w3up
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat!: add
index/add
handler (storacha#1421)
Adds a Ucanto handler for `index/add` invocations. * Fetches index archive from the network * Ensures index and all DAG shards are stored in the agent's space * Publishes to IPNI refs storacha#1401 --------- Co-authored-by: Irakli Gozalishvili <contact@gozala.io>
- Loading branch information
Showing
24 changed files
with
828 additions
and
19 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
import { provide } from './index/add.js' | ||
import * as API from './types.js' | ||
|
||
/** @param {API.IndexServiceContext} context */ | ||
export const createService = (context) => ({ add: provide(context) }) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,78 @@ | ||
import * as Server from '@ucanto/server' | ||
import { ok, error } from '@ucanto/server' | ||
import * as Index from '@web3-storage/capabilities/index' | ||
import * as ShardedDAGIndex from './lib/sharded-dag-index.js' | ||
import * as API from '../types.js' | ||
|
||
/** | ||
* @param {API.IndexServiceContext} context | ||
* @returns {API.ServiceMethod<API.IndexAdd, API.IndexAddSuccess, API.IndexAddFailure>} | ||
*/ | ||
export const provide = (context) => | ||
Server.provide(Index.add, (input) => add(input, context)) | ||
|
||
/** | ||
* @param {API.Input<Index.add>} input | ||
* @param {API.IndexServiceContext} context | ||
* @returns {Promise<API.Result<API.IndexAddSuccess, API.IndexAddFailure>>} | ||
*/ | ||
const add = async ({ capability }, context) => { | ||
const space = capability.with | ||
const idxLink = capability.nb.index | ||
|
||
// ensure the index was stored in the agent's space | ||
const idxAllocRes = await assertAllocated( | ||
context, | ||
space, | ||
idxLink.multihash, | ||
'IndexNotFound' | ||
) | ||
if (!idxAllocRes.ok) return idxAllocRes | ||
|
||
// fetch the index from the network | ||
const idxBlobRes = await context.blobRetriever.stream(idxLink.multihash) | ||
if (!idxBlobRes.ok) { | ||
if (idxBlobRes.error.name === 'BlobNotFound') { | ||
return error( | ||
/** @type {API.IndexNotFound} */ | ||
({ name: 'IndexNotFound', digest: idxLink.multihash.bytes }) | ||
) | ||
} | ||
return idxBlobRes | ||
} | ||
|
||
const idxRes = await ShardedDAGIndex.extract(idxBlobRes.ok) | ||
if (!idxRes.ok) return idxAllocRes | ||
|
||
// ensure indexed shards are allocated in the agent's space | ||
const shardDigests = [...idxRes.ok.shards.keys()] | ||
const shardAllocRes = await Promise.all( | ||
shardDigests.map((s) => assertAllocated(context, space, s, 'ShardNotFound')) | ||
) | ||
for (const res of shardAllocRes) { | ||
if (!res.ok) return res | ||
} | ||
|
||
// TODO: randomly validate slices in the index correspond to slices in the blob | ||
|
||
// publish the index data to IPNI | ||
return context.ipniService.publish(idxRes.ok) | ||
} | ||
|
||
/** | ||
* @param {{ allocationsStorage: import('../types.js').AllocationsStorage }} context | ||
* @param {API.SpaceDID} space | ||
* @param {import('multiformats').MultihashDigest} digest | ||
* @param {'IndexNotFound'|'ShardNotFound'|'SliceNotFound'} errorName | ||
* @returns {Promise<API.Result<API.Unit, API.IndexNotFound|API.ShardNotFound|API.SliceNotFound|API.Failure>>} | ||
*/ | ||
const assertAllocated = async (context, space, digest, errorName) => { | ||
const result = await context.allocationsStorage.exists(space, digest.bytes) | ||
if (result.error) return result | ||
if (!result.ok) | ||
return error( | ||
/** @type {API.IndexNotFound|API.ShardNotFound|API.SliceNotFound} */ | ||
({ name: errorName, digest: digest.bytes }) | ||
) | ||
return ok({}) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export {} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
import { Failure } from '@ucanto/interface' | ||
import { MultihashDigest, UnknownLink } from 'multiformats' | ||
|
||
export type { Result } from '@ucanto/interface' | ||
export type { UnknownFormat } from '@web3-storage/capabilities/types' | ||
export type { MultihashDigest, UnknownLink } | ||
|
||
export type ShardDigest = MultihashDigest | ||
export type SliceDigest = MultihashDigest | ||
|
||
/** | ||
* A sharded DAG index. | ||
* | ||
* @see https://github.com/w3s-project/specs/blob/main/w3-index.md | ||
*/ | ||
export interface ShardedDAGIndex { | ||
content: UnknownLink | ||
shards: Map<ShardDigest, Map<SliceDigest, [offset: number, length: number]>> | ||
} | ||
|
||
export interface DecodeFailure extends Failure { | ||
name: 'DecodeFailure' | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,116 @@ | ||
import { base58btc } from 'multiformats/bases/base58' | ||
|
||
/** @type {WeakMap<Uint8Array, string>} */ | ||
const cache = new WeakMap() | ||
|
||
/** @param {import('multiformats').MultihashDigest} digest */ | ||
const toBase58String = (digest) => { | ||
let str = cache.get(digest.bytes) | ||
if (!str) { | ||
str = base58btc.encode(digest.bytes) | ||
cache.set(digest.bytes, str) | ||
} | ||
return str | ||
} | ||
|
||
/** | ||
* @template {import('multiformats').MultihashDigest} Key | ||
* @template Value | ||
* @implements {Map<Key, Value>} | ||
*/ | ||
export class DigestMap { | ||
/** @type {Map<string, [Key, Value]>} */ | ||
#data | ||
|
||
/** | ||
* @param {Array<[Key, Value]>} [entries] | ||
*/ | ||
constructor(entries) { | ||
this.#data = new Map() | ||
for (const [k, v] of entries ?? []) { | ||
this.set(k, v) | ||
} | ||
} | ||
|
||
get [Symbol.toStringTag]() { | ||
return 'DigestMap' | ||
} | ||
|
||
clear() { | ||
this.#data.clear() | ||
} | ||
|
||
/** | ||
* @param {Key} key | ||
* @returns {boolean} | ||
*/ | ||
delete(key) { | ||
const mhstr = toBase58String(key) | ||
return this.#data.delete(mhstr) | ||
} | ||
|
||
/** | ||
* @param {(value: Value, key: Key, map: Map<Key, Value>) => void} callbackfn | ||
* @param {any} [thisArg] | ||
*/ | ||
forEach(callbackfn, thisArg) { | ||
for (const [k, v] of this.#data.values()) { | ||
callbackfn.call(thisArg, v, k, this) | ||
} | ||
} | ||
|
||
/** | ||
* @param {Key} key | ||
* @returns {Value|undefined} | ||
*/ | ||
get(key) { | ||
const data = this.#data.get(toBase58String(key)) | ||
if (data) return data[1] | ||
} | ||
|
||
/** | ||
* @param {Key} key | ||
* @returns {boolean} | ||
*/ | ||
has(key) { | ||
return this.#data.has(toBase58String(key)) | ||
} | ||
|
||
/** | ||
* @param {Key} key | ||
* @param {Value} value | ||
*/ | ||
set(key, value) { | ||
this.#data.set(toBase58String(key), [key, value]) | ||
return this | ||
} | ||
|
||
/** @returns {number} */ | ||
get size() { | ||
return this.#data.size | ||
} | ||
|
||
/** @returns */ | ||
[Symbol.iterator]() { | ||
return this.entries() | ||
} | ||
|
||
/** @returns {IterableIterator<[Key, Value]>} */ | ||
*entries() { | ||
yield* this.#data.values() | ||
} | ||
|
||
/** @returns {IterableIterator<Key>} */ | ||
*keys() { | ||
for (const [k] of this.#data.values()) { | ||
yield k | ||
} | ||
} | ||
|
||
/** @returns {IterableIterator<Value>} */ | ||
*values() { | ||
for (const [, v] of this.#data.values()) { | ||
yield v | ||
} | ||
} | ||
} |
Oops, something went wrong.