mppx 0.6.25 → 0.6.26
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +8 -0
- package/dist/Method.d.ts +6 -4
- package/dist/Method.d.ts.map +1 -1
- package/dist/Method.js +2 -1
- package/dist/Method.js.map +1 -1
- package/dist/middlewares/internal/mppx.d.ts +3 -2
- package/dist/middlewares/internal/mppx.d.ts.map +1 -1
- package/dist/middlewares/internal/mppx.js +1 -0
- package/dist/middlewares/internal/mppx.js.map +1 -1
- package/dist/server/Mppx.d.ts +8 -3
- package/dist/server/Mppx.d.ts.map +1 -1
- package/dist/server/Mppx.js +4 -1
- package/dist/server/Mppx.js.map +1 -1
- package/dist/stripe/server/Charge.d.ts +1 -1
- package/dist/stripe/server/Charge.d.ts.map +1 -1
- package/dist/stripe/server/Methods.d.ts +1 -1
- package/dist/stripe/server/Methods.d.ts.map +1 -1
- package/dist/tempo/Methods.d.ts +3 -2
- package/dist/tempo/Methods.d.ts.map +1 -1
- package/dist/tempo/Methods.js +13 -4
- package/dist/tempo/Methods.js.map +1 -1
- package/dist/tempo/client/Subscription.d.ts +3 -2
- package/dist/tempo/client/Subscription.d.ts.map +1 -1
- package/dist/tempo/server/Charge.d.ts +1 -1
- package/dist/tempo/server/Charge.d.ts.map +1 -1
- package/dist/tempo/server/Methods.d.ts +2 -2
- package/dist/tempo/server/Methods.d.ts.map +1 -1
- package/dist/tempo/server/Session.d.ts +1 -1
- package/dist/tempo/server/Session.d.ts.map +1 -1
- package/dist/tempo/server/Subscription.d.ts +28 -12
- package/dist/tempo/server/Subscription.d.ts.map +1 -1
- package/dist/tempo/server/Subscription.js +82 -30
- package/dist/tempo/server/Subscription.js.map +1 -1
- package/dist/tempo/server/internal/html.gen.d.ts +1 -1
- package/dist/tempo/server/internal/html.gen.d.ts.map +1 -1
- package/dist/tempo/server/internal/html.gen.js +1 -1
- package/dist/tempo/server/internal/html.gen.js.map +1 -1
- package/dist/tempo/subscription/KeyAuthorization.d.ts +3 -14
- package/dist/tempo/subscription/KeyAuthorization.d.ts.map +1 -1
- package/dist/tempo/subscription/KeyAuthorization.js +11 -20
- package/dist/tempo/subscription/KeyAuthorization.js.map +1 -1
- package/dist/tempo/subscription/Types.d.ts +2 -2
- package/dist/tempo/subscription/Types.d.ts.map +1 -1
- package/package.json +1 -1
- package/src/Method.ts +21 -5
- package/src/middlewares/internal/mppx.test.ts +24 -0
- package/src/middlewares/internal/mppx.ts +6 -2
- package/src/server/Mppx.ts +17 -4
- package/src/tempo/Methods.test.ts +17 -0
- package/src/tempo/Methods.ts +12 -4
- package/src/tempo/client/Subscription.test.ts +5 -7
- package/src/tempo/server/Subscription.test.ts +1 -4
- package/src/tempo/server/Subscription.ts +156 -67
- package/src/tempo/server/internal/html/package.json +1 -1
- package/src/tempo/server/internal/html.gen.ts +1 -1
- package/src/tempo/subscription/KeyAuthorization.test.ts +13 -4
- package/src/tempo/subscription/KeyAuthorization.ts +11 -20
- package/src/tempo/subscription/Types.ts +2 -2
|
@@ -11,6 +11,8 @@ import {
|
|
|
11
11
|
toSubscriptionExpiryDate,
|
|
12
12
|
toSubscriptionExpirySeconds,
|
|
13
13
|
toSubscriptionPeriodSeconds,
|
|
14
|
+
transferSelector,
|
|
15
|
+
transferWithMemoSelector,
|
|
14
16
|
verifySubscriptionKeyAuthorization,
|
|
15
17
|
} from './KeyAuthorization.js'
|
|
16
18
|
import type { SubscriptionAccessKey } from './Types.js'
|
|
@@ -89,13 +91,12 @@ describe('tempo subscription key authorization', () => {
|
|
|
89
91
|
const request = parseRequest()
|
|
90
92
|
|
|
91
93
|
expect(getSubscriptionScopes(request)).toMatchObject([
|
|
92
|
-
{ address: currency, recipients: [recipient] },
|
|
93
|
-
{ address: currency, recipients: [recipient] },
|
|
94
|
+
{ address: currency, recipients: [recipient], selector: transferWithMemoSelector },
|
|
94
95
|
])
|
|
95
96
|
expect(getSubscriptionRpcAllowedCalls(request)).toMatchObject([
|
|
96
97
|
{
|
|
97
98
|
target: currency,
|
|
98
|
-
selectorRules: [{ recipients: [recipient]
|
|
99
|
+
selectorRules: [{ recipients: [recipient], selector: transferWithMemoSelector }],
|
|
99
100
|
},
|
|
100
101
|
])
|
|
101
102
|
})
|
|
@@ -158,7 +159,10 @@ describe('tempo subscription key authorization', () => {
|
|
|
158
159
|
const authorization = KeyAuthorization.deserialize(payload.signature)
|
|
159
160
|
const transferOnly = KeyAuthorization.serialize({
|
|
160
161
|
...authorization,
|
|
161
|
-
scopes: authorization.scopes?.
|
|
162
|
+
scopes: authorization.scopes?.map((scope) => ({
|
|
163
|
+
...scope,
|
|
164
|
+
selector: transferSelector,
|
|
165
|
+
})),
|
|
162
166
|
})
|
|
163
167
|
|
|
164
168
|
expect(() =>
|
|
@@ -183,6 +187,11 @@ describe('tempo subscription key authorization', () => {
|
|
|
183
187
|
).toThrow('subscription period cannot be represented exactly by this Tempo client')
|
|
184
188
|
})
|
|
185
189
|
|
|
190
|
+
test('accepts dev-prefixed subscription periods for development and tests', () => {
|
|
191
|
+
expect(toSubscriptionPeriodSeconds({ periodCount: '15', periodUnit: 'dev_second' })).toBe(15)
|
|
192
|
+
expect(toSubscriptionPeriodSeconds({ periodCount: '120', periodUnit: 'dev_second' })).toBe(120)
|
|
193
|
+
})
|
|
194
|
+
|
|
186
195
|
test('rejects subscription expiries that cannot be represented by Tempo key authorizations', () => {
|
|
187
196
|
expect(() =>
|
|
188
197
|
toSubscriptionExpirySeconds(toSubscriptionExpiryDate('2026-01-01T00:00:00.500Z')),
|
|
@@ -12,12 +12,15 @@ import type {
|
|
|
12
12
|
/** 4-byte selector for TIP-20 `transfer(address,uint256)`. */
|
|
13
13
|
export const transferSelector = '0xa9059cbb'
|
|
14
14
|
|
|
15
|
-
/** 4-byte selector for TIP-20 `transferWithMemo(address,uint256,
|
|
15
|
+
/** 4-byte selector for TIP-20 `transferWithMemo(address,uint256,bytes32)`. */
|
|
16
16
|
export const transferWithMemoSelector = '0x95777d59'
|
|
17
17
|
|
|
18
18
|
const uint64Max = (1n << 64n) - 1n
|
|
19
|
-
const
|
|
20
|
-
|
|
19
|
+
const subscriptionPeriodUnitSeconds = {
|
|
20
|
+
dev_second: 1n,
|
|
21
|
+
day: 86_400n,
|
|
22
|
+
week: 604_800n,
|
|
23
|
+
} satisfies Record<SubscriptionPeriodUnit, bigint>
|
|
21
24
|
|
|
22
25
|
type SubscriptionRequest = ReturnType<typeof Methods.subscription.schema.request.parse>
|
|
23
26
|
type Authorization = KeyAuthorization.KeyAuthorization
|
|
@@ -63,11 +66,11 @@ export function toSubscriptionPeriodSeconds(request: {
|
|
|
63
66
|
if (!/^[1-9]\d*$/.test(request.periodCount)) {
|
|
64
67
|
throw new VerificationFailedError({ reason: 'periodCount is invalid' })
|
|
65
68
|
}
|
|
66
|
-
|
|
69
|
+
const unitSeconds = subscriptionPeriodUnitSeconds[request.periodUnit]
|
|
70
|
+
if (unitSeconds === undefined) {
|
|
67
71
|
throw new VerificationFailedError({ reason: 'periodUnit is invalid' })
|
|
68
72
|
}
|
|
69
73
|
|
|
70
|
-
const unitSeconds = request.periodUnit === 'day' ? secondsPerDay : secondsPerWeek
|
|
71
74
|
const value = BigInt(request.periodCount) * unitSeconds
|
|
72
75
|
if (value > uint64Max) {
|
|
73
76
|
throw new VerificationFailedError({
|
|
@@ -113,11 +116,6 @@ export function getSubscriptionScopes(
|
|
|
113
116
|
const currency = normalizeAddress(request.currency, 'currency')
|
|
114
117
|
const recipient = normalizeAddress(request.recipient, 'recipient')
|
|
115
118
|
return [
|
|
116
|
-
{
|
|
117
|
-
address: currency,
|
|
118
|
-
selector: transferSelector,
|
|
119
|
-
recipients: [recipient],
|
|
120
|
-
},
|
|
121
119
|
{
|
|
122
120
|
address: currency,
|
|
123
121
|
selector: transferWithMemoSelector,
|
|
@@ -130,15 +128,11 @@ export function getSubscriptionScopes(
|
|
|
130
128
|
export function getSubscriptionRpcAllowedCalls(
|
|
131
129
|
request: Pick<SubscriptionRequest, 'currency' | 'recipient'>,
|
|
132
130
|
) {
|
|
133
|
-
const [
|
|
131
|
+
const [transferWithMemo] = getSubscriptionScopes(request)
|
|
134
132
|
return [
|
|
135
133
|
{
|
|
136
134
|
target: normalizeAddress(request.currency, 'currency'),
|
|
137
135
|
selectorRules: [
|
|
138
|
-
{
|
|
139
|
-
selector: transfer.selector,
|
|
140
|
-
recipients: transfer.recipients,
|
|
141
|
-
},
|
|
142
136
|
{
|
|
143
137
|
selector: transferWithMemo.selector,
|
|
144
138
|
recipients: transferWithMemo.recipients,
|
|
@@ -325,9 +319,9 @@ function assertAuthorizationScopes(
|
|
|
325
319
|
scopes: readonly KeyAuthorization.Scope[] | undefined,
|
|
326
320
|
request: Pick<SubscriptionRequest, 'currency' | 'recipient'>,
|
|
327
321
|
) {
|
|
328
|
-
if (!scopes || scopes.length
|
|
322
|
+
if (!scopes || scopes.length !== 1) {
|
|
329
323
|
throw new VerificationFailedError({
|
|
330
|
-
reason: 'keyAuthorization must contain recipient-scoped
|
|
324
|
+
reason: 'keyAuthorization must contain a recipient-scoped transferWithMemo call',
|
|
331
325
|
})
|
|
332
326
|
}
|
|
333
327
|
|
|
@@ -353,9 +347,6 @@ function assertAuthorizationScopes(
|
|
|
353
347
|
}
|
|
354
348
|
}
|
|
355
349
|
|
|
356
|
-
if (!seen.has(transferSelector)) {
|
|
357
|
-
throw new VerificationFailedError({ reason: 'keyAuthorization must allow transfer' })
|
|
358
|
-
}
|
|
359
350
|
if (!seen.has(transferWithMemoSelector)) {
|
|
360
351
|
throw new VerificationFailedError({ reason: 'keyAuthorization must allow transferWithMemo' })
|
|
361
352
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import type { Address } from 'viem'
|
|
2
2
|
|
|
3
|
-
/** Tempo-supported subscription period units.
|
|
4
|
-
export type SubscriptionPeriodUnit = 'day' | 'week'
|
|
3
|
+
/** Tempo-supported subscription period units. Use `day` or `week` in production; `dev_*` units are for development and tests. */
|
|
4
|
+
export type SubscriptionPeriodUnit = 'dev_second' | 'day' | 'week'
|
|
5
5
|
|
|
6
6
|
/** Access key information used to authorize recurring Tempo payments. */
|
|
7
7
|
export type SubscriptionAccessKey = {
|