Compare commits
10 Commits
1abfd74ec7
...
3f93d8cabf
Author | SHA1 | Date | |
---|---|---|---|
|
3f93d8cabf | ||
|
17057144d8 | ||
|
a44db38d05 | ||
|
e277158f70 | ||
|
d97272de0b | ||
|
5ece8c6aea | ||
|
2889c76995 | ||
|
48a0e9d606 | ||
|
9dc7251fc7 | ||
|
5b6e0611d6 |
@ -1,5 +0,0 @@
|
|||||||
---
|
|
||||||
"@atproto/crypto": patch
|
|
||||||
---
|
|
||||||
|
|
||||||
Update noble crypto libraries
|
|
@ -30,6 +30,10 @@
|
|||||||
"type": "ref",
|
"type": "ref",
|
||||||
"ref": "app.bsky.actor.defs#profileView"
|
"ref": "app.bsky.actor.defs#profileView"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"recId": {
|
||||||
|
"type": "integer",
|
||||||
|
"description": "Snowflake for this recommendation, use when submitting recommendation events."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -29,6 +29,10 @@
|
|||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"description": "If true, response has fallen-back to generic results, and is not scoped using relativeToDid",
|
"description": "If true, response has fallen-back to generic results, and is not scoped using relativeToDid",
|
||||||
"default": false
|
"default": false
|
||||||
|
},
|
||||||
|
"recId": {
|
||||||
|
"type": "integer",
|
||||||
|
"description": "Snowflake for this recommendation, use when submitting recommendation events."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -45,6 +45,10 @@
|
|||||||
"type": "string",
|
"type": "string",
|
||||||
"format": "did",
|
"format": "did",
|
||||||
"description": "DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer."
|
"description": "DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer."
|
||||||
|
},
|
||||||
|
"recId": {
|
||||||
|
"type": "integer",
|
||||||
|
"description": "Snowflake for this recommendation, use when submitting recommendation events."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,11 @@
|
|||||||
# @atproto/api
|
# @atproto/api
|
||||||
|
|
||||||
|
## 0.13.27
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- [#3364](https://github.com/bluesky-social/atproto/pull/3364) [`e277158f7`](https://github.com/bluesky-social/atproto/commit/e277158f70a831b04fde3ec84b3c1eaa6ce82e9d) Thanks [@iwsmith](https://github.com/iwsmith)! - add recId to getSuggestions
|
||||||
|
|
||||||
## 0.13.26
|
## 0.13.26
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/api",
|
"name": "@atproto/api",
|
||||||
"version": "0.13.26",
|
"version": "0.13.27",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Client library for atproto and Bluesky",
|
"description": "Client library for atproto and Bluesky",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -12359,7 +12359,7 @@ export const schemaDict = {
|
|||||||
items: {
|
items: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
description:
|
description:
|
||||||
'If specified, only events where the policy matches the given policy are returned',
|
'If specified, only events where the action policies match any of the given policies are returned',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
cursor: {
|
cursor: {
|
||||||
|
@ -18,6 +18,8 @@ export type InputSchema = undefined
|
|||||||
export interface OutputSchema {
|
export interface OutputSchema {
|
||||||
cursor?: string
|
cursor?: string
|
||||||
actors: AppBskyActorDefs.ProfileView[]
|
actors: AppBskyActorDefs.ProfileView[]
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,6 +18,8 @@ export interface OutputSchema {
|
|||||||
suggestions: AppBskyActorDefs.ProfileView[]
|
suggestions: AppBskyActorDefs.ProfileView[]
|
||||||
/** If true, response has fallen-back to generic results, and is not scoped using relativeToDid */
|
/** If true, response has fallen-back to generic results, and is not scoped using relativeToDid */
|
||||||
isFallback: boolean
|
isFallback: boolean
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,6 +24,8 @@ export interface OutputSchema {
|
|||||||
actors: AppBskyUnspeccedDefs.SkeletonSearchActor[]
|
actors: AppBskyUnspeccedDefs.SkeletonSearchActor[]
|
||||||
/** DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer. */
|
/** DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer. */
|
||||||
relativeToDid?: string
|
relativeToDid?: string
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,13 @@
|
|||||||
# @atproto/aws
|
# @atproto/aws
|
||||||
|
|
||||||
|
## 0.2.12
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277)]:
|
||||||
|
- @atproto/crypto@0.4.3
|
||||||
|
- @atproto/repo@0.6.2
|
||||||
|
|
||||||
## 0.2.11
|
## 0.2.11
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/aws",
|
"name": "@atproto/aws",
|
||||||
"version": "0.2.11",
|
"version": "0.2.12",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Shared AWS cloud API helpers for atproto services",
|
"description": "Shared AWS cloud API helpers for atproto services",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,25 @@
|
|||||||
# @atproto/bsky
|
# @atproto/bsky
|
||||||
|
|
||||||
|
## 0.0.106
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`e277158f7`](https://github.com/bluesky-social/atproto/commit/e277158f70a831b04fde3ec84b3c1eaa6ce82e9d)]:
|
||||||
|
- @atproto/api@0.13.27
|
||||||
|
- @atproto-labs/fetch-node@0.1.5
|
||||||
|
|
||||||
|
## 0.0.105
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277)]:
|
||||||
|
- @atproto/crypto@0.4.3
|
||||||
|
- @atproto/identity@0.4.5
|
||||||
|
- @atproto/repo@0.6.2
|
||||||
|
- @atproto/xrpc-server@0.7.6
|
||||||
|
- @atproto/sync@0.1.9
|
||||||
|
- @atproto-labs/xrpc-utils@0.0.2
|
||||||
|
|
||||||
## 0.0.104
|
## 0.0.104
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/bsky",
|
"name": "@atproto/bsky",
|
||||||
"version": "0.0.104",
|
"version": "0.0.106",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Reference implementation of app.bsky App View (Bluesky API)",
|
"description": "Reference implementation of app.bsky App View (Bluesky API)",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -71,6 +71,7 @@ const skeleton = async (input: {
|
|||||||
return {
|
return {
|
||||||
dids: res.data.actors.map((a) => a.did),
|
dids: res.data.actors.map((a) => a.did),
|
||||||
cursor: res.data.cursor,
|
cursor: res.data.cursor,
|
||||||
|
recId: res.data.recId,
|
||||||
resHeaders: res.headers,
|
resHeaders: res.headers,
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -129,6 +130,7 @@ const presentation = (input: {
|
|||||||
return {
|
return {
|
||||||
actors,
|
actors,
|
||||||
cursor: skeleton.cursor,
|
cursor: skeleton.cursor,
|
||||||
|
recId: skeleton.recId,
|
||||||
resHeaders: skeleton.resHeaders,
|
resHeaders: skeleton.resHeaders,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -148,5 +150,6 @@ type Params = QueryParams & {
|
|||||||
type Skeleton = {
|
type Skeleton = {
|
||||||
dids: string[]
|
dids: string[]
|
||||||
cursor?: string
|
cursor?: string
|
||||||
|
recId?: number
|
||||||
resHeaders?: Record<string, string>
|
resHeaders?: Record<string, string>
|
||||||
}
|
}
|
||||||
|
@ -73,6 +73,7 @@ const skeleton = async (input: SkeletonFnInput<Context, Params>) => {
|
|||||||
return {
|
return {
|
||||||
isFallback: !res.data.relativeToDid,
|
isFallback: !res.data.relativeToDid,
|
||||||
suggestedDids: res.data.actors.map((a) => a.did),
|
suggestedDids: res.data.actors.map((a) => a.did),
|
||||||
|
recId: res.data.recId,
|
||||||
headers: res.headers,
|
headers: res.headers,
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -115,7 +116,12 @@ const presentation = (
|
|||||||
const suggestions = mapDefined(suggestedDids, (did) =>
|
const suggestions = mapDefined(suggestedDids, (did) =>
|
||||||
ctx.views.profileDetailed(did, hydration),
|
ctx.views.profileDetailed(did, hydration),
|
||||||
)
|
)
|
||||||
return { isFallback: skeleton.isFallback, suggestions, headers }
|
return {
|
||||||
|
isFallback: skeleton.isFallback,
|
||||||
|
suggestions,
|
||||||
|
recId: skeleton.recId,
|
||||||
|
headers,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type Context = {
|
type Context = {
|
||||||
@ -133,5 +139,6 @@ type Params = QueryParams & {
|
|||||||
type SkeletonState = {
|
type SkeletonState = {
|
||||||
isFallback: boolean
|
isFallback: boolean
|
||||||
suggestedDids: string[]
|
suggestedDids: string[]
|
||||||
|
recId?: number
|
||||||
headers?: Record<string, string>
|
headers?: Record<string, string>
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,8 @@ export type InputSchema = undefined
|
|||||||
export interface OutputSchema {
|
export interface OutputSchema {
|
||||||
cursor?: string
|
cursor?: string
|
||||||
actors: AppBskyActorDefs.ProfileView[]
|
actors: AppBskyActorDefs.ProfileView[]
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,6 +19,8 @@ export interface OutputSchema {
|
|||||||
suggestions: AppBskyActorDefs.ProfileView[]
|
suggestions: AppBskyActorDefs.ProfileView[]
|
||||||
/** If true, response has fallen-back to generic results, and is not scoped using relativeToDid */
|
/** If true, response has fallen-back to generic results, and is not scoped using relativeToDid */
|
||||||
isFallback?: boolean
|
isFallback?: boolean
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,6 +25,8 @@ export interface OutputSchema {
|
|||||||
actors: AppBskyUnspeccedDefs.SkeletonSearchActor[]
|
actors: AppBskyUnspeccedDefs.SkeletonSearchActor[]
|
||||||
/** DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer. */
|
/** DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer. */
|
||||||
relativeToDid?: string
|
relativeToDid?: string
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,11 @@
|
|||||||
# @atproto/crypto
|
# @atproto/crypto
|
||||||
|
|
||||||
|
## 0.4.3
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- [#3335](https://github.com/bluesky-social/atproto/pull/3335) [`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277) Thanks [@dholms](https://github.com/dholms)! - Update noble crypto libraries
|
||||||
|
|
||||||
## 0.4.2
|
## 0.4.2
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/crypto",
|
"name": "@atproto/crypto",
|
||||||
"version": "0.4.2",
|
"version": "0.4.3",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Library for cryptographic keys and signing in atproto",
|
"description": "Library for cryptographic keys and signing in atproto",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,30 @@
|
|||||||
# @atproto/dev-env
|
# @atproto/dev-env
|
||||||
|
|
||||||
|
## 0.3.76
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- [#3344](https://github.com/bluesky-social/atproto/pull/3344) [`48a0e9d60`](https://github.com/bluesky-social/atproto/commit/48a0e9d6060c2dc93899f13f2fc7cc76c04fbcd9) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Properly dispose of unused http responses
|
||||||
|
|
||||||
|
- Updated dependencies [[`48a0e9d60`](https://github.com/bluesky-social/atproto/commit/48a0e9d6060c2dc93899f13f2fc7cc76c04fbcd9), [`e277158f7`](https://github.com/bluesky-social/atproto/commit/e277158f70a831b04fde3ec84b3c1eaa6ce82e9d)]:
|
||||||
|
- @atproto/ozone@0.1.67
|
||||||
|
- @atproto/api@0.13.27
|
||||||
|
- @atproto/bsky@0.0.106
|
||||||
|
- @atproto/pds@0.4.84
|
||||||
|
|
||||||
|
## 0.3.75
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277)]:
|
||||||
|
- @atproto/crypto@0.4.3
|
||||||
|
- @atproto/bsky@0.0.105
|
||||||
|
- @atproto/identity@0.4.5
|
||||||
|
- @atproto/ozone@0.1.66
|
||||||
|
- @atproto/pds@0.4.83
|
||||||
|
- @atproto/xrpc-server@0.7.6
|
||||||
|
- @atproto/sync@0.1.9
|
||||||
|
|
||||||
## 0.3.74
|
## 0.3.74
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/dev-env",
|
"name": "@atproto/dev-env",
|
||||||
"version": "0.3.74",
|
"version": "0.3.76",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Local development environment helper for atproto development",
|
"description": "Local development environment helper for atproto development",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -45,7 +45,7 @@ export const mockResolvers = (idResolver: IdResolver, pds: TestPds) => {
|
|||||||
try {
|
try {
|
||||||
const res = await request(url, { headers: { host: handle } })
|
const res = await request(url, { headers: { host: handle } })
|
||||||
if (res.statusCode !== 200) {
|
if (res.statusCode !== 200) {
|
||||||
res.body.destroy()
|
await res.body.dump()
|
||||||
return undefined
|
return undefined
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto/identity
|
# @atproto/identity
|
||||||
|
|
||||||
|
## 0.4.5
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277)]:
|
||||||
|
- @atproto/crypto@0.4.3
|
||||||
|
|
||||||
## 0.4.4
|
## 0.4.4
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/identity",
|
"name": "@atproto/identity",
|
||||||
"version": "0.4.4",
|
"version": "0.4.5",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Library for decentralized identities in atproto using DIDs and handles",
|
"description": "Library for decentralized identities in atproto using DIDs and handles",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto-labs/did-resolver
|
# @atproto-labs/did-resolver
|
||||||
|
|
||||||
|
## 0.1.8
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2), [`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2)]:
|
||||||
|
- @atproto-labs/fetch@0.2.0
|
||||||
|
|
||||||
## 0.1.7
|
## 0.1.7
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto-labs/did-resolver",
|
"name": "@atproto-labs/did-resolver",
|
||||||
"version": "0.1.7",
|
"version": "0.1.8",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "DID resolution and verification library",
|
"description": "DID resolution and verification library",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto-labs/fetch-node
|
# @atproto-labs/fetch-node
|
||||||
|
|
||||||
|
## 0.1.5
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2), [`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2)]:
|
||||||
|
- @atproto-labs/fetch@0.2.0
|
||||||
|
|
||||||
## 0.1.4
|
## 0.1.4
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto-labs/fetch-node",
|
"name": "@atproto-labs/fetch-node",
|
||||||
"version": "0.1.4",
|
"version": "0.1.5",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "SSRF protection for fetch() in Node.js",
|
"description": "SSRF protection for fetch() in Node.js",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,15 @@
|
|||||||
# @atproto-labs/fetch
|
# @atproto-labs/fetch
|
||||||
|
|
||||||
|
## 0.2.0
|
||||||
|
|
||||||
|
### Minor Changes
|
||||||
|
|
||||||
|
- [#3343](https://github.com/bluesky-social/atproto/pull/3343) [`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Fix typo in `ResponseTranformer` and `fetchResponseJsonTranformer`
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- [#3343](https://github.com/bluesky-social/atproto/pull/3343) [`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Response mime type check is now case-insensitive (as per rfc2616)
|
||||||
|
|
||||||
## 0.1.2
|
## 0.1.2
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto-labs/fetch",
|
"name": "@atproto-labs/fetch",
|
||||||
"version": "0.1.2",
|
"version": "0.2.0",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Isomorphic wrapper utilities for fetch API",
|
"description": "Isomorphic wrapper utilities for fetch API",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -14,7 +14,24 @@ import {
|
|||||||
logCancellationError,
|
logCancellationError,
|
||||||
} from './util.js'
|
} from './util.js'
|
||||||
|
|
||||||
export type ResponseTranformer = Transformer<Response>
|
/**
|
||||||
|
* media-type = type "/" subtype *( ";" parameter )
|
||||||
|
* type = token
|
||||||
|
* subtype = token
|
||||||
|
* token = 1*<any CHAR except CTLs or separators>
|
||||||
|
* separators = "(" | ")" | "<" | ">" | "@"
|
||||||
|
* | "," | ";" | ":" | "\" | <">
|
||||||
|
* | "/" | "[" | "]" | "?" | "="
|
||||||
|
* | "{" | "}" | SP | HT
|
||||||
|
* CTL = <any US-ASCII control character (octets 0 - 31) and DEL (127)>
|
||||||
|
* SP = <US-ASCII SP, space (32)>
|
||||||
|
* HT = <US-ASCII HT, horizontal-tab (9)>
|
||||||
|
* @note The type, subtype, and parameter attribute names are case-insensitive.
|
||||||
|
* @see {@link https://datatracker.ietf.org/doc/html/rfc2616#autoid-23}
|
||||||
|
*/
|
||||||
|
const JSON_MIME = /^application\/(?:[^()<>@,;:/[\]\\?={} \t]+\+)?json$/i
|
||||||
|
|
||||||
|
export type ResponseTransformer = Transformer<Response>
|
||||||
export type ResponseMessageGetter = Transformer<Response, string | undefined>
|
export type ResponseMessageGetter = Transformer<Response, string | undefined>
|
||||||
|
|
||||||
export class FetchResponseError extends FetchError {
|
export class FetchResponseError extends FetchError {
|
||||||
@ -51,7 +68,7 @@ const extractResponseMessage: ResponseMessageGetter = async (response) => {
|
|||||||
try {
|
try {
|
||||||
if (mimeType === 'text/plain') {
|
if (mimeType === 'text/plain') {
|
||||||
return await response.text()
|
return await response.text()
|
||||||
} else if (/^application\/(?:[^+]+\+)?json$/i.test(mimeType)) {
|
} else if (JSON_MIME.test(mimeType)) {
|
||||||
const json: unknown = await response.json()
|
const json: unknown = await response.json()
|
||||||
|
|
||||||
if (typeof json === 'string') return json
|
if (typeof json === 'string') return json
|
||||||
@ -155,7 +172,7 @@ export function cancelBodyOnError<T>(
|
|||||||
|
|
||||||
export function fetchOkProcessor(
|
export function fetchOkProcessor(
|
||||||
customMessage?: string | ResponseMessageGetter,
|
customMessage?: string | ResponseMessageGetter,
|
||||||
): ResponseTranformer {
|
): ResponseTransformer {
|
||||||
return cancelBodyOnError((response) => {
|
return cancelBodyOnError((response) => {
|
||||||
return fetchOkTransformer(response, customMessage)
|
return fetchOkTransformer(response, customMessage)
|
||||||
})
|
})
|
||||||
@ -169,7 +186,7 @@ export async function fetchOkTransformer(
|
|||||||
throw await FetchResponseError.from(response, customMessage)
|
throw await FetchResponseError.from(response, customMessage)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function fetchMaxSizeProcessor(maxBytes: number): ResponseTranformer {
|
export function fetchMaxSizeProcessor(maxBytes: number): ResponseTransformer {
|
||||||
if (maxBytes === Infinity) return (response) => response
|
if (maxBytes === Infinity) return (response) => response
|
||||||
if (!Number.isFinite(maxBytes) || maxBytes < 0) {
|
if (!Number.isFinite(maxBytes) || maxBytes < 0) {
|
||||||
throw new TypeError('maxBytes must be a 0, Infinity or a positive number')
|
throw new TypeError('maxBytes must be a 0, Infinity or a positive number')
|
||||||
@ -200,7 +217,7 @@ export type MimeTypeCheck = string | RegExp | MimeTypeCheckFn
|
|||||||
export function fetchTypeProcessor(
|
export function fetchTypeProcessor(
|
||||||
expectedMime: MimeTypeCheck,
|
expectedMime: MimeTypeCheck,
|
||||||
contentTypeRequired = true,
|
contentTypeRequired = true,
|
||||||
): ResponseTranformer {
|
): ResponseTransformer {
|
||||||
const isExpected: MimeTypeCheckFn =
|
const isExpected: MimeTypeCheckFn =
|
||||||
typeof expectedMime === 'string'
|
typeof expectedMime === 'string'
|
||||||
? (mimeType) => mimeType === expectedMime
|
? (mimeType) => mimeType === expectedMime
|
||||||
@ -220,7 +237,7 @@ export async function fetchResponseTypeChecker(
|
|||||||
): Promise<Response> {
|
): Promise<Response> {
|
||||||
const mimeType = extractMime(response)
|
const mimeType = extractMime(response)
|
||||||
if (mimeType) {
|
if (mimeType) {
|
||||||
if (!isExpectedMime(mimeType)) {
|
if (!isExpectedMime(mimeType.toLowerCase())) {
|
||||||
throw await FetchResponseError.from(
|
throw await FetchResponseError.from(
|
||||||
response,
|
response,
|
||||||
`Unexpected response Content-Type (${mimeType})`,
|
`Unexpected response Content-Type (${mimeType})`,
|
||||||
@ -243,7 +260,7 @@ export type ParsedJsonResponse<T = Json> = {
|
|||||||
json: T
|
json: T
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function fetchResponseJsonTranformer<T = Json>(
|
export async function fetchResponseJsonTransformer<T = Json>(
|
||||||
response: Response,
|
response: Response,
|
||||||
): Promise<ParsedJsonResponse<T>> {
|
): Promise<ParsedJsonResponse<T>> {
|
||||||
try {
|
try {
|
||||||
@ -260,12 +277,12 @@ export async function fetchResponseJsonTranformer<T = Json>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function fetchJsonProcessor<T = Json>(
|
export function fetchJsonProcessor<T = Json>(
|
||||||
expectedMime: MimeTypeCheck = /^application\/(?:[^+]+\+)?json$/,
|
expectedMime: MimeTypeCheck = JSON_MIME,
|
||||||
contentTypeRequired = true,
|
contentTypeRequired = true,
|
||||||
): Transformer<Response, ParsedJsonResponse<T>> {
|
): Transformer<Response, ParsedJsonResponse<T>> {
|
||||||
return pipe(
|
return pipe(
|
||||||
fetchTypeProcessor(expectedMime, contentTypeRequired),
|
fetchTypeProcessor(expectedMime, contentTypeRequired),
|
||||||
cancelBodyOnError(fetchResponseJsonTranformer<T>),
|
cancelBodyOnError(fetchResponseJsonTransformer<T>),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto-labs/handle-resolver-node
|
# @atproto-labs/handle-resolver-node
|
||||||
|
|
||||||
|
## 0.1.10
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies []:
|
||||||
|
- @atproto-labs/fetch-node@0.1.5
|
||||||
|
|
||||||
## 0.1.9
|
## 0.1.9
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto-labs/handle-resolver-node",
|
"name": "@atproto-labs/handle-resolver-node",
|
||||||
"version": "0.1.9",
|
"version": "0.1.10",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Node specific ATProto handle to DID resolver",
|
"description": "Node specific ATProto handle to DID resolver",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto-labs/identity-resolver
|
# @atproto-labs/identity-resolver
|
||||||
|
|
||||||
|
## 0.1.10
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies []:
|
||||||
|
- @atproto-labs/did-resolver@0.1.8
|
||||||
|
|
||||||
## 0.1.9
|
## 0.1.9
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto-labs/identity-resolver",
|
"name": "@atproto-labs/identity-resolver",
|
||||||
"version": "0.1.9",
|
"version": "0.1.10",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "A library resolving ATPROTO identities",
|
"description": "A library resolving ATPROTO identities",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto-labs/xrpc-utils
|
# @atproto-labs/xrpc-utils
|
||||||
|
|
||||||
|
## 0.0.2
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies []:
|
||||||
|
- @atproto/xrpc-server@0.7.6
|
||||||
|
|
||||||
## 0.0.1
|
## 0.0.1
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto-labs/xrpc-utils",
|
"name": "@atproto-labs/xrpc-utils",
|
||||||
"version": "0.0.1",
|
"version": "0.0.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "XRPC server utilities for Node.JS",
|
"description": "XRPC server utilities for Node.JS",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,16 @@
|
|||||||
# @atproto/jwk-jose
|
# @atproto/jwk-jose
|
||||||
|
|
||||||
|
## 0.1.3
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- [#2879](https://github.com/bluesky-social/atproto/pull/2879) [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Improve compatibility with runtimes relying on webcrypto (by explicit JOSE's importJWK() "alg" argument).
|
||||||
|
|
||||||
|
- [#2879](https://github.com/bluesky-social/atproto/pull/2879) [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Remove unsafe type casting during JWT verification
|
||||||
|
|
||||||
|
- Updated dependencies [[`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0)]:
|
||||||
|
- @atproto/jwk@0.1.2
|
||||||
|
|
||||||
## 0.1.2
|
## 0.1.2
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/jwk-jose",
|
"name": "@atproto/jwk-jose",
|
||||||
"version": "0.1.2",
|
"version": "0.1.3",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "`jose` based implementation of @atproto/jwk Key's",
|
"description": "`jose` based implementation of @atproto/jwk Key's",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,4 +1,19 @@
|
|||||||
import { JwtVerifyError } from '@atproto/jwk'
|
import {
|
||||||
|
Jwk,
|
||||||
|
JwkError,
|
||||||
|
JwtCreateError,
|
||||||
|
JwtHeader,
|
||||||
|
JwtPayload,
|
||||||
|
JwtVerifyError,
|
||||||
|
Key,
|
||||||
|
RequiredKey,
|
||||||
|
SignedJwt,
|
||||||
|
VerifyOptions,
|
||||||
|
VerifyResult,
|
||||||
|
jwkValidator,
|
||||||
|
jwtHeaderSchema,
|
||||||
|
jwtPayloadSchema,
|
||||||
|
} from '@atproto/jwk'
|
||||||
import {
|
import {
|
||||||
SignJWT,
|
SignJWT,
|
||||||
errors,
|
errors,
|
||||||
@ -14,19 +29,6 @@ import {
|
|||||||
type KeyLike,
|
type KeyLike,
|
||||||
} from 'jose'
|
} from 'jose'
|
||||||
|
|
||||||
import {
|
|
||||||
Jwk,
|
|
||||||
JwkError,
|
|
||||||
JwtCreateError,
|
|
||||||
JwtHeader,
|
|
||||||
JwtPayload,
|
|
||||||
Key,
|
|
||||||
SignedJwt,
|
|
||||||
VerifyOptions,
|
|
||||||
VerifyPayload,
|
|
||||||
VerifyResult,
|
|
||||||
jwkValidator,
|
|
||||||
} from '@atproto/jwk'
|
|
||||||
import { either } from './util'
|
import { either } from './util'
|
||||||
|
|
||||||
const { JOSEError } = errors
|
const { JOSEError } = errors
|
||||||
@ -35,53 +37,97 @@ export type Importable = string | KeyLike | Jwk
|
|||||||
|
|
||||||
export type { GenerateKeyPairOptions, GenerateKeyPairResult }
|
export type { GenerateKeyPairOptions, GenerateKeyPairResult }
|
||||||
|
|
||||||
export class JoseKey extends Key {
|
export class JoseKey<J extends Jwk = Jwk> extends Key<J> {
|
||||||
#keyObj?: KeyLike | Uint8Array
|
/**
|
||||||
|
* Some runtimes (e.g. Bun) require an `alg` second argument to be set when
|
||||||
protected async getKey() {
|
* invoking `importJWK`. In order to be compatible with these runtimes, we
|
||||||
|
* provide the following method to ensure the `alg` is always set. We also
|
||||||
|
* take the opportunity to ensure that the `alg` is compatible with this key.
|
||||||
|
*/
|
||||||
|
protected async getKeyObj(alg: string) {
|
||||||
|
if (!this.algorithms.includes(alg)) {
|
||||||
|
throw new JwkError(`Key cannot be used with algorithm "${alg}"`)
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
return (this.#keyObj ||= await importJWK(this.jwk as JWK))
|
return await importJWK(this.jwk as JWK, alg)
|
||||||
} catch (cause) {
|
} catch (cause) {
|
||||||
throw new JwkError('Failed to import JWK', undefined, { cause })
|
throw new JwkError('Failed to import JWK', undefined, { cause })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async createJwt(header: JwtHeader, payload: JwtPayload) {
|
async createJwt(header: JwtHeader, payload: JwtPayload): Promise<SignedJwt> {
|
||||||
if (header.kid && header.kid !== this.kid) {
|
|
||||||
throw new JwtCreateError(
|
|
||||||
`Invalid "kid" (${header.kid}) used to sign with key "${this.kid}"`,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!header.alg || !this.algorithms.includes(header.alg)) {
|
|
||||||
throw new JwtCreateError(
|
|
||||||
`Invalid "alg" (${header.alg}) used to sign with key "${this.kid}"`,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
const keyObj = await this.getKey()
|
|
||||||
return new SignJWT(payload)
|
|
||||||
.setProtectedHeader({ ...header, kid: this.kid })
|
|
||||||
.sign(keyObj) as Promise<SignedJwt>
|
|
||||||
}
|
|
||||||
|
|
||||||
async verifyJwt<
|
|
||||||
P extends VerifyPayload = JwtPayload,
|
|
||||||
C extends string = string,
|
|
||||||
>(token: SignedJwt, options?: VerifyOptions<C>): Promise<VerifyResult<P, C>> {
|
|
||||||
try {
|
try {
|
||||||
const keyObj = await this.getKey()
|
const { kid } = header
|
||||||
const result = await jwtVerify(token, keyObj, {
|
if (kid && kid !== this.kid) {
|
||||||
...options,
|
throw new JwtCreateError(
|
||||||
algorithms: this.algorithms,
|
`Invalid "kid" (${kid}) used to sign with key "${this.kid}"`,
|
||||||
} as JWTVerifyOptions)
|
)
|
||||||
|
}
|
||||||
|
|
||||||
return result as VerifyResult<P, C>
|
const { alg } = header
|
||||||
} catch (error) {
|
if (!alg) {
|
||||||
if (error instanceof JOSEError) {
|
throw new JwtCreateError('Missing "alg" in JWT header')
|
||||||
throw new JwtVerifyError(error.message, error.code, { cause: error })
|
}
|
||||||
|
|
||||||
|
const keyObj = await this.getKeyObj(alg)
|
||||||
|
const jwtBuilder = new SignJWT(payload).setProtectedHeader({
|
||||||
|
...header,
|
||||||
|
alg,
|
||||||
|
kid: this.kid,
|
||||||
|
})
|
||||||
|
|
||||||
|
const signedJwt = await jwtBuilder.sign(keyObj)
|
||||||
|
|
||||||
|
return signedJwt as SignedJwt
|
||||||
|
} catch (cause) {
|
||||||
|
if (cause instanceof JOSEError) {
|
||||||
|
throw new JwtCreateError(cause.message, cause.code, { cause })
|
||||||
} else {
|
} else {
|
||||||
throw JwtVerifyError.from(error)
|
throw JwtCreateError.from(cause)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async verifyJwt<C extends string = never>(
|
||||||
|
token: SignedJwt,
|
||||||
|
options?: VerifyOptions<C>,
|
||||||
|
): Promise<VerifyResult<C>> {
|
||||||
|
try {
|
||||||
|
const result = await jwtVerify(
|
||||||
|
token,
|
||||||
|
async ({ alg }) => this.getKeyObj(alg),
|
||||||
|
{ ...options, algorithms: this.algorithms } as JWTVerifyOptions,
|
||||||
|
)
|
||||||
|
|
||||||
|
// @NOTE if all tokens are signed exclusively through createJwt(), then
|
||||||
|
// there should be no need to parse the payload and headers here. But
|
||||||
|
// since the JWT could have been signed with the same key from somewhere
|
||||||
|
// else, let's parse it to ensure the integrity (and type safety) of the
|
||||||
|
// data.
|
||||||
|
const headerParsed = jwtHeaderSchema.safeParse(result.protectedHeader)
|
||||||
|
if (!headerParsed.success) {
|
||||||
|
throw new JwtVerifyError('Invalid JWT header', undefined, {
|
||||||
|
cause: headerParsed.error,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const payloadParsed = jwtPayloadSchema.safeParse(result.payload)
|
||||||
|
if (!payloadParsed.success) {
|
||||||
|
throw new JwtVerifyError('Invalid JWT payload', undefined, {
|
||||||
|
cause: payloadParsed.error,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
protectedHeader: headerParsed.data,
|
||||||
|
// "requiredClaims" enforced by jwtVerify()
|
||||||
|
payload: payloadParsed.data as RequiredKey<JwtPayload, C>,
|
||||||
|
}
|
||||||
|
} catch (cause) {
|
||||||
|
if (cause instanceof JOSEError) {
|
||||||
|
throw new JwtVerifyError(cause.message, cause.code, { cause })
|
||||||
|
} else {
|
||||||
|
throw JwtVerifyError.from(cause)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,13 @@
|
|||||||
# @atproto/jwk-webcrypto
|
# @atproto/jwk-webcrypto
|
||||||
|
|
||||||
|
## 0.1.3
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0)]:
|
||||||
|
- @atproto/jwk@0.1.2
|
||||||
|
- @atproto/jwk-jose@0.1.3
|
||||||
|
|
||||||
## 0.1.2
|
## 0.1.2
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/jwk-webcrypto",
|
"name": "@atproto/jwk-webcrypto",
|
||||||
"version": "0.1.2",
|
"version": "0.1.3",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Webcrypto based implementation of @atproto/jwk Key's",
|
"description": "Webcrypto based implementation of @atproto/jwk Key's",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
@ -25,7 +25,8 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@atproto/jwk": "workspace:*",
|
"@atproto/jwk": "workspace:*",
|
||||||
"@atproto/jwk-jose": "workspace:*"
|
"@atproto/jwk-jose": "workspace:*",
|
||||||
|
"zod": "^3.23.8"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"typescript": "^5.6.3"
|
"typescript": "^5.6.3"
|
||||||
|
@ -1,9 +1,20 @@
|
|||||||
import { Jwk, jwkSchema } from '@atproto/jwk'
|
import { JwkError, jwkSchema } from '@atproto/jwk'
|
||||||
import { GenerateKeyPairOptions, JoseKey } from '@atproto/jwk-jose'
|
import { GenerateKeyPairOptions, JoseKey } from '@atproto/jwk-jose'
|
||||||
|
import z from 'zod'
|
||||||
|
|
||||||
import { fromSubtleAlgorithm, isCryptoKeyPair } from './util.js'
|
import { fromSubtleAlgorithm, isCryptoKeyPair } from './util.js'
|
||||||
|
|
||||||
export class WebcryptoKey extends JoseKey {
|
// Webcrypto keys are bound to a single algorithm
|
||||||
|
export const jwkWithAlgSchema = z.intersection(
|
||||||
|
jwkSchema,
|
||||||
|
z.object({ alg: z.string() }),
|
||||||
|
)
|
||||||
|
|
||||||
|
export type JwkWithAlg = z.infer<typeof jwkWithAlgSchema>
|
||||||
|
|
||||||
|
export class WebcryptoKey<
|
||||||
|
J extends JwkWithAlg = JwkWithAlg,
|
||||||
|
> extends JoseKey<J> {
|
||||||
// We need to override the static method generate from JoseKey because
|
// We need to override the static method generate from JoseKey because
|
||||||
// the browser needs both the private and public keys
|
// the browser needs both the private and public keys
|
||||||
static override async generate(
|
static override async generate(
|
||||||
@ -26,29 +37,35 @@ export class WebcryptoKey extends JoseKey {
|
|||||||
// > The "use" and "key_ops" JWK members SHOULD NOT be used together; [...]
|
// > The "use" and "key_ops" JWK members SHOULD NOT be used together; [...]
|
||||||
// > Applications should specify which of these members they use.
|
// > Applications should specify which of these members they use.
|
||||||
|
|
||||||
const { key_ops: _, ...jwk } = await crypto.subtle.exportKey(
|
const {
|
||||||
|
key_ops,
|
||||||
|
use,
|
||||||
|
alg = fromSubtleAlgorithm(cryptoKeyPair.privateKey.algorithm),
|
||||||
|
...jwk
|
||||||
|
} = await crypto.subtle.exportKey(
|
||||||
'jwk',
|
'jwk',
|
||||||
cryptoKeyPair.privateKey.extractable
|
cryptoKeyPair.privateKey.extractable
|
||||||
? cryptoKeyPair.privateKey
|
? cryptoKeyPair.privateKey
|
||||||
: cryptoKeyPair.publicKey,
|
: cryptoKeyPair.publicKey,
|
||||||
)
|
)
|
||||||
|
|
||||||
const use = jwk.use ?? 'sig'
|
if (use && use !== 'sig') {
|
||||||
const alg =
|
throw new TypeError(`Unsupported JWK use "${use}"`)
|
||||||
jwk.alg ?? fromSubtleAlgorithm(cryptoKeyPair.privateKey.algorithm)
|
}
|
||||||
|
|
||||||
if (use !== 'sig') {
|
if (key_ops && !key_ops.some((o) => o === 'sign' || o === 'verify')) {
|
||||||
throw new TypeError('Unsupported JWK use')
|
// Make sure that "key_ops", if present, is compatible with "use"
|
||||||
|
throw new TypeError(`Invalid key_ops "${key_ops}" for "sig" use`)
|
||||||
}
|
}
|
||||||
|
|
||||||
return new WebcryptoKey(
|
return new WebcryptoKey(
|
||||||
jwkSchema.parse({ ...jwk, use, kid, alg }),
|
jwkWithAlgSchema.parse({ ...jwk, kid, alg, use: 'sig' }),
|
||||||
cryptoKeyPair,
|
cryptoKeyPair,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
jwk: Jwk,
|
jwk: Readonly<J>,
|
||||||
readonly cryptoKeyPair: CryptoKeyPair,
|
readonly cryptoKeyPair: CryptoKeyPair,
|
||||||
) {
|
) {
|
||||||
super(jwk)
|
super(jwk)
|
||||||
@ -58,12 +75,15 @@ export class WebcryptoKey extends JoseKey {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
get privateJwk(): Jwk | undefined {
|
get privateJwk(): Readonly<J> | undefined {
|
||||||
if (super.isPrivate) return this.jwk
|
if (super.isPrivate) return this.jwk
|
||||||
throw new Error('Private Webcrypto Key not exportable')
|
throw new Error('Private Webcrypto Key not exportable')
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override async getKey() {
|
protected override async getKeyObj(alg: string) {
|
||||||
|
if (this.jwk.alg !== alg) {
|
||||||
|
throw new JwkError(`Key cannot be used with algorithm "${alg}"`)
|
||||||
|
}
|
||||||
return this.cryptoKeyPair.privateKey
|
return this.cryptoKeyPair.privateKey
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,17 @@
|
|||||||
# @atproto/jwk
|
# @atproto/jwk
|
||||||
|
|
||||||
|
## 0.1.2
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- [#2879](https://github.com/bluesky-social/atproto/pull/2879) [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Mark jwk key fields as readonly
|
||||||
|
|
||||||
|
- [#2879](https://github.com/bluesky-social/atproto/pull/2879) [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Remove unsafe type casting during JWT verification
|
||||||
|
|
||||||
|
- [#2879](https://github.com/bluesky-social/atproto/pull/2879) [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Allow (passthrough) unknown properties in JWT payload & headers
|
||||||
|
|
||||||
|
- [#2879](https://github.com/bluesky-social/atproto/pull/2879) [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Expose ValidationError to allow for proper error handling
|
||||||
|
|
||||||
## 0.1.1
|
## 0.1.1
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/jwk",
|
"name": "@atproto/jwk",
|
||||||
"version": "0.1.1",
|
"version": "0.1.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "A library for working with JSON Web Keys (JWKs) in TypeScript. This is meant to be extended by environment-specific libraries like @atproto/jwk-jose.",
|
"description": "A library for working with JSON Web Keys (JWKs) in TypeScript. This is meant to be extended by environment-specific libraries like @atproto/jwk-jose.",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,3 +1,8 @@
|
|||||||
|
// Since we expose zod schemas, let's expose ZodError (under a generic name) so
|
||||||
|
// that dependents can catch schema parsing errors without requiring an explicit
|
||||||
|
// dependency on zod, or risking a conflict in case of mismatching zob versions.
|
||||||
|
export { ZodError as ValidationError } from 'zod'
|
||||||
|
|
||||||
export * from './alg.js'
|
export * from './alg.js'
|
||||||
export * from './errors.js'
|
export * from './errors.js'
|
||||||
export * from './jwk.js'
|
export * from './jwk.js'
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { JwtHeader, JwtPayload } from './jwt.js'
|
import { JwtHeader, JwtPayload } from './jwt.js'
|
||||||
import { RequiredKey } from './util.js'
|
import { RequiredKey } from './util.js'
|
||||||
|
|
||||||
export type VerifyOptions<C extends string = string> = {
|
export type VerifyOptions<C extends string = never> = {
|
||||||
audience?: string | readonly string[]
|
audience?: string | readonly string[]
|
||||||
/** in seconds */
|
/** in seconds */
|
||||||
clockTolerance?: number
|
clockTolerance?: number
|
||||||
@ -14,9 +14,7 @@ export type VerifyOptions<C extends string = string> = {
|
|||||||
requiredClaims?: readonly C[]
|
requiredClaims?: readonly C[]
|
||||||
}
|
}
|
||||||
|
|
||||||
export type VerifyPayload = Record<string, unknown>
|
export type VerifyResult<C extends string = never> = {
|
||||||
|
payload: RequiredKey<JwtPayload, C>
|
||||||
export type VerifyResult<P extends VerifyPayload, C extends string> = {
|
|
||||||
payload: RequiredKey<P & JwtPayload, C>
|
|
||||||
protectedHeader: JwtHeader
|
protectedHeader: JwtHeader
|
||||||
}
|
}
|
||||||
|
@ -24,7 +24,8 @@ export const isUnsignedJwt = (data: unknown): data is UnsignedJwt =>
|
|||||||
/**
|
/**
|
||||||
* @see {@link https://www.rfc-editor.org/rfc/rfc7515.html#section-4}
|
* @see {@link https://www.rfc-editor.org/rfc/rfc7515.html#section-4}
|
||||||
*/
|
*/
|
||||||
export const jwtHeaderSchema = z.object({
|
export const jwtHeaderSchema = z
|
||||||
|
.object({
|
||||||
/** "alg" (Algorithm) Header Parameter */
|
/** "alg" (Algorithm) Header Parameter */
|
||||||
alg: z.string(),
|
alg: z.string(),
|
||||||
/** "jku" (JWK Set URL) Header Parameter */
|
/** "jku" (JWK Set URL) Header Parameter */
|
||||||
@ -57,11 +58,13 @@ export const jwtHeaderSchema = z.object({
|
|||||||
/** "crit" (Critical) Header Parameter */
|
/** "crit" (Critical) Header Parameter */
|
||||||
crit: z.array(z.string()).optional(),
|
crit: z.array(z.string()).optional(),
|
||||||
})
|
})
|
||||||
|
.passthrough()
|
||||||
|
|
||||||
export type JwtHeader = z.infer<typeof jwtHeaderSchema>
|
export type JwtHeader = z.infer<typeof jwtHeaderSchema>
|
||||||
|
|
||||||
// https://www.iana.org/assignments/jwt/jwt.xhtml
|
// https://www.iana.org/assignments/jwt/jwt.xhtml
|
||||||
export const jwtPayloadSchema = z.object({
|
export const jwtPayloadSchema = z
|
||||||
|
.object({
|
||||||
iss: z.string().optional(),
|
iss: z.string().optional(),
|
||||||
aud: z.union([z.string(), z.array(z.string()).nonempty()]).optional(),
|
aud: z.union([z.string(), z.array(z.string()).nonempty()]).optional(),
|
||||||
sub: z.string().optional(),
|
sub: z.string().optional(),
|
||||||
@ -169,5 +172,6 @@ export const jwtPayloadSchema = z.object({
|
|||||||
)
|
)
|
||||||
.optional(),
|
.optional(),
|
||||||
})
|
})
|
||||||
|
.passthrough()
|
||||||
|
|
||||||
export type JwtPayload = z.infer<typeof jwtPayloadSchema>
|
export type JwtPayload = z.infer<typeof jwtPayloadSchema>
|
||||||
|
@ -1,12 +1,14 @@
|
|||||||
import { jwkAlgorithms } from './alg.js'
|
import { jwkAlgorithms } from './alg.js'
|
||||||
import { JwkError } from './errors.js'
|
import { JwkError } from './errors.js'
|
||||||
import { Jwk, jwkSchema } from './jwk.js'
|
import { Jwk, jwkSchema } from './jwk.js'
|
||||||
import { VerifyOptions, VerifyPayload, VerifyResult } from './jwt-verify.js'
|
import { VerifyOptions, VerifyResult } from './jwt-verify.js'
|
||||||
import { JwtHeader, JwtPayload, SignedJwt } from './jwt.js'
|
import { JwtHeader, JwtPayload, SignedJwt } from './jwt.js'
|
||||||
import { cachedGetter } from './util.js'
|
import { cachedGetter } from './util.js'
|
||||||
|
|
||||||
export abstract class Key {
|
const jwkSchemaReadonly = jwkSchema.readonly()
|
||||||
constructor(protected readonly jwk: Readonly<Jwk>) {
|
|
||||||
|
export abstract class Key<J extends Jwk = Jwk> {
|
||||||
|
constructor(protected readonly jwk: Readonly<J>) {
|
||||||
// A key should always be used either for signing or encryption.
|
// A key should always be used either for signing or encryption.
|
||||||
if (!jwk.use) throw new JwkError('Missing "use" Parameter value')
|
if (!jwk.use) throw new JwkError('Missing "use" Parameter value')
|
||||||
}
|
}
|
||||||
@ -24,25 +26,28 @@ export abstract class Key {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
get privateJwk(): Jwk | undefined {
|
get privateJwk(): Readonly<J> | undefined {
|
||||||
return this.isPrivate ? this.jwk : undefined
|
return this.isPrivate ? this.jwk : undefined
|
||||||
}
|
}
|
||||||
|
|
||||||
@cachedGetter
|
@cachedGetter
|
||||||
get publicJwk(): Jwk | undefined {
|
get publicJwk():
|
||||||
|
| Readonly<Exclude<J, { kty: 'oct' }> & { d?: never }>
|
||||||
|
| undefined {
|
||||||
if (this.isSymetric) return undefined
|
if (this.isSymetric) return undefined
|
||||||
if (this.isPrivate) {
|
|
||||||
const { d: _, ...jwk } = this.jwk as any
|
return jwkSchemaReadonly.parse({
|
||||||
return jwk
|
...this.jwk,
|
||||||
}
|
d: undefined,
|
||||||
return this.jwk
|
k: undefined,
|
||||||
|
}) as Exclude<J, { kty: 'oct' }> & { d?: never }
|
||||||
}
|
}
|
||||||
|
|
||||||
@cachedGetter
|
@cachedGetter
|
||||||
get bareJwk(): Jwk | undefined {
|
get bareJwk(): Readonly<Jwk> | undefined {
|
||||||
if (this.isSymetric) return undefined
|
if (this.isSymetric) return undefined
|
||||||
const { kty, crv, e, n, x, y } = this.jwk as any
|
const { kty, crv, e, n, x, y } = this.jwk as any
|
||||||
return jwkSchema.parse({ crv, e, kty, n, x, y })
|
return jwkSchemaReadonly.parse({ crv, e, kty, n, x, y })
|
||||||
}
|
}
|
||||||
|
|
||||||
get use() {
|
get use() {
|
||||||
@ -64,7 +69,7 @@ export abstract class Key {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get crv() {
|
get crv() {
|
||||||
return (this.jwk as { crv: undefined } | Extract<Jwk, { crv: unknown }>).crv
|
return (this.jwk as { crv: undefined } | Extract<J, { crv: unknown }>).crv
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -73,7 +78,7 @@ export abstract class Key {
|
|||||||
*/
|
*/
|
||||||
@cachedGetter
|
@cachedGetter
|
||||||
get algorithms(): readonly string[] {
|
get algorithms(): readonly string[] {
|
||||||
return Array.from(jwkAlgorithms(this.jwk))
|
return Object.freeze(Array.from(jwkAlgorithms(this.jwk)))
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -86,8 +91,8 @@ export abstract class Key {
|
|||||||
*
|
*
|
||||||
* @throws {JwtVerifyError} if the JWT is invalid
|
* @throws {JwtVerifyError} if the JWT is invalid
|
||||||
*/
|
*/
|
||||||
abstract verifyJwt<
|
abstract verifyJwt<C extends string = never>(
|
||||||
P extends VerifyPayload = JwtPayload,
|
token: SignedJwt,
|
||||||
C extends string = string,
|
options?: VerifyOptions<C>,
|
||||||
>(token: SignedJwt, options?: VerifyOptions<C>): Promise<VerifyResult<P, C>>
|
): Promise<VerifyResult<C>>
|
||||||
}
|
}
|
||||||
|
@ -213,13 +213,10 @@ export class Keyset<K extends Key = Key> implements Iterable<K> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async verifyJwt<
|
async verifyJwt<C extends string = never>(
|
||||||
P extends Record<string, unknown> = JwtPayload,
|
|
||||||
C extends string = string,
|
|
||||||
>(
|
|
||||||
token: SignedJwt,
|
token: SignedJwt,
|
||||||
options?: VerifyOptions<C>,
|
options?: VerifyOptions<C>,
|
||||||
): Promise<VerifyResult<P, C> & { key: K }> {
|
): Promise<VerifyResult<C> & { key: K }> {
|
||||||
const { header } = unsafeDecodeJwt(token)
|
const { header } = unsafeDecodeJwt(token)
|
||||||
const { kid, alg } = header
|
const { kid, alg } = header
|
||||||
|
|
||||||
@ -227,7 +224,7 @@ export class Keyset<K extends Key = Key> implements Iterable<K> {
|
|||||||
|
|
||||||
for (const key of this.list({ kid, alg })) {
|
for (const key of this.list({ kid, alg })) {
|
||||||
try {
|
try {
|
||||||
const result = await key.verifyJwt<P, C>(token, options)
|
const result = await key.verifyJwt<C>(token, options)
|
||||||
return { ...result, key }
|
return { ...result, key }
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
errors.push(err)
|
errors.push(err)
|
||||||
|
@ -5,12 +5,12 @@ import { RefinementCtx, ZodIssueCode } from 'zod'
|
|||||||
export type Simplify<T> = { [K in keyof T]: T[K] } & {}
|
export type Simplify<T> = { [K in keyof T]: T[K] } & {}
|
||||||
export type Override<T, V> = Simplify<V & Omit<T, keyof V>>
|
export type Override<T, V> = Simplify<V & Omit<T, keyof V>>
|
||||||
|
|
||||||
export type RequiredKey<T, K extends string> = Simplify<
|
export type RequiredKey<T, K extends keyof T = never> = Simplify<
|
||||||
string extends K
|
T & {
|
||||||
? T
|
[L in K]-?: unknown extends T[L]
|
||||||
: {
|
? NonNullable<unknown> | null
|
||||||
[L in K]: Exclude<L extends keyof T ? T[L] : unknown, undefined>
|
: Exclude<T[L], undefined>
|
||||||
} & Omit<T, K>
|
}
|
||||||
>
|
>
|
||||||
|
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-types
|
// eslint-disable-next-line @typescript-eslint/ban-types
|
||||||
|
@ -1,5 +1,16 @@
|
|||||||
# @atproto/oauth-client-browser
|
# @atproto/oauth-client-browser
|
||||||
|
|
||||||
|
## 0.3.7
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0)]:
|
||||||
|
- @atproto/jwk@0.1.2
|
||||||
|
- @atproto/jwk-webcrypto@0.1.3
|
||||||
|
- @atproto/oauth-client@0.3.7
|
||||||
|
- @atproto/oauth-types@0.2.2
|
||||||
|
- @atproto-labs/did-resolver@0.1.8
|
||||||
|
|
||||||
## 0.3.6
|
## 0.3.6
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/oauth-client-browser",
|
"name": "@atproto/oauth-client-browser",
|
||||||
"version": "0.3.6",
|
"version": "0.3.7",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "ATPROTO OAuth client for the browser (relies on WebCrypto & Indexed DB)",
|
"description": "ATPROTO OAuth client for the browser (relies on WebCrypto & Indexed DB)",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,18 @@
|
|||||||
# @atproto/oauth-client-node
|
# @atproto/oauth-client-node
|
||||||
|
|
||||||
|
## 0.2.7
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0)]:
|
||||||
|
- @atproto/jwk@0.1.2
|
||||||
|
- @atproto/jwk-jose@0.1.3
|
||||||
|
- @atproto/jwk-webcrypto@0.1.3
|
||||||
|
- @atproto/oauth-client@0.3.7
|
||||||
|
- @atproto/oauth-types@0.2.2
|
||||||
|
- @atproto-labs/did-resolver@0.1.8
|
||||||
|
- @atproto-labs/handle-resolver-node@0.1.10
|
||||||
|
|
||||||
## 0.2.6
|
## 0.2.6
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/oauth-client-node",
|
"name": "@atproto/oauth-client-node",
|
||||||
"version": "0.2.6",
|
"version": "0.2.7",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "ATPROTO OAuth client for the NodeJS",
|
"description": "ATPROTO OAuth client for the NodeJS",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,16 @@
|
|||||||
# @atproto/oauth-client
|
# @atproto/oauth-client
|
||||||
|
|
||||||
|
## 0.3.7
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2)]:
|
||||||
|
- @atproto/jwk@0.1.2
|
||||||
|
- @atproto-labs/fetch@0.2.0
|
||||||
|
- @atproto/oauth-types@0.2.2
|
||||||
|
- @atproto-labs/did-resolver@0.1.8
|
||||||
|
- @atproto-labs/identity-resolver@0.1.10
|
||||||
|
|
||||||
## 0.3.6
|
## 0.3.6
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/oauth-client",
|
"name": "@atproto/oauth-client",
|
||||||
"version": "0.3.6",
|
"version": "0.3.7",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "OAuth client for ATPROTO PDS. This package serves as common base for environment-specific implementations (NodeJS, Browser, React-Native).",
|
"description": "OAuth client for ATPROTO PDS. This package serves as common base for environment-specific implementations (NodeJS, Browser, React-Native).",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,16 @@
|
|||||||
# @atproto/oauth-provider
|
# @atproto/oauth-provider
|
||||||
|
|
||||||
|
## 0.2.12
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`5ece8c6ae`](https://github.com/bluesky-social/atproto/commit/5ece8c6aeab9c5c3f51295d93ed6e27c3c6095c2)]:
|
||||||
|
- @atproto/jwk@0.1.2
|
||||||
|
- @atproto/jwk-jose@0.1.3
|
||||||
|
- @atproto-labs/fetch@0.2.0
|
||||||
|
- @atproto/oauth-types@0.2.2
|
||||||
|
- @atproto-labs/fetch-node@0.1.5
|
||||||
|
|
||||||
## 0.2.11
|
## 0.2.11
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/oauth-provider",
|
"name": "@atproto/oauth-provider",
|
||||||
"version": "0.2.11",
|
"version": "0.2.12",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Generic OAuth2 and OpenID Connect provider for Node.js. Currently only supports features needed for Atproto.",
|
"description": "Generic OAuth2 and OpenID Connect provider for Node.js. Currently only supports features needed for Atproto.",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -27,7 +27,8 @@ export const signedTokenPayloadSchema = z.intersection(
|
|||||||
jti: tokenIdSchema,
|
jti: tokenIdSchema,
|
||||||
sub: subSchema,
|
sub: subSchema,
|
||||||
client_id: clientIdSchema,
|
client_id: clientIdSchema,
|
||||||
}),
|
})
|
||||||
|
.passthrough(),
|
||||||
)
|
)
|
||||||
|
|
||||||
export type SignedTokenPayload = Simplify<
|
export type SignedTokenPayload = Simplify<
|
||||||
|
@ -19,7 +19,7 @@ import {
|
|||||||
signedTokenPayloadSchema,
|
signedTokenPayloadSchema,
|
||||||
} from './signed-token-payload.js'
|
} from './signed-token-payload.js'
|
||||||
|
|
||||||
export type SignPayload = Omit<JwtPayload, 'iss'>
|
export type SignPayload = JwtPayload & { iss?: never }
|
||||||
|
|
||||||
export class Signer {
|
export class Signer {
|
||||||
constructor(
|
constructor(
|
||||||
@ -27,11 +27,11 @@ export class Signer {
|
|||||||
public readonly keyset: Keyset,
|
public readonly keyset: Keyset,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
async verify<P extends Record<string, unknown> = JwtPayload>(
|
async verify<C extends string = never>(
|
||||||
token: SignedJwt,
|
token: SignedJwt,
|
||||||
options?: Omit<VerifyOptions, 'issuer'>,
|
options?: Omit<VerifyOptions<C>, 'issuer'>,
|
||||||
) {
|
) {
|
||||||
return this.keyset.verifyJwt<P>(token, {
|
return this.keyset.verifyJwt<C>(token, {
|
||||||
...options,
|
...options,
|
||||||
issuer: [this.issuer],
|
issuer: [this.issuer],
|
||||||
})
|
})
|
||||||
@ -84,18 +84,16 @@ export class Signer {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
async verifyAccessToken(token: SignedJwt) {
|
async verifyAccessToken<C extends string = never>(
|
||||||
const result = await this.verify<SignedTokenPayload>(token, {
|
token: SignedJwt,
|
||||||
typ: 'at+jwt',
|
options?: Omit<VerifyOptions<C>, 'issuer' | 'typ'>,
|
||||||
})
|
) {
|
||||||
|
const result = await this.verify<C>(token, { ...options, typ: 'at+jwt' })
|
||||||
// The result is already type casted as an AccessTokenPayload, but we need
|
type Payload = typeof result.payload // RequiredKey<JwtPayload, C>
|
||||||
// to actually verify this. That should already be covered by the fact that
|
return {
|
||||||
// we don't sign 'at+jwt' tokens without a valid token ID. Let's double
|
protectedHeader: result.protectedHeader,
|
||||||
// check in case another version/implementation was used to generate the
|
payload: signedTokenPayloadSchema.parse(result.payload) as Payload &
|
||||||
// token.
|
SignedTokenPayload,
|
||||||
signedTokenPayloadSchema.parse(result.payload)
|
}
|
||||||
|
|
||||||
return result
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -462,6 +462,7 @@ export class TokenManager {
|
|||||||
case isSignedJwt(token): {
|
case isSignedJwt(token): {
|
||||||
const { payload } = await this.signer.verify(token, {
|
const { payload } = await this.signer.verify(token, {
|
||||||
clockTolerance: Infinity,
|
clockTolerance: Infinity,
|
||||||
|
requiredClaims: ['jti'],
|
||||||
})
|
})
|
||||||
const tokenId = tokenIdSchema.parse(payload.jti)
|
const tokenId = tokenIdSchema.parse(payload.jti)
|
||||||
await this.store.deleteToken(tokenId)
|
await this.store.deleteToken(tokenId)
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto/oauth-types
|
# @atproto/oauth-types
|
||||||
|
|
||||||
|
## 0.2.2
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0), [`2889c7699`](https://github.com/bluesky-social/atproto/commit/2889c76995ce3c569f595ac3c678218e9ce659f0)]:
|
||||||
|
- @atproto/jwk@0.1.2
|
||||||
|
|
||||||
## 0.2.1
|
## 0.2.1
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/oauth-types",
|
"name": "@atproto/oauth-types",
|
||||||
"version": "0.2.1",
|
"version": "0.2.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "OAuth typing & validation library",
|
"description": "OAuth typing & validation library",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,23 @@
|
|||||||
# @atproto/ozone
|
# @atproto/ozone
|
||||||
|
|
||||||
|
## 0.1.67
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- [#3344](https://github.com/bluesky-social/atproto/pull/3344) [`48a0e9d60`](https://github.com/bluesky-social/atproto/commit/48a0e9d6060c2dc93899f13f2fc7cc76c04fbcd9) Thanks [@matthieusieben](https://github.com/matthieusieben)! - Properly dispose of unused http responses
|
||||||
|
|
||||||
|
- Updated dependencies [[`e277158f7`](https://github.com/bluesky-social/atproto/commit/e277158f70a831b04fde3ec84b3c1eaa6ce82e9d)]:
|
||||||
|
- @atproto/api@0.13.27
|
||||||
|
|
||||||
|
## 0.1.66
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277)]:
|
||||||
|
- @atproto/crypto@0.4.3
|
||||||
|
- @atproto/identity@0.4.5
|
||||||
|
- @atproto/xrpc-server@0.7.6
|
||||||
|
|
||||||
## 0.1.65
|
## 0.1.65
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/ozone",
|
"name": "@atproto/ozone",
|
||||||
"version": "0.1.65",
|
"version": "0.1.67",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Backend service for moderating the Bluesky network.",
|
"description": "Backend service for moderating the Bluesky network.",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -46,7 +46,7 @@ export class BlobDiverter {
|
|||||||
})
|
})
|
||||||
|
|
||||||
if (blobResponse.statusCode !== 200) {
|
if (blobResponse.statusCode !== 200) {
|
||||||
blobResponse.body.destroy()
|
await blobResponse.body.dump()
|
||||||
throw new XRPCError(
|
throw new XRPCError(
|
||||||
blobResponse.statusCode,
|
blobResponse.statusCode,
|
||||||
undefined,
|
undefined,
|
||||||
@ -72,7 +72,7 @@ export class BlobDiverter {
|
|||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
// Typically un-supported content encoding
|
// Typically un-supported content encoding
|
||||||
blobResponse.body.destroy()
|
await blobResponse.body.dump()
|
||||||
throw err
|
throw err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -99,7 +99,7 @@ export class BlobDiverter {
|
|||||||
})
|
})
|
||||||
|
|
||||||
if (result.statusCode !== 200) {
|
if (result.statusCode !== 200) {
|
||||||
result.body.destroy()
|
await result.body.dump()
|
||||||
throw new XRPCError(
|
throw new XRPCError(
|
||||||
result.statusCode,
|
result.statusCode,
|
||||||
undefined,
|
undefined,
|
||||||
|
@ -12359,7 +12359,7 @@ export const schemaDict = {
|
|||||||
items: {
|
items: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
description:
|
description:
|
||||||
'If specified, only events where the policy matches the given policy are returned',
|
'If specified, only events where the action policies match any of the given policies are returned',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
cursor: {
|
cursor: {
|
||||||
|
@ -19,6 +19,8 @@ export type InputSchema = undefined
|
|||||||
export interface OutputSchema {
|
export interface OutputSchema {
|
||||||
cursor?: string
|
cursor?: string
|
||||||
actors: AppBskyActorDefs.ProfileView[]
|
actors: AppBskyActorDefs.ProfileView[]
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,6 +19,8 @@ export interface OutputSchema {
|
|||||||
suggestions: AppBskyActorDefs.ProfileView[]
|
suggestions: AppBskyActorDefs.ProfileView[]
|
||||||
/** If true, response has fallen-back to generic results, and is not scoped using relativeToDid */
|
/** If true, response has fallen-back to generic results, and is not scoped using relativeToDid */
|
||||||
isFallback?: boolean
|
isFallback?: boolean
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,6 +25,8 @@ export interface OutputSchema {
|
|||||||
actors: AppBskyUnspeccedDefs.SkeletonSearchActor[]
|
actors: AppBskyUnspeccedDefs.SkeletonSearchActor[]
|
||||||
/** DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer. */
|
/** DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer. */
|
||||||
relativeToDid?: string
|
relativeToDid?: string
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,26 @@
|
|||||||
# @atproto/pds
|
# @atproto/pds
|
||||||
|
|
||||||
|
## 0.4.84
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`e277158f7`](https://github.com/bluesky-social/atproto/commit/e277158f70a831b04fde3ec84b3c1eaa6ce82e9d)]:
|
||||||
|
- @atproto/api@0.13.27
|
||||||
|
- @atproto/oauth-provider@0.2.12
|
||||||
|
- @atproto-labs/fetch-node@0.1.5
|
||||||
|
|
||||||
|
## 0.4.83
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277)]:
|
||||||
|
- @atproto/crypto@0.4.3
|
||||||
|
- @atproto/aws@0.2.12
|
||||||
|
- @atproto/identity@0.4.5
|
||||||
|
- @atproto/repo@0.6.2
|
||||||
|
- @atproto/xrpc-server@0.7.6
|
||||||
|
- @atproto-labs/xrpc-utils@0.0.2
|
||||||
|
|
||||||
## 0.4.82
|
## 0.4.82
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/pds",
|
"name": "@atproto/pds",
|
||||||
"version": "0.4.82",
|
"version": "0.4.84",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Reference implementation of atproto Personal Data Server (PDS)",
|
"description": "Reference implementation of atproto Personal Data Server (PDS)",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -12359,7 +12359,7 @@ export const schemaDict = {
|
|||||||
items: {
|
items: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
description:
|
description:
|
||||||
'If specified, only events where the policy matches the given policy are returned',
|
'If specified, only events where the action policies match any of the given policies are returned',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
cursor: {
|
cursor: {
|
||||||
|
@ -19,6 +19,8 @@ export type InputSchema = undefined
|
|||||||
export interface OutputSchema {
|
export interface OutputSchema {
|
||||||
cursor?: string
|
cursor?: string
|
||||||
actors: AppBskyActorDefs.ProfileView[]
|
actors: AppBskyActorDefs.ProfileView[]
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,6 +19,8 @@ export interface OutputSchema {
|
|||||||
suggestions: AppBskyActorDefs.ProfileView[]
|
suggestions: AppBskyActorDefs.ProfileView[]
|
||||||
/** If true, response has fallen-back to generic results, and is not scoped using relativeToDid */
|
/** If true, response has fallen-back to generic results, and is not scoped using relativeToDid */
|
||||||
isFallback?: boolean
|
isFallback?: boolean
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,6 +25,8 @@ export interface OutputSchema {
|
|||||||
actors: AppBskyUnspeccedDefs.SkeletonSearchActor[]
|
actors: AppBskyUnspeccedDefs.SkeletonSearchActor[]
|
||||||
/** DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer. */
|
/** DID of the account these suggestions are relative to. If this is returned undefined, suggestions are based on the viewer. */
|
||||||
relativeToDid?: string
|
relativeToDid?: string
|
||||||
|
/** Snowflake for this recommendation, use when submitting recommendation events. */
|
||||||
|
recId?: number
|
||||||
[k: string]: unknown
|
[k: string]: unknown
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,6 +5,7 @@ import { CID } from 'multiformats/cid'
|
|||||||
import { Server } from 'node:http'
|
import { Server } from 'node:http'
|
||||||
import { AddressInfo } from 'node:net'
|
import { AddressInfo } from 'node:net'
|
||||||
import { FeedViewPost } from '../src/lexicon/types/app/bsky/feed/defs'
|
import { FeedViewPost } from '../src/lexicon/types/app/bsky/feed/defs'
|
||||||
|
import { ToolsOzoneModerationDefs } from '@atproto/api'
|
||||||
|
|
||||||
// Swap out identifiers and dates with stable
|
// Swap out identifiers and dates with stable
|
||||||
// values for the purpose of snapshot testing
|
// values for the purpose of snapshot testing
|
||||||
@ -202,3 +203,21 @@ export async function stopServer(server: Server) {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const normalizeSubjectStatus = (
|
||||||
|
subject: ToolsOzoneModerationDefs.SubjectStatusView,
|
||||||
|
) => {
|
||||||
|
return { ...subject, tags: subject.tags?.sort() }
|
||||||
|
}
|
||||||
|
|
||||||
|
export const forSubjectStatusSnapshot = (
|
||||||
|
status:
|
||||||
|
| ToolsOzoneModerationDefs.SubjectStatusView
|
||||||
|
| ToolsOzoneModerationDefs.SubjectStatusView[],
|
||||||
|
) => {
|
||||||
|
if (Array.isArray(status)) {
|
||||||
|
return forSnapshot(status.map(normalizeSubjectStatus))
|
||||||
|
}
|
||||||
|
|
||||||
|
return forSnapshot(normalizeSubjectStatus(status))
|
||||||
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { AtpAgent, ComAtprotoModerationDefs } from '@atproto/api'
|
import { AtpAgent, ComAtprotoModerationDefs } from '@atproto/api'
|
||||||
import { SeedClient, TestNetwork } from '@atproto/dev-env'
|
import { SeedClient, TestNetwork } from '@atproto/dev-env'
|
||||||
import { forSnapshot } from './_util'
|
import { forSubjectStatusSnapshot } from './_util'
|
||||||
import { ids } from '../src/lexicon/lexicons'
|
import { ids } from '../src/lexicon/lexicons'
|
||||||
|
|
||||||
describe('appeal account takedown', () => {
|
describe('appeal account takedown', () => {
|
||||||
@ -35,6 +35,7 @@ describe('appeal account takedown', () => {
|
|||||||
email: 'jeff@test.com',
|
email: 'jeff@test.com',
|
||||||
password: 'password',
|
password: 'password',
|
||||||
})
|
})
|
||||||
|
await network.processAll()
|
||||||
|
|
||||||
// Emit a takedown event
|
// Emit a takedown event
|
||||||
await network.ozone.getModClient().performTakedown({
|
await network.ozone.getModClient().performTakedown({
|
||||||
@ -97,7 +98,9 @@ describe('appeal account takedown', () => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
expect(result.subjectStatuses[0].appealed).toBe(true)
|
expect(result.subjectStatuses[0].appealed).toBe(true)
|
||||||
expect(forSnapshot(result.subjectStatuses[0])).toMatchSnapshot()
|
expect(
|
||||||
|
forSubjectStatusSnapshot(result.subjectStatuses[0]),
|
||||||
|
).toMatchSnapshot()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('takendown actor is not allowed to create reports.', async () => {
|
it('takendown actor is not allowed to create reports.', async () => {
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto/repo
|
# @atproto/repo
|
||||||
|
|
||||||
|
## 0.6.2
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277)]:
|
||||||
|
- @atproto/crypto@0.4.3
|
||||||
|
|
||||||
## 0.6.1
|
## 0.6.1
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/repo",
|
"name": "@atproto/repo",
|
||||||
"version": "0.6.1",
|
"version": "0.6.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "atproto repo and MST implementation",
|
"description": "atproto repo and MST implementation",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,14 @@
|
|||||||
# @atproto/sync
|
# @atproto/sync
|
||||||
|
|
||||||
|
## 0.1.9
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies []:
|
||||||
|
- @atproto/identity@0.4.5
|
||||||
|
- @atproto/repo@0.6.2
|
||||||
|
- @atproto/xrpc-server@0.7.6
|
||||||
|
|
||||||
## 0.1.8
|
## 0.1.8
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/sync",
|
"name": "@atproto/sync",
|
||||||
"version": "0.1.8",
|
"version": "0.1.9",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "atproto sync library",
|
"description": "atproto sync library",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
@ -1,5 +1,12 @@
|
|||||||
# @atproto/xrpc-server
|
# @atproto/xrpc-server
|
||||||
|
|
||||||
|
## 0.7.6
|
||||||
|
|
||||||
|
### Patch Changes
|
||||||
|
|
||||||
|
- Updated dependencies [[`1abfd74ec`](https://github.com/bluesky-social/atproto/commit/1abfd74ec7114e5d8e2411f7a4fa10bdce97e277)]:
|
||||||
|
- @atproto/crypto@0.4.3
|
||||||
|
|
||||||
## 0.7.5
|
## 0.7.5
|
||||||
|
|
||||||
### Patch Changes
|
### Patch Changes
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@atproto/xrpc-server",
|
"name": "@atproto/xrpc-server",
|
||||||
"version": "0.7.5",
|
"version": "0.7.6",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "atproto HTTP API (XRPC) server library",
|
"description": "atproto HTTP API (XRPC) server library",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
@ -37,7 +37,7 @@
|
|||||||
"@atproto/crypto": "workspace:^",
|
"@atproto/crypto": "workspace:^",
|
||||||
"@types/express": "^4.17.13",
|
"@types/express": "^4.17.13",
|
||||||
"@types/express-serve-static-core": "^4.17.36",
|
"@types/express-serve-static-core": "^4.17.36",
|
||||||
"@types/http-errors": "^2.0.1",
|
"@types/http-errors": "^2.0.4",
|
||||||
"@types/ws": "^8.5.4",
|
"@types/ws": "^8.5.4",
|
||||||
"get-port": "^6.1.2",
|
"get-port": "^6.1.2",
|
||||||
"jest": "^28.1.2",
|
"jest": "^28.1.2",
|
||||||
|
15
pnpm-lock.yaml
generated
15
pnpm-lock.yaml
generated
@ -201,7 +201,7 @@ importers:
|
|||||||
version: 0.0.1
|
version: 0.0.1
|
||||||
'@types/http-errors':
|
'@types/http-errors':
|
||||||
specifier: ^2.0.1
|
specifier: ^2.0.1
|
||||||
version: 2.0.1
|
version: 2.0.4
|
||||||
compression:
|
compression:
|
||||||
specifier: ^1.7.4
|
specifier: ^1.7.4
|
||||||
version: 1.7.4
|
version: 1.7.4
|
||||||
@ -812,6 +812,9 @@ importers:
|
|||||||
'@atproto/jwk-jose':
|
'@atproto/jwk-jose':
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../jwk-jose
|
version: link:../jwk-jose
|
||||||
|
zod:
|
||||||
|
specifier: ^3.23.8
|
||||||
|
version: 3.23.8
|
||||||
devDependencies:
|
devDependencies:
|
||||||
typescript:
|
typescript:
|
||||||
specifier: ^5.6.3
|
specifier: ^5.6.3
|
||||||
@ -1566,8 +1569,8 @@ importers:
|
|||||||
specifier: ^4.17.36
|
specifier: ^4.17.36
|
||||||
version: 4.17.36
|
version: 4.17.36
|
||||||
'@types/http-errors':
|
'@types/http-errors':
|
||||||
specifier: ^2.0.1
|
specifier: ^2.0.4
|
||||||
version: 2.0.1
|
version: 2.0.4
|
||||||
'@types/ws':
|
'@types/ws':
|
||||||
specifier: ^8.5.4
|
specifier: ^8.5.4
|
||||||
version: 8.5.4
|
version: 8.5.4
|
||||||
@ -6310,8 +6313,8 @@ packages:
|
|||||||
'@types/node': 18.19.67
|
'@types/node': 18.19.67
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/@types/http-errors@2.0.1:
|
/@types/http-errors@2.0.4:
|
||||||
resolution: {integrity: sha512-/K3ds8TRAfBvi5vfjuz8y6+GiAYBZ0x4tXv1Av6CWBWn0IlADc+ZX9pMq7oU0fNQPnBwIZl3rmeLp6SBApbxSQ==}
|
resolution: {integrity: sha512-D0CFMMtydbJAegzOyHjtiKPLlvnm3iTZyZRSZoLq2mRhDdmLfIWOCYPfQJ4cu2erKghU++QvjcUjp/5h7hESpA==}
|
||||||
|
|
||||||
/@types/is-ci@3.0.0:
|
/@types/is-ci@3.0.0:
|
||||||
resolution: {integrity: sha512-Q0Op0hdWbYd1iahB+IFNQcWXFq4O0Q5MwQP7uN0souuQ4rPg1vEYcnIOfr1gY+M+6rc8FGoRaBO1mOOvL29sEQ==}
|
resolution: {integrity: sha512-Q0Op0hdWbYd1iahB+IFNQcWXFq4O0Q5MwQP7uN0souuQ4rPg1vEYcnIOfr1gY+M+6rc8FGoRaBO1mOOvL29sEQ==}
|
||||||
@ -6447,7 +6450,7 @@ packages:
|
|||||||
/@types/serve-static@1.15.2:
|
/@types/serve-static@1.15.2:
|
||||||
resolution: {integrity: sha512-J2LqtvFYCzaj8pVYKw8klQXrLLk7TBZmQ4ShlcdkELFKGwGMfevMLneMMRkMgZxotOD9wg497LpC7O8PcvAmfw==}
|
resolution: {integrity: sha512-J2LqtvFYCzaj8pVYKw8klQXrLLk7TBZmQ4ShlcdkELFKGwGMfevMLneMMRkMgZxotOD9wg497LpC7O8PcvAmfw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
'@types/http-errors': 2.0.1
|
'@types/http-errors': 2.0.4
|
||||||
'@types/mime': 3.0.1
|
'@types/mime': 3.0.1
|
||||||
'@types/node': 18.19.67
|
'@types/node': 18.19.67
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user