]>
Commit | Line | Data |
---|---|---|
fadf619a | 1 | import * as validator from 'validator' |
74dc3bca | 2 | import { CONSTRAINTS_FIELDS } from '../../../initializers/constants' |
f37dc0dd | 3 | import { exists, isArray } from '../misc' |
938d3fa0 | 4 | import { truncate } from 'lodash' |
50d6de9c | 5 | import { isActivityPubUrlValid, isBaseActivityValid, setValidAttributedTo } from './misc' |
06a05d5f | 6 | import { isHostValid } from '../servers' |
fadf619a C |
7 | |
8 | function isActorEndpointsObjectValid (endpointObject: any) { | |
9 | return isActivityPubUrlValid(endpointObject.sharedInbox) | |
10 | } | |
11 | ||
12 | function isActorPublicKeyObjectValid (publicKeyObject: any) { | |
13 | return isActivityPubUrlValid(publicKeyObject.id) && | |
14 | isActivityPubUrlValid(publicKeyObject.owner) && | |
15 | isActorPublicKeyValid(publicKeyObject.publicKeyPem) | |
16 | } | |
17 | ||
18 | function isActorTypeValid (type: string) { | |
19 | return type === 'Person' || type === 'Application' || type === 'Group' | |
20 | } | |
21 | ||
22 | function isActorPublicKeyValid (publicKey: string) { | |
23 | return exists(publicKey) && | |
24 | typeof publicKey === 'string' && | |
25 | publicKey.startsWith('-----BEGIN PUBLIC KEY-----') && | |
e12a0092 | 26 | publicKey.indexOf('-----END PUBLIC KEY-----') !== -1 && |
01de67b9 | 27 | validator.isLength(publicKey, CONSTRAINTS_FIELDS.ACTORS.PUBLIC_KEY) |
fadf619a C |
28 | } |
29 | ||
0b5c385b | 30 | const actorNameAlphabet = '[ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\\-_.:]' |
f7cc67b4 | 31 | const actorNameRegExp = new RegExp(`^${actorNameAlphabet}+$`) |
fadf619a | 32 | function isActorPreferredUsernameValid (preferredUsername: string) { |
e12a0092 | 33 | return exists(preferredUsername) && validator.matches(preferredUsername, actorNameRegExp) |
50d6de9c C |
34 | } |
35 | ||
fadf619a C |
36 | function isActorPrivateKeyValid (privateKey: string) { |
37 | return exists(privateKey) && | |
38 | typeof privateKey === 'string' && | |
39 | privateKey.startsWith('-----BEGIN RSA PRIVATE KEY-----') && | |
e12a0092 C |
40 | // Sometimes there is a \n at the end, so just assert the string contains the end mark |
41 | privateKey.indexOf('-----END RSA PRIVATE KEY-----') !== -1 && | |
01de67b9 | 42 | validator.isLength(privateKey, CONSTRAINTS_FIELDS.ACTORS.PRIVATE_KEY) |
fadf619a C |
43 | } |
44 | ||
265ba139 C |
45 | function isActorObjectValid (actor: any) { |
46 | return exists(actor) && | |
47 | isActivityPubUrlValid(actor.id) && | |
48 | isActorTypeValid(actor.type) && | |
265ba139 | 49 | isActivityPubUrlValid(actor.inbox) && |
265ba139 C |
50 | isActorPreferredUsernameValid(actor.preferredUsername) && |
51 | isActivityPubUrlValid(actor.url) && | |
52 | isActorPublicKeyObjectValid(actor.publicKey) && | |
53 | isActorEndpointsObjectValid(actor.endpoints) && | |
f47776e2 | 54 | |
0b5c385b C |
55 | (!actor.outbox || isActivityPubUrlValid(actor.outbox)) && |
56 | (!actor.following || isActivityPubUrlValid(actor.following)) && | |
57 | (!actor.followers || isActivityPubUrlValid(actor.followers)) && | |
58 | ||
59 | setValidAttributedTo(actor) && | |
60 | // If this is a group (a channel), it should be attributed to an account | |
50d6de9c | 61 | // In PeerTube we use this to attach a video channel to a specific account |
0b5c385b | 62 | (actor.type !== 'Group' || actor.attributedTo.length !== 0) |
fadf619a C |
63 | } |
64 | ||
65 | function isActorFollowingCountValid (value: string) { | |
66 | return exists(value) && validator.isInt('' + value, { min: 0 }) | |
67 | } | |
68 | ||
69 | function isActorFollowersCountValid (value: string) { | |
70 | return exists(value) && validator.isInt('' + value, { min: 0 }) | |
71 | } | |
72 | ||
73 | function isActorDeleteActivityValid (activity: any) { | |
74 | return isBaseActivityValid(activity, 'Delete') | |
75 | } | |
76 | ||
848f499d C |
77 | function sanitizeAndCheckActorObject (object: any) { |
78 | normalizeActor(object) | |
f47776e2 | 79 | |
848f499d | 80 | return isActorObjectValid(object) |
265ba139 C |
81 | } |
82 | ||
938d3fa0 | 83 | function normalizeActor (actor: any) { |
575712a5 | 84 | if (!actor || !actor.url) return |
938d3fa0 C |
85 | |
86 | if (typeof actor.url !== 'string') { | |
87 | actor.url = actor.url.href || actor.url.url | |
88 | } | |
89 | ||
90 | if (actor.summary && typeof actor.summary === 'string') { | |
91 | actor.summary = truncate(actor.summary, { length: CONSTRAINTS_FIELDS.USERS.DESCRIPTION.max }) | |
92 | ||
93 | if (actor.summary.length < CONSTRAINTS_FIELDS.USERS.DESCRIPTION.min) { | |
94 | actor.summary = null | |
95 | } | |
96 | } | |
97 | ||
98 | return | |
99 | } | |
100 | ||
06a05d5f C |
101 | function isValidActorHandle (handle: string) { |
102 | if (!exists(handle)) return false | |
103 | ||
104 | const parts = handle.split('@') | |
105 | if (parts.length !== 2) return false | |
106 | ||
107 | return isHostValid(parts[1]) | |
108 | } | |
109 | ||
f37dc0dd C |
110 | function areValidActorHandles (handles: string[]) { |
111 | return isArray(handles) && handles.every(h => isValidActorHandle(h)) | |
112 | } | |
113 | ||
fadf619a C |
114 | // --------------------------------------------------------------------------- |
115 | ||
116 | export { | |
938d3fa0 | 117 | normalizeActor, |
f7cc67b4 | 118 | actorNameAlphabet, |
f37dc0dd | 119 | areValidActorHandles, |
fadf619a C |
120 | isActorEndpointsObjectValid, |
121 | isActorPublicKeyObjectValid, | |
122 | isActorTypeValid, | |
123 | isActorPublicKeyValid, | |
124 | isActorPreferredUsernameValid, | |
125 | isActorPrivateKeyValid, | |
265ba139 | 126 | isActorObjectValid, |
fadf619a C |
127 | isActorFollowingCountValid, |
128 | isActorFollowersCountValid, | |
50d6de9c | 129 | isActorDeleteActivityValid, |
848f499d | 130 | sanitizeAndCheckActorObject, |
06a05d5f | 131 | isValidActorHandle |
fadf619a | 132 | } |