Skip to content

Commit

Permalink
Fix weird bug where CPU jumps and stays to 100%
Browse files Browse the repository at this point in the history
Seems related to lazy import of custom-jsonld-signature
So we refactored jsonld function calls a little bit
  • Loading branch information
Chocobozzz committed Nov 24, 2023
1 parent f93bc6a commit b017d4d
Show file tree
Hide file tree
Showing 5 changed files with 108 additions and 105 deletions.
3 changes: 2 additions & 1 deletion server/core/helpers/activity-pub-utils.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { ContextType } from '@peertube/peertube-models'
import { ACTIVITY_PUB } from '@server/initializers/constants.js'
import { buildDigest, signJsonLDObject } from './peertube-crypto.js'
import { buildDigest } from './peertube-crypto.js'
import type { signJsonLDObject } from './peertube-jsonld.js'

export type ContextFilter = <T> (arg: T) => Promise<T>

Expand Down
103 changes: 1 addition & 102 deletions server/core/helpers/peertube-crypto.ts
Original file line number Diff line number Diff line change
@@ -1,13 +1,11 @@
import httpSignature from '@peertube/http-signature'
import { sha256 } from '@peertube/peertube-node-utils'
import { createCipheriv, createDecipheriv, createSign, createVerify } from 'crypto'
import { createCipheriv, createDecipheriv } from 'crypto'
import { Request } from 'express'
import cloneDeep from 'lodash-es/cloneDeep.js'
import { BCRYPT_SALT_SIZE, ENCRYPTION, HTTP_SIGNATURE, PRIVATE_RSA_KEY_SIZE } from '../initializers/constants.js'
import { MActor } from '../types/models/index.js'
import { generateRSAKeyPairPromise, randomBytesPromise, scryptPromise } from './core-utils.js'
import { logger } from './logger.js'
import { assertIsInWorkerThread } from './threads.js'

