-
Notifications
You must be signed in to change notification settings - Fork 22
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
feat: access-api handles provider/add invocations #462
Changes from 19 commits
1c5d196
679e3c2
b8849a4
d15d0db
6da1479
25c4e0e
da6adb6
2f64d27
624b355
bd05cbd
fbb0ce4
27acf17
0c62dbc
e537211
d9768ad
6c1d1be
49004e8
e403bb3
237d188
e62faaa
2e90305
cd7dc57
184c55a
754463d
4a48928
a199c11
6903891
40a4f16
c1e37f9
9b9a14d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||
---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,21 @@ | ||||||||||
-- Migration number: 0006 2023-03-02T16:40:04.407Z | ||||||||||
|
||||||||||
/* | ||||||||||
goal: add a table to keep track of the storage provider for a space. | ||||||||||
to enable https://github.com/web3-storage/w3protocol/issues/459 | ||||||||||
*/ | ||||||||||
|
||||||||||
CREATE TABLE | ||||||||||
-- provision: the action of providing or supplying something for use | ||||||||||
-- use case: representing the registration of a storage provider to a space | ||||||||||
IF NOT EXISTS provisions ( | ||||||||||
id INTEGER NOT NULL PRIMARY KEY, | ||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Mind adding comment also what is the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's meant to just be an auto id that other tables could join to. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. added a comment. 184c55a fwiw it will not be autoincrementing it will be a 64-bit ROWID https://www.sqlite.org/lang_createtable.html#rowid There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can we still add the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. |
||||||||||
-- DID of the actor that is consuming the provider. e.g. a space DID | ||||||||||
consumer TEXT NOT NULL, | ||||||||||
-- DID of the provider e.g. a storage provider | ||||||||||
provider TEXT NOT NULL, | ||||||||||
-- DID of the actor that issued this provision | ||||||||||
issuer TEXT NOT NULL, | ||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm not sure I understand this, is it supposed to be an account here ? If so I think
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @Gozala The issuer isn't necessarily going to be billed, e.g. if there are several provisions for the same space. I don't want to attach any semantics to this other than 'the id of the thing that created this'. In ucan parlance that's
I can see how that might lead to confusion since the actor to attribute the capability to is not necessarily the final
The thing is I tried to make this table generic enough to support this field not necessarily being an account did nor a subscriber...
I also considered There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. changed to sponsor 754463d |
||||||||||
inserted_at TEXT NOT NULL DEFAULT (strftime ('%Y-%m-%dT%H:%M:%fZ', 'now')), | ||||||||||
updated_at TEXT NOT NULL DEFAULT (strftime ('%Y-%m-%dT%H:%M:%fZ', 'now')) | ||||||||||
); |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,99 @@ | ||
/* eslint-disable no-void */ | ||
|
||
/** | ||
* @typedef {import("../types/provisions").StorageProvisions} StorageProvisions | ||
*/ | ||
|
||
/** | ||
* @param {Array<import("../types/provisions").StorageProvisionCreation>} storage | ||
* @returns {StorageProvisions} | ||
*/ | ||
export function createStorageProvisions(storage = []) { | ||
/** @type {StorageProvisions['hasStorageProvider']} */ | ||
const hasStorageProvider = async (consumerId) => { | ||
const hasRowWithSpace = storage.some(({ space }) => space === consumerId) | ||
return hasRowWithSpace | ||
} | ||
/** @type {StorageProvisions['putMany']} */ | ||
const putMany = async (...items) => { | ||
storage.push(...items) | ||
} | ||
/** @type {StorageProvisions['count']} */ | ||
const count = async () => { | ||
return BigInt(storage.length) | ||
} | ||
return { | ||
count, | ||
putMany, | ||
hasStorageProvider, | ||
} | ||
} | ||
|
||
/** | ||
* @typedef ProvsionsRow | ||
* @property {string} consumer | ||
* @property {string} provider | ||
* @property {string} issuer - did of actor who did the provisioning | ||
*/ | ||
|
||
/** | ||
* @typedef {import("../types/database").Database<{ provisions: ProvsionsRow }>} ProvisionsDatabase | ||
*/ | ||
|
||
/** | ||
* StorageProvisions backed by a kyseli database (e.g. sqlite or cloudflare d1) | ||
*/ | ||
export class DbStorageProvisions { | ||
/** @type {ProvisionsDatabase} */ | ||
#db | ||
|
||
/** | ||
* @param {ProvisionsDatabase} db | ||
*/ | ||
constructor(db) { | ||
this.#db = db | ||
this.tableNames = { | ||
provisions: /** @type {const} */ ('provisions'), | ||
} | ||
void (/** @type {StorageProvisions} */ (this)) | ||
} | ||
|
||
/** @type {StorageProvisions['count']} */ | ||
async count(...items) { | ||
const { size } = await this.#db | ||
.selectFrom(this.tableNames.provisions) | ||
.select((e) => e.fn.count('provider').as('size')) | ||
.executeTakeFirstOrThrow() | ||
return BigInt(size) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What is it otherwise a string ? Can you plz add a comment explaining why this needs to be a BigInt. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I made it a BigInt to match the interface. I used bigint in the interface for consistency with the |
||
} | ||
|
||
/** @type {StorageProvisions['putMany']} */ | ||
async putMany(...items) { | ||
if (items.length === 0) { | ||
return | ||
} | ||
/** @type {ProvsionsRow[]} */ | ||
const rows = items.map((item) => { | ||
return { | ||
consumer: item.space, | ||
provider: item.provider, | ||
issuer: item.account, | ||
} | ||
}) | ||
await this.#db | ||
.insertInto(this.tableNames.provisions) | ||
.values(rows) | ||
.executeTakeFirstOrThrow() | ||
} | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think it would be nice to have method that gives you providers you have on space in shape of There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. By There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Yes |
||
/** @type {StorageProvisions['hasStorageProvider']} */ | ||
async hasStorageProvider(consumerDid) { | ||
const { provisions } = this.tableNames | ||
const { size } = await this.#db | ||
.selectFrom(provisions) | ||
.select((e) => e.fn.count('provider').as('size')) | ||
.where(`${provisions}.consumer`, '=', consumerDid) | ||
.executeTakeFirstOrThrow() | ||
return size > 0 | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,5 @@ | ||
import * as ucanto from '@ucanto/core' | ||
import * as Ucanto from '@ucanto/interface' | ||
import * as Server from '@ucanto/server' | ||
import { Failure } from '@ucanto/server' | ||
import * as Space from '@web3-storage/capabilities/space' | ||
|
@@ -13,6 +14,7 @@ import * as uploadApi from './upload-api-proxy.js' | |
import { accessAuthorizeProvider } from './access-authorize.js' | ||
import { accessDelegateProvider } from './access-delegate.js' | ||
import { accessClaimProvider } from './access-claim.js' | ||
import { providerAddProvider } from './provider-add.js' | ||
|
||
/** | ||
* @param {import('../bindings').RouteContext} ctx | ||
|
@@ -22,6 +24,12 @@ import { accessClaimProvider } from './access-claim.js' | |
* } | ||
*/ | ||
export function service(ctx) { | ||
/** | ||
* @param {Ucanto.DID<'key'>} uri | ||
*/ | ||
const hasStorageProvider = async (uri) => { | ||
return Boolean(await ctx.models.spaces.get(uri)) | ||
} | ||
return { | ||
store: uploadApi.createStoreProxy(ctx), | ||
upload: uploadApi.createUploadProxy(ctx), | ||
|
@@ -45,12 +53,21 @@ export function service(ctx) { | |
} | ||
return accessDelegateProvider({ | ||
delegations: ctx.models.delegations, | ||
hasStorageProvider: async (uri) => { | ||
return Boolean(await ctx.models.spaces.get(uri)) | ||
}, | ||
hasStorageProvider, | ||
})(...args) | ||
}, | ||
}, | ||
|
||
provider: { | ||
add: (...args) => { | ||
// disable until hardened in test/staging | ||
if (ctx.config.ENV === 'production') { | ||
throw new Error(`provider/add invocation handling is not enabled`) | ||
} | ||
return providerAddProvider(ctx)(...args) | ||
}, | ||
}, | ||
|
||
voucher: { | ||
claim: voucherClaimProvider(ctx), | ||
redeem: voucherRedeemProvider(ctx), | ||
|
@@ -181,6 +198,18 @@ export function service(ctx) { | |
fail() { | ||
throw new Error('test fail') | ||
}, | ||
/** | ||
* @param {Ucanto.Invocation<Ucanto.Capability<'testing/space-storage', Ucanto.DID<'key'>, Ucanto.Failure>>} invocation | ||
*/ | ||
'space-storage': async (invocation) => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can we just amend There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. yeah. I thought about it but didn't want to extend the public api without discussion. In this case I am very supportive because that will also be helpful to upload-api and things like storacha/w3infra#134 (comment) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. can I do it as a fast follow once I have verified that the space registration via |
||
const spaceId = invocation.capabilities[0].with | ||
const hasStorageProvider = | ||
await ctx.models.storageProvisions.hasStorageProvider(spaceId) | ||
return { | ||
hasStorageProvider, | ||
foo: 'ben', | ||
} | ||
}, | ||
}, | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,58 @@ | ||
import * as Ucanto from '@ucanto/interface' | ||
import * as Server from '@ucanto/server' | ||
import { Provider } from '@web3-storage/capabilities' | ||
import * as validator from '@ucanto/validator' | ||
|
||
/** | ||
* @typedef {import('@web3-storage/capabilities/types').ProviderAdd} ProviderAdd | ||
* @typedef {import('@web3-storage/capabilities/types').ProviderAddSuccess} ProviderAddSuccess | ||
* @typedef {import('@web3-storage/capabilities/types').ProviderAddFailure} ProviderAddFailure | ||
*/ | ||
|
||
/** | ||
* @callback ProviderAddHandler | ||
* @param {Ucanto.Invocation<import('@web3-storage/capabilities/types').ProviderAdd>} invocation | ||
* @returns {Promise<Ucanto.Result<ProviderAddSuccess, ProviderAddFailure>>} | ||
*/ | ||
|
||
/** | ||
* @param {object} options | ||
* @param {import('../types/provisions').StorageProvisions} options.storageProvisions | ||
* @returns {ProviderAddHandler} | ||
*/ | ||
export function createProviderAddHandler(options) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Nit: This obviously follows the style of code that is in the repo already, but I do find amount of indirection adds so much unnecessary complexity e.g. I would much rather have something like: /**
* @param {object} input
* @param {Ucanto.Invocation<import('@web3-storage/capabilities/types').ProviderAdd>} input.invocation
* @param {{ storageProvisions: import('../types/provisions').StorageProvisions }} input.context
*/
export const add ({ invocation, context: { storageProvisions } }) => {
// ...
} And in the service defs something like Server.provide(Provider.add, ({ invocation }) => add({ invocation, context }) Than all this layers and closures all over. Also for what it's worth I'm going to add There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this is a good idea and I'll try it out across all the relevant things I wrote in that style to use the less-closures style you suggest. #481 |
||
/** @type {ProviderAddHandler} */ | ||
return async (invocation) => { | ||
const [providerAddCap] = invocation.capabilities | ||
const { | ||
nb: { consumer, provider }, | ||
with: accountDID, | ||
} = providerAddCap | ||
if (!validator.DID.match({ method: 'mailto' }).is(accountDID)) { | ||
return { | ||
error: true, | ||
name: 'Unauthorized', | ||
message: 'Issuer must be a mailto DID', | ||
} | ||
} | ||
await options.storageProvisions.putMany({ | ||
space: consumer, | ||
provider, | ||
account: accountDID, | ||
}) | ||
return {} | ||
} | ||
} | ||
|
||
/** | ||
* @param {object} ctx | ||
* @param {Pick<import('../bindings').RouteContext['models'], 'storageProvisions'>} ctx.models | ||
*/ | ||
export function providerAddProvider(ctx) { | ||
return Server.provide(Provider.add, async ({ invocation }) => { | ||
const handler = createProviderAddHandler({ | ||
storageProvisions: ctx.models.storageProvisions, | ||
}) | ||
return handler(/** @type {Ucanto.Invocation<ProviderAdd>} */ (invocation)) | ||
}) | ||
} |
Original file line number | Diff line number | Diff line change | ||||||||
---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,30 @@ | ||||||||||
import * as Ucanto from '@ucanto/interface' | ||||||||||
|
||||||||||
export type AlphaStorageProvider = 'did:web:web3.storage:providers:w3up-alpha' | ||||||||||
|
||||||||||
/** | ||||||||||
* action which results in provisionment of a space consuming a storage provider | ||||||||||
*/ | ||||||||||
export interface StorageProvisionCreation { | ||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Nit: I would prefer if we did not had There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. At this layer of the types I wanted to model this particular use case really explicitly (e.g. with
Because the underlying table schema accomodates this, I don't want to change this particular action type now. I'm not against revising it or adding other events in subsequent PR. or e.g. renaming the model to not have storage prefix but still maybe having this particular event be specific to adding a storage provider. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I removed 'Storage' prefix from the model object itself e62faaa There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Nit: I find naming to be very confusing, can we simply call this
Suggested change
Or better yet
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. |
||||||||||
space: Ucanto.DID<'key'> | ||||||||||
account: Ucanto.DID<'mailto'> | ||||||||||
provider: AlphaStorageProvider | ||||||||||
} | ||||||||||
|
||||||||||
/** | ||||||||||
* stores instances of a storage provider being consumed by a consumer | ||||||||||
*/ | ||||||||||
export interface StorageProvisions { | ||||||||||
hasStorageProvider: (consumer: Ucanto.DID<'key'>) => Promise<boolean> | ||||||||||
/** | ||||||||||
* write several items into storage | ||||||||||
* | ||||||||||
* @param items - provisions to store | ||||||||||
*/ | ||||||||||
putMany: (...items: StorageProvisionCreation[]) => Promise<void> | ||||||||||
|
||||||||||
/** | ||||||||||
* get number of stored items | ||||||||||
*/ | ||||||||||
count: () => Promise<bigint> | ||||||||||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nit: In the past when I had to do more DB work using plural names for tables was considered a bad practice, stack overflow still seems to think the singular is a the way to go with bunch of reasons there.
I'm not going to get upset if we use plural, but might be worth considering singular names instead.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
in general I don't mind singular at all, and maybe even slightly prefer it, but here I made it plural because
delegations
accounts
spaces
already was.I don't have any objection to batch changing them to singular in one big other PR