Skip to content

Commit

Permalink
Merge pull request #226 from nevermined-io/feat/credits
Browse files Browse the repository at this point in the history
Credits flows
  • Loading branch information
aaitor authored Aug 21, 2023
2 parents 95eab12 + 89685ea commit b8fee00
Show file tree
Hide file tree
Showing 9 changed files with 121 additions and 40 deletions.
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -400,3 +400,5 @@ tsconfig.build.

circuits
*.tgz
.yalc*
.env*
19 changes: 18 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,23 @@ All notable changes to this project will be documented in this file. Dates are d

Generated by [`auto-changelog`](https://github.com/CookPete/auto-changelog).

#### [v2.0.0-rc3](https://github.com/nevermined-io/node/compare/v1.4.1...v2.0.0-rc3)

> 10 August 2023
- Feat/sdk 2.0.0 rc2 [`#219`](https://github.com/nevermined-io/node/pull/219)
- Adapting to SDK v2.0.0-rc1 [`#218`](https://github.com/nevermined-io/node/pull/218)
- build(deps): bump aws-sdk from 2.1404.0 to 2.1421.0 [`#217`](https://github.com/nevermined-io/node/pull/217)
- build(deps): bump word-wrap from 1.2.3 to 1.2.4 [`#210`](https://github.com/nevermined-io/node/pull/210)
- build(deps): bump @sideway/address from 4.1.4 to 5.0.0 [`#216`](https://github.com/nevermined-io/node/pull/216)
- feat: update sdk to version 2.0.0-rc0 [`#212`](https://github.com/nevermined-io/node/pull/212)
- build(deps): bump semver from 5.7.1 to 5.7.2 [`#203`](https://github.com/nevermined-io/node/pull/203)
- build(deps-dev): bump lint-staged from 13.2.2 to 13.2.3 [`#196`](https://github.com/nevermined-io/node/pull/196)
- build(deps): bump fetch-blob from 3.2.0 to 4.0.0 [`#205`](https://github.com/nevermined-io/node/pull/205)
- feat: credits flows [`e89e980`](https://github.com/nevermined-io/node/commit/e89e980adcd6e6b782e21bc30177e6fe6cc1b476)
- feat: update to sdk 2.0.0-rc2 [`85b9057`](https://github.com/nevermined-io/node/commit/85b9057a45a488a0741f7ee2c99d678cc5b79457)
- chore: adapting to sdk v2.0.0-rc1 [`a34b4a0`](https://github.com/nevermined-io/node/commit/a34b4a0dc26696cee210b5520f840913a30cd172)

#### [v1.4.1](https://github.com/nevermined-io/node/compare/v1.4.0...v1.4.1)

> 19 July 2023
Expand Down Expand Up @@ -294,7 +311,7 @@ Generated by [`auto-changelog`](https://github.com/CookPete/auto-changelog).
- feat: set jwt expiry time based on subscription duration [`04806b6`](https://github.com/nevermined-io/node/commit/04806b6eea7addc886cc2bf2eb6397bdcea5cd15)
- feat: added configuration for the subscriptions module [`dfffb81`](https://github.com/nevermined-io/node/commit/dfffb816a1df1e9a53dbf05e1321c1e522a71826)

### [v1.0.0](https://github.com/nevermined-io/node/compare/v1.0.0-rc9...v1.0.0)
#### [v1.0.0](https://github.com/nevermined-io/node/compare/v1.0.0-rc9...v1.0.0)

> 14 February 2023
Expand Down
10 changes: 5 additions & 5 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "node-ts",
"version": "2.0.0-rc1",
"version": "2.0.0-rc4",
"description": "Nevermined Node",
"main": "main.ts",
"scripts": {
Expand All @@ -21,8 +21,8 @@
"test:cov": "jest ./src --setupFiles dotenv/config --coverage",
"integration": "jest ./integration --setupFiles dotenv/config",
"integration:cov": "jest ./integration --setupFiles dotenv/config --coverage",
"prettier": "prettier --config ./.prettierrc.js --write \"**/*.{js,json,md,sol,ts,yml}\"",
"prettier:check": "prettier --check --config ./.prettierrc.json \"**/*.{js,json,md,sol,ts,yml}\"",
"prettier": "prettier --config ./.prettierrc.js --write \"**/*.{json,md,sol,ts,yml}\"",
"prettier:check": "prettier --check --config ./.prettierrc.json \"**/*.{json,md,sol,ts,yml}\"",
"artifacts:mumbai": "sh ./scripts/download-artifacts.sh $CONTRACT_VERSION mumbai"
},
"dependencies": {
Expand All @@ -36,8 +36,8 @@
"@nestjs/typeorm": "^10.0.0",
"@nevermined-io/argo-workflows-api": "^0.1.3",
"@nevermined-io/passport-nevermined": "^0.2.0",
"@nevermined-io/sdk": "^2.0.0-rc2",
"@nevermined-io/sdk-dtp": "^0.6.0",
"@nevermined-io/sdk": "^2.0.0-rc8",
"@nevermined-io/sdk-dtp": "^0.7.0-rc4",
"@sideway/address": "^5.0.0",
"@sideway/formula": "^3.0.1",
"@sideway/pinpoint": "^2.0.0",
Expand Down
25 changes: 22 additions & 3 deletions src/access/access.controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,6 @@ export class AccessController {
description: 'Bad Request. DID missing',
type: BadRequestException,
})
@ApiBearerAuth('Authorization')
async doAccess(
@Req() req: Request<unknown>,
@Response({ passthrough: true }) res,
Expand Down Expand Up @@ -184,7 +183,18 @@ export class AccessController {
}

const subscriptionDDO = await this.nvmService.nevermined.assets.resolve(did.getDid())
const duration = await this.nvmService.getDuration(subscriptionDDO, template as ServiceType)
const serviceReference =
transferData.serviceIndex && transferData.serviceIndex >= 0
? transferData.serviceIndex
: (template as ServiceType)

const service = subscriptionDDO.findServiceByReference(serviceReference)

const duration = await this.nvmService.getDuration(subscriptionDDO, serviceReference)

Logger.debug(
`Transferring NFT with Service Reference ${serviceReference} and duration ${duration} with amount ${transferData.nftAmount}`,
)

let expiration = 0
if (duration > 0) {
Expand All @@ -195,9 +205,11 @@ export class AccessController {
const params: ValidationParams = {
consumer_address: transferData.nftReceiver,
did: did.getDid(),
service_index: service.index,
agreement_id: transferData.agreementId,
nft_amount: BigInt(transferData.nftAmount || '0'),
buyer: (req.user || {}).buyer,
duration,
expiration,
}

Expand All @@ -209,6 +221,7 @@ export class AccessController {
`[${did.getDid()}] Fulfilling transfer NFT with agreement ${transferData.agreementId}`,
)
await plugin.process(params, from, undefined)
Logger.debug(`NFT Transfered to ${transferData.nftReceiver}`)
} catch (e) {
Logger.error(`Failed to transfer NFT ${e}`)
throw new ForbiddenException(
Expand Down Expand Up @@ -251,7 +264,6 @@ export class AccessController {
description: 'Bad Request. DID missing',
type: BadRequestException,
})
@ApiBearerAuth('Authorization')
async doDownload(
@Req() req: Request<unknown>,
@Response({ passthrough: true }) res,
Expand Down Expand Up @@ -322,3 +334,10 @@ export class AccessController {
}
}
}
export const jsonReplacer = (key, value) => {
// Modify the value or return undefined to exclude the property
if (typeof value === 'bigint') {
return value.toString()
}
return value
}
8 changes: 8 additions & 0 deletions src/access/dto/transfer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,4 +44,12 @@ export class TransferDto {
})
@IsNumber()
nftType: number

@ApiProperty({
description: 'The service index of the NFT to claim',
example: '3',
})
@IsOptional()
@IsNumber()
serviceIndex: number
}
3 changes: 2 additions & 1 deletion src/auth/auth.service.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,8 @@ describe('AuthService', () => {
it('should validate the owner and return true if the owner has permission to access', async () => {
const did = 'did:nft:0x123'
const consumer_address = '0x456'
await authService.validateOwner(did, consumer_address)
const params: ValidationParams = { agreement_id: '0x789', did, consumer_address }
await authService.validateOwner(params)
expect(
nvmServiceMock.getNevermined().keeper.conditions.accessCondition.checkPermissions,
).toHaveBeenCalledWith(consumer_address, did)
Expand Down
62 changes: 49 additions & 13 deletions src/auth/auth.service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import {
Account,
Babysig,
DDO,
NeverminedNFT1155Type,
} from '@nevermined-io/sdk'
import { NeverminedService } from '../shared/nevermined/nvm.service'
import { JWTPayload } from '@nevermined-io/passport-nevermined'
Expand All @@ -21,14 +22,20 @@ const BASE_URL = '/api/v1/node/services/'
export class AuthService {
constructor(private jwtService: JwtService, private nvmService: NeverminedService) {}

async validateOwner(did: string, consumer_address: string): Promise<void> {
async validateOwner(params: ValidationParams): Promise<void> {
const nevermined = this.nvmService.getNevermined()
const getNftAccess = async () => {
const ddo = await nevermined.assets.resolve(did)
const ddo = await nevermined.assets.resolve(params.did)
Logger.debug(`Validating owner for ${params.did}`)

const getNftAccess = async (ddo: DDO, serviceIndex?: number) => {
if (!ddo) {
return null
}
const service = ddo.findServiceByType('nft-access')
const service =
serviceIndex && serviceIndex > 0
? ddo.findServiceByIndex(serviceIndex)
: ddo.findServiceByType('nft-access')

if (!service) {
return null
}
Expand All @@ -44,15 +51,37 @@ export class AuthService {
}

const granted = await nevermined.keeper.conditions.accessCondition.checkPermissions(
consumer_address,
did,
params.consumer_address,
params.did,
)
if (!granted) {
const limit = await getNftAccess()
const balance = await nevermined.nfts1155.balance(did, new Account(consumer_address))
const limit = await getNftAccess(ddo, params.service_index)
const balance = await nevermined.nfts1155.balance(
params.did,
new Account(params.consumer_address),
)
if (!limit || balance < limit) {
throw new UnauthorizedException(
`Address ${consumer_address} has no permission to access ${did}`,
`Address ${params.consumer_address} has no permission to access ${params.did}`,
)
}
}

const metadataService = ddo.findServiceByType('metadata')
const isNft1155Credit =
metadataService.attributes.main.nftType &&
metadataService.attributes.main.nftType.toString() ===
NeverminedNFT1155Type.nft1155Credit.toString()
if (isNft1155Credit) {
Logger.debug(`Validating NFT1155 Credit for ${params.did}`)
const [from] = await nevermined.accounts.list()
const plugin = nevermined.nfts1155.servicePlugin['nft-access']

try {
await plugin.track(params, from)
} catch (error) {
throw new UnauthorizedException(
`Address ${params.consumer_address} could not use the credits to access ${params.did}`,
)
}
}
Expand All @@ -64,10 +93,16 @@ export class AuthService {
const plugin =
nevermined.assets.servicePlugin[service] || nevermined.nfts1155.servicePlugin[service]

const granted = await plugin.accept(params)
if (!granted) {
try {
const [from] = await nevermined.accounts.list()
await plugin.process(params, from, undefined)
const granted = await plugin.accept(params)
if (!granted) {
await plugin.process(params, from, undefined)
}

if (plugin.track) await plugin.track(params, from)
} catch (error) {
throw new UnauthorizedException(`Error processing request: ${error.message}`)
}
}

Expand Down Expand Up @@ -96,12 +131,13 @@ export class AuthService {
agreement_id: payload.sub,
buyer: payload.buyer as string,
babysig: payload.babysig as Babysig,
service_index: payload.service_index as number,
}

if (payload.aud === BASE_URL + 'access') {
await this.validateAccess(params, 'access')
} else if (payload.aud === BASE_URL + 'download') {
await this.validateOwner(payload.did as string, payload.iss)
await this.validateOwner(params)
} else if (payload.aud === BASE_URL + 'nft-access') {
await this.validateAccess(params, 'nft-access')
} else if (payload.aud === BASE_URL + 'nft-sales-proof') {
Expand Down
16 changes: 7 additions & 9 deletions src/shared/nevermined/nvm.service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -421,35 +421,33 @@ export class NeverminedService {
* Get the duration of the subscription in number of blocks
*
* @param subscriptionDDO - The DDO of the subscription
* @param serviceType - The service to fetch the duration from. Usually 'nft-sales' and 'nft-sales-proof'
* @param serviceReference - The service reference to fetch the duration from. Usually 'nft-sales' and 'nft-sales-proof'
*
* @throws {@link BadRequestException}
* @returns {@link Promise<number>} The duration in number of blocks
*/
public async getDuration(
subscriptionDDO: DDO,
serviceType: ServiceType = 'nft-sales',
serviceReference: number | ServiceType = 'nft-sales',
): Promise<number> {
// get the nft-sales service
let nftSalesService: Service
try {
nftSalesService = subscriptionDDO.findServiceByType(serviceType)
nftSalesService = subscriptionDDO.findServiceByReference(serviceReference)
} catch (e) {
if (e instanceof DDOServiceNotFoundError) {
throw new BadRequestException(
`${subscriptionDDO.id} does not contain an '${serviceType}' service`,
`${subscriptionDDO.id} does not contain an '${serviceReference}' service`,
)
} else {
throw e
}
}

// get the nft-holder condition
const transferNftCondition = DDO.findServiceConditionByName(nftSalesService, 'transferNFT')
const duration = transferNftCondition.parameters.find((p) => p.name === '_duration')

// get the duration parameter from the transferNFT condition
const duration = DDO.getParameterFromCondition(nftSalesService, 'transferNFT', '_duration')
// non-subscription nfts have no expiration
return Number(duration?.value) || 0
return Number(duration) || 0
}

/**
Expand Down
16 changes: 8 additions & 8 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -1386,10 +1386,10 @@
jose "^4.11.2"
passport-strategy "^1.0.0"

"@nevermined-io/sdk-dtp@^0.6.0":
version "0.6.0"
resolved "https://registry.yarnpkg.com/@nevermined-io/sdk-dtp/-/sdk-dtp-0.6.0.tgz#73a02849943f1b15b5e20dff828a6809d65afcad"
integrity sha512-LPqscWEtHCkBXgQDBmIbyhOhTa4N2XvvE9gt5JTRKfxEwaCh2r1i3nb6nka0ZLOfQ2021fqDsUfXj8zDYVLsZA==
"@nevermined-io/sdk-dtp@^0.7.0-rc4":
version "0.7.0-rc4"
resolved "https://registry.yarnpkg.com/@nevermined-io/sdk-dtp/-/sdk-dtp-0.7.0-rc4.tgz#f91050c2b7fdc44279e395f257a56bde0c91194e"
integrity sha512-0OQ/HlqOMWt2fv86+pjbZsXFmte2bkYIIQ7iVkdNzaQq+ly6WRAlS2/Y7YBrm2QTJT1rbAtwi+dKcGvl2e1ynw==
dependencies:
circomlibjs "^0.1.1"
eciesjs "^0.3.15"
Expand All @@ -1398,10 +1398,10 @@
snarkjs "^0.4.26"
web3-utils "^1.7.4"

"@nevermined-io/sdk@^2.0.0-rc2":
version "2.0.0-rc2"
resolved "https://registry.yarnpkg.com/@nevermined-io/sdk/-/sdk-2.0.0-rc2.tgz#d0ad7f629eaf67d04971759a9e225dc8cb81917d"
integrity sha512-wEUDbIL9Z1iRSxyEzpv3rFkZcm4VT9HXLZYv9zaLNnyzyaO+Y0piE5TwZ5q1o+WZMFB6WbcUiQTYbuLBfgnY4A==
"@nevermined-io/sdk@^2.0.0-rc7":
version "2.0.0-rc7"
resolved "https://registry.yarnpkg.com/@nevermined-io/sdk/-/sdk-2.0.0-rc7.tgz#fdd385991a5974b0ac01bccdfd51771f619e9093"
integrity sha512-fkJzd7vYxCTIiAHekPDeGJSPwPWZDierRKHOvqXlf+5EYeip3txNVEHWwe99MvV91sJM0yop6CSgqUxhZJYDyQ==
dependencies:
"@apollo/client" "^3.7.16"
assert "^2.0.0"
Expand Down

0 comments on commit b8fee00

Please sign in to comment.