function createPrivateAndPublicKeys () {
logger.info('Generating a RSA key...')
Expand Down Expand Up @@ -66,66 +64,6 @@ function parseHTTPSignature (req: Request, clockSkew?: number) {
return parsed
}

// ---------------------------------------------------------------------------
// JSONLD
// ---------------------------------------------------------------------------

function isJsonLDSignatureVerified (fromActor: MActor, signedDocument: any): Promise<boolean> {
if (signedDocument.signature.type === 'RsaSignature2017') {
return isJsonLDRSA2017Verified(fromActor, signedDocument)
}

logger.warn('Unknown JSON LD signature %s.', signedDocument.signature.type, signedDocument)

return Promise.resolve(false)
}

// Backward compatibility with "other" implementations
async function isJsonLDRSA2017Verified (fromActor: MActor, signedDocument: any) {
const [ documentHash, optionsHash ] = await Promise.all([
createDocWithoutSignatureHash(signedDocument),
createSignatureHash(signedDocument.signature)
])

const toVerify = optionsHash + documentHash

const verify = createVerify('RSA-SHA256')
verify.update(toVerify, 'utf8')

return verify.verify(fromActor.publicKey, signedDocument.signature.signatureValue, 'base64')
}

async function signJsonLDObject <T> (options: {
byActor: { url: string, privateKey: string }
data: T
disableWorkerThreadAssertion?: boolean
}) {
const { byActor, data, disableWorkerThreadAssertion = false } = options

if (!disableWorkerThreadAssertion) assertIsInWorkerThread()

const signature = {
type: 'RsaSignature2017',
creator: byActor.url,
created: new Date().toISOString()
}

const [ documentHash, optionsHash ] = await Promise.all([
createDocWithoutSignatureHash(data),
createSignatureHash(signature)
])

const toSign = optionsHash + documentHash

const sign = createSign('RSA-SHA256')
sign.update(toSign, 'utf8')

const signatureValue = sign.sign(byActor.privateKey, 'base64')
Object.assign(signature, { signatureValue })

return Object.assign(data, { signature })
}

// ---------------------------------------------------------------------------

function buildDigest (body: any) {
Expand Down Expand Up @@ -169,49 +107,10 @@ export {
parseHTTPSignature,
isHTTPSignatureVerified,
buildDigest,
isJsonLDSignatureVerified,
comparePassword,
createPrivateAndPublicKeys,
cryptPassword,
signJsonLDObject,

encrypt,
decrypt
}

// ---------------------------------------------------------------------------

async function hashObject (obj: any): Promise<any> {
const { jsonld } = await import('./custom-jsonld-signature.js')

const res = await (jsonld as any).promises.normalize(obj, {
safe: false,
algorithm: 'URDNA2015',
format: 'application/n-quads'
})

return sha256(res)
}

function createSignatureHash (signature: any) {
const signatureCopy = cloneDeep(signature)
Object.assign(signatureCopy, {
'@context': [
'https://w3id.org/security/v1',
{ RsaSignature2017: 'https://w3id.org/security#RsaSignature2017' }
]
})

delete signatureCopy.type
delete signatureCopy.id
delete signatureCopy.signatureValue

return hashObject(signatureCopy)
}

function createDocWithoutSignatureHash (doc: any) {
const docWithoutSignature = cloneDeep(doc)
delete docWithoutSignature.signature

return hashObject(docWithoutSignature)
}
100 changes: 100 additions & 0 deletions server/core/helpers/peertube-jsonld.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
import { sha256 } from '@peertube/peertube-node-utils'
import { createSign, createVerify } from 'crypto'
import cloneDeep from 'lodash-es/cloneDeep.js'
import { MActor } from '../types/models/index.js'
import { logger } from './logger.js'
import { assertIsInWorkerThread } from './threads.js'
import { jsonld } from './custom-jsonld-signature.js'

export function isJsonLDSignatureVerified (fromActor: MActor, signedDocument: any): Promise<boolean> {
if (signedDocument.signature.type === 'RsaSignature2017') {
return isJsonLDRSA2017Verified(fromActor, signedDocument)
}

logger.warn('Unknown JSON LD signature %s.', signedDocument.signature.type, signedDocument)

return Promise.resolve(false)
}

// Backward compatibility with "other" implementations
export async function isJsonLDRSA2017Verified (fromActor: MActor, signedDocument: any) {
const [ documentHash, optionsHash ] = await Promise.all([
createDocWithoutSignatureHash(signedDocument),
createSignatureHash(signedDocument.signature)
])

const toVerify = optionsHash + documentHash

const verify = createVerify('RSA-SHA256')
verify.update(toVerify, 'utf8')

return verify.verify(fromActor.publicKey, signedDocument.signature.signatureValue, 'base64')
}

export async function signJsonLDObject <T> (options: {
byActor: { url: string, privateKey: string }
data: T
disableWorkerThreadAssertion?: boolean
}) {
const { byActor, data, disableWorkerThreadAssertion = false } = options

if (!disableWorkerThreadAssertion) assertIsInWorkerThread()

const signature = {
type: 'RsaSignature2017',
creator: byActor.url,
created: new Date().toISOString()
}

const [ documentHash, optionsHash ] = await Promise.all([
createDocWithoutSignatureHash(data),
createSignatureHash(signature)
])

const toSign = optionsHash + documentHash

const sign = createSign('RSA-SHA256')
sign.update(toSign, 'utf8')

const signatureValue = sign.sign(byActor.privateKey, 'base64')
Object.assign(signature, { signatureValue })

return Object.assign(data, { signature })
}

// ---------------------------------------------------------------------------
// Private
// ---------------------------------------------------------------------------

async function hashObject (obj: any): Promise<any> {
const res = await (jsonld as any).promises.normalize(obj, {
safe: false,
algorithm: 'URDNA2015',
format: 'application/n-quads'
})

return sha256(res)
}

function createSignatureHash (signature: any) {
const signatureCopy = cloneDeep(signature)
Object.assign(signatureCopy, {
'@context': [
'https://w3id.org/security/v1',
{ RsaSignature2017: 'https://w3id.org/security#RsaSignature2017' }
]
})

delete signatureCopy.type
delete signatureCopy.id
delete signatureCopy.signatureValue

return hashObject(signatureCopy)
}

function createDocWithoutSignatureHash (doc: any) {
const docWithoutSignature = cloneDeep(doc)
delete docWithoutSignature.signature

return hashObject(docWithoutSignature)
}
2 changes: 1 addition & 1 deletion server/core/lib/worker/workers/sign-json-ld-object.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
import { signJsonLDObject } from '@server/helpers/peertube-crypto.js'
import { signJsonLDObject } from '@server/helpers/peertube-jsonld.js'

export default signJsonLDObject
5 changes: 4 additions & 1 deletion server/core/middlewares/activitypub.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import { getAPId } from '@server/lib/activitypub/activity.js'
import { wrapWithSpanAndContext } from '@server/lib/opentelemetry/tracing.js'
import { ActivityDelete, ActivityPubSignature, HttpStatusCode } from '@peertube/peertube-models'
import { logger } from '../helpers/logger.js'
import { isHTTPSignatureVerified, isJsonLDSignatureVerified, parseHTTPSignature } from '../helpers/peertube-crypto.js'
import { isHTTPSignatureVerified, parseHTTPSignature } from '../helpers/peertube-crypto.js'
import { ACCEPT_HEADERS, ACTIVITY_PUB, HTTP_SIGNATURE } from '../initializers/constants.js'
import { getOrCreateAPActor, loadActorUrlOrGetFromWebfinger } from '../lib/activitypub/actors/index.js'

Expand Down Expand Up @@ -122,6 +122,9 @@ async function checkHttpSignature (req: Request, res: Response) {
}

async function checkJsonLDSignature (req: Request, res: Response) {
// Lazy load the module as it's quite big with json.ld dependency
const { isJsonLDSignatureVerified } = await import('../helpers/peertube-jsonld.js')

return wrapWithSpanAndContext('peertube.activitypub.JSONLDSignature', async () => {
const signatureObject: ActivityPubSignature = req.body.signature

Expand Down

0 comments on commit b017d4d

Please sign in to comment.