diff options
Diffstat (limited to 'shared/extra-utils/requests')
-rw-r--r-- | shared/extra-utils/requests/activitypub.ts | 2 | ||||
-rw-r--r-- | shared/extra-utils/requests/check-api-params.ts | 19 | ||||
-rw-r--r-- | shared/extra-utils/requests/index.ts | 3 | ||||
-rw-r--r-- | shared/extra-utils/requests/requests.ts | 238 |
4 files changed, 124 insertions, 138 deletions
diff --git a/shared/extra-utils/requests/activitypub.ts b/shared/extra-utils/requests/activitypub.ts index ecd8ce823..4ae878384 100644 --- a/shared/extra-utils/requests/activitypub.ts +++ b/shared/extra-utils/requests/activitypub.ts | |||
@@ -1,7 +1,7 @@ | |||
1 | import { activityPubContextify } from '../../../server/helpers/activitypub' | ||
1 | import { doRequest } from '../../../server/helpers/requests' | 2 | import { doRequest } from '../../../server/helpers/requests' |
2 | import { HTTP_SIGNATURE } from '../../../server/initializers/constants' | 3 | import { HTTP_SIGNATURE } from '../../../server/initializers/constants' |
3 | import { buildGlobalHeaders } from '../../../server/lib/job-queue/handlers/utils/activitypub-http-utils' | 4 | import { buildGlobalHeaders } from '../../../server/lib/job-queue/handlers/utils/activitypub-http-utils' |
4 | import { activityPubContextify } from '../../../server/helpers/activitypub' | ||
5 | 5 | ||
6 | function makePOSTAPRequest (url: string, body: any, httpSignature: any, headers: any) { | 6 | function makePOSTAPRequest (url: string, body: any, httpSignature: any, headers: any) { |
7 | const options = { | 7 | const options = { |
diff --git a/shared/extra-utils/requests/check-api-params.ts b/shared/extra-utils/requests/check-api-params.ts index 7f5ff775c..26ba1e913 100644 --- a/shared/extra-utils/requests/check-api-params.ts +++ b/shared/extra-utils/requests/check-api-params.ts | |||
@@ -1,14 +1,13 @@ | |||
1 | import { HttpStatusCode } from '@shared/models' | ||
1 | import { makeGetRequest } from './requests' | 2 | import { makeGetRequest } from './requests' |
2 | import { immutableAssign } from '../miscs/miscs' | ||
3 | import { HttpStatusCode } from '../../../shared/core-utils/miscs/http-error-codes' | ||
4 | 3 | ||
5 | function checkBadStartPagination (url: string, path: string, token?: string, query = {}) { | 4 | function checkBadStartPagination (url: string, path: string, token?: string, query = {}) { |
6 | return makeGetRequest({ | 5 | return makeGetRequest({ |
7 | url, | 6 | url, |
8 | path, | 7 | path, |
9 | token, | 8 | token, |
10 | query: immutableAssign(query, { start: 'hello' }), | 9 | query: { ...query, start: 'hello' }, |
11 | statusCodeExpected: HttpStatusCode.BAD_REQUEST_400 | 10 | expectedStatus: HttpStatusCode.BAD_REQUEST_400 |
12 | }) | 11 | }) |
13 | } | 12 | } |
14 | 13 | ||
@@ -17,16 +16,16 @@ async function checkBadCountPagination (url: string, path: string, token?: strin | |||
17 | url, | 16 | url, |
18 | path, | 17 | path, |
19 | token, | 18 | token, |
20 | query: immutableAssign(query, { count: 'hello' }), | 19 | query: { ...query, count: 'hello' }, |
21 | statusCodeExpected: HttpStatusCode.BAD_REQUEST_400 | 20 | expectedStatus: HttpStatusCode.BAD_REQUEST_400 |
22 | }) | 21 | }) |
23 | 22 | ||
24 | await makeGetRequest({ | 23 | await makeGetRequest({ |
25 | url, | 24 | url, |
26 | path, | 25 | path, |
27 | token, | 26 | token, |
28 | query: immutableAssign(query, { count: 2000 }), | 27 | query: { ...query, count: 2000 }, |
29 | statusCodeExpected: HttpStatusCode.BAD_REQUEST_400 | 28 | expectedStatus: HttpStatusCode.BAD_REQUEST_400 |
30 | }) | 29 | }) |
31 | } | 30 | } |
32 | 31 | ||
@@ -35,8 +34,8 @@ function checkBadSortPagination (url: string, path: string, token?: string, quer | |||
35 | url, | 34 | url, |
36 | path, | 35 | path, |
37 | token, | 36 | token, |
38 | query: immutableAssign(query, { sort: 'hello' }), | 37 | query: { ...query, sort: 'hello' }, |
39 | statusCodeExpected: HttpStatusCode.BAD_REQUEST_400 | 38 | expectedStatus: HttpStatusCode.BAD_REQUEST_400 |
40 | }) | 39 | }) |
41 | } | 40 | } |
42 | 41 | ||
diff --git a/shared/extra-utils/requests/index.ts b/shared/extra-utils/requests/index.ts new file mode 100644 index 000000000..501163f92 --- /dev/null +++ b/shared/extra-utils/requests/index.ts | |||
@@ -0,0 +1,3 @@ | |||
1 | // Don't include activitypub that import stuff from server | ||
2 | export * from './check-api-params' | ||
3 | export * from './requests' | ||
diff --git a/shared/extra-utils/requests/requests.ts b/shared/extra-utils/requests/requests.ts index 38e24d897..70f790222 100644 --- a/shared/extra-utils/requests/requests.ts +++ b/shared/extra-utils/requests/requests.ts | |||
@@ -1,103 +1,82 @@ | |||
1 | /* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/no-floating-promises */ | 1 | /* eslint-disable @typescript-eslint/no-floating-promises */ |
2 | 2 | ||
3 | import { decode } from 'querystring' | ||
3 | import * as request from 'supertest' | 4 | import * as request from 'supertest' |
4 | import { buildAbsoluteFixturePath, root } from '../miscs/miscs' | ||
5 | import { isAbsolute, join } from 'path' | ||
6 | import { URL } from 'url' | 5 | import { URL } from 'url' |
7 | import { decode } from 'querystring' | 6 | import { HttpStatusCode } from '@shared/models' |
8 | import { HttpStatusCode } from '../../../shared/core-utils/miscs/http-error-codes' | 7 | import { buildAbsoluteFixturePath } from '../miscs/tests' |
9 | 8 | ||
10 | function get4KFileUrl () { | 9 | export type CommonRequestParams = { |
11 | return 'https://download.cpy.re/peertube/4k_file.txt' | 10 | url: string |
11 | path?: string | ||
12 | contentType?: string | ||
13 | range?: string | ||
14 | redirects?: number | ||
15 | accept?: string | ||
16 | host?: string | ||
17 | token?: string | ||
18 | headers?: { [ name: string ]: string } | ||
19 | type?: string | ||
20 | xForwardedFor?: string | ||
21 | expectedStatus?: HttpStatusCode | ||
12 | } | 22 | } |
13 | 23 | ||
14 | function makeRawRequest (url: string, statusCodeExpected?: HttpStatusCode, range?: string) { | 24 | function makeRawRequest (url: string, expectedStatus?: HttpStatusCode, range?: string) { |
15 | const { host, protocol, pathname } = new URL(url) | 25 | const { host, protocol, pathname } = new URL(url) |
16 | 26 | ||
17 | return makeGetRequest({ url: `${protocol}//${host}`, path: pathname, statusCodeExpected, range }) | 27 | return makeGetRequest({ url: `${protocol}//${host}`, path: pathname, expectedStatus, range }) |
18 | } | 28 | } |
19 | 29 | ||
20 | function makeGetRequest (options: { | 30 | function makeGetRequest (options: CommonRequestParams & { |
21 | url: string | ||
22 | path?: string | ||
23 | query?: any | 31 | query?: any |
24 | token?: string | ||
25 | statusCodeExpected?: HttpStatusCode | ||
26 | contentType?: string | ||
27 | range?: string | ||
28 | redirects?: number | ||
29 | accept?: string | ||
30 | }) { | 32 | }) { |
31 | if (!options.statusCodeExpected) options.statusCodeExpected = HttpStatusCode.BAD_REQUEST_400 | ||
32 | if (options.contentType === undefined) options.contentType = 'application/json' | ||
33 | |||
34 | const req = request(options.url).get(options.path) | 33 | const req = request(options.url).get(options.path) |
34 | .query(options.query) | ||
35 | 35 | ||
36 | if (options.contentType) req.set('Accept', options.contentType) | 36 | return buildRequest(req, { contentType: 'application/json', expectedStatus: HttpStatusCode.BAD_REQUEST_400, ...options }) |
37 | if (options.token) req.set('Authorization', 'Bearer ' + options.token) | ||
38 | if (options.query) req.query(options.query) | ||
39 | if (options.range) req.set('Range', options.range) | ||
40 | if (options.accept) req.set('Accept', options.accept) | ||
41 | if (options.redirects) req.redirects(options.redirects) | ||
42 | |||
43 | return req.expect(options.statusCodeExpected) | ||
44 | } | 37 | } |
45 | 38 | ||
46 | function makeDeleteRequest (options: { | 39 | function makeHTMLRequest (url: string, path: string) { |
47 | url: string | 40 | return makeGetRequest({ |
48 | path: string | 41 | url, |
49 | token?: string | 42 | path, |
50 | statusCodeExpected?: HttpStatusCode | 43 | accept: 'text/html', |
51 | }) { | 44 | expectedStatus: HttpStatusCode.OK_200 |
52 | if (!options.statusCodeExpected) options.statusCodeExpected = HttpStatusCode.BAD_REQUEST_400 | 45 | }) |
46 | } | ||
53 | 47 | ||
54 | const req = request(options.url) | 48 | function makeActivityPubGetRequest (url: string, path: string, expectedStatus = HttpStatusCode.OK_200) { |
55 | .delete(options.path) | 49 | return makeGetRequest({ |
56 | .set('Accept', 'application/json') | 50 | url, |
51 | path, | ||
52 | expectedStatus: expectedStatus, | ||
53 | accept: 'application/activity+json,text/html;q=0.9,\\*/\\*;q=0.8' | ||
54 | }) | ||
55 | } | ||
57 | 56 | ||
58 | if (options.token) req.set('Authorization', 'Bearer ' + options.token) | 57 | function makeDeleteRequest (options: CommonRequestParams) { |
58 | const req = request(options.url).delete(options.path) | ||
59 | 59 | ||
60 | return req.expect(options.statusCodeExpected) | 60 | return buildRequest(req, { accept: 'application/json', expectedStatus: HttpStatusCode.BAD_REQUEST_400, ...options }) |
61 | } | 61 | } |
62 | 62 | ||
63 | function makeUploadRequest (options: { | 63 | function makeUploadRequest (options: CommonRequestParams & { |
64 | url: string | ||
65 | method?: 'POST' | 'PUT' | 64 | method?: 'POST' | 'PUT' |
66 | path: string | 65 | |
67 | token?: string | ||
68 | fields: { [ fieldName: string ]: any } | 66 | fields: { [ fieldName: string ]: any } |
69 | attaches?: { [ attachName: string ]: any | any[] } | 67 | attaches?: { [ attachName: string ]: any | any[] } |
70 | statusCodeExpected?: HttpStatusCode | ||
71 | }) { | 68 | }) { |
72 | if (!options.statusCodeExpected) options.statusCodeExpected = HttpStatusCode.BAD_REQUEST_400 | 69 | let req = options.method === 'PUT' |
73 | 70 | ? request(options.url).put(options.path) | |
74 | let req: request.Test | 71 | : request(options.url).post(options.path) |
75 | if (options.method === 'PUT') { | ||
76 | req = request(options.url).put(options.path) | ||
77 | } else { | ||
78 | req = request(options.url).post(options.path) | ||
79 | } | ||
80 | |||
81 | req.set('Accept', 'application/json') | ||
82 | 72 | ||
83 | if (options.token) req.set('Authorization', 'Bearer ' + options.token) | 73 | req = buildRequest(req, { accept: 'application/json', expectedStatus: HttpStatusCode.BAD_REQUEST_400, ...options }) |
84 | 74 | ||
85 | Object.keys(options.fields).forEach(field => { | 75 | buildFields(req, options.fields) |
86 | const value = options.fields[field] | ||
87 | |||
88 | if (value === undefined) return | ||
89 | |||
90 | if (Array.isArray(value)) { | ||
91 | for (let i = 0; i < value.length; i++) { | ||
92 | req.field(field + '[' + i + ']', value[i]) | ||
93 | } | ||
94 | } else { | ||
95 | req.field(field, value) | ||
96 | } | ||
97 | }) | ||
98 | 76 | ||
99 | Object.keys(options.attaches || {}).forEach(attach => { | 77 | Object.keys(options.attaches || {}).forEach(attach => { |
100 | const value = options.attaches[attach] | 78 | const value = options.attaches[attach] |
79 | |||
101 | if (Array.isArray(value)) { | 80 | if (Array.isArray(value)) { |
102 | req.attach(attach, buildAbsoluteFixturePath(value[0]), value[1]) | 81 | req.attach(attach, buildAbsoluteFixturePath(value[0]), value[1]) |
103 | } else { | 82 | } else { |
@@ -105,27 +84,16 @@ function makeUploadRequest (options: { | |||
105 | } | 84 | } |
106 | }) | 85 | }) |
107 | 86 | ||
108 | return req.expect(options.statusCodeExpected) | 87 | return req |
109 | } | 88 | } |
110 | 89 | ||
111 | function makePostBodyRequest (options: { | 90 | function makePostBodyRequest (options: CommonRequestParams & { |
112 | url: string | ||
113 | path: string | ||
114 | token?: string | ||
115 | fields?: { [ fieldName: string ]: any } | 91 | fields?: { [ fieldName: string ]: any } |
116 | statusCodeExpected?: HttpStatusCode | ||
117 | }) { | 92 | }) { |
118 | if (!options.fields) options.fields = {} | 93 | const req = request(options.url).post(options.path) |
119 | if (!options.statusCodeExpected) options.statusCodeExpected = HttpStatusCode.BAD_REQUEST_400 | 94 | .send(options.fields) |
120 | |||
121 | const req = request(options.url) | ||
122 | .post(options.path) | ||
123 | .set('Accept', 'application/json') | ||
124 | 95 | ||
125 | if (options.token) req.set('Authorization', 'Bearer ' + options.token) | 96 | return buildRequest(req, { accept: 'application/json', expectedStatus: HttpStatusCode.BAD_REQUEST_400, ...options }) |
126 | |||
127 | return req.send(options.fields) | ||
128 | .expect(options.statusCodeExpected) | ||
129 | } | 97 | } |
130 | 98 | ||
131 | function makePutBodyRequest (options: { | 99 | function makePutBodyRequest (options: { |
@@ -133,59 +101,29 @@ function makePutBodyRequest (options: { | |||
133 | path: string | 101 | path: string |
134 | token?: string | 102 | token?: string |
135 | fields: { [ fieldName: string ]: any } | 103 | fields: { [ fieldName: string ]: any } |
136 | statusCodeExpected?: HttpStatusCode | 104 | expectedStatus?: HttpStatusCode |
137 | }) { | 105 | }) { |
138 | if (!options.statusCodeExpected) options.statusCodeExpected = HttpStatusCode.BAD_REQUEST_400 | 106 | const req = request(options.url).put(options.path) |
139 | 107 | .send(options.fields) | |
140 | const req = request(options.url) | ||
141 | .put(options.path) | ||
142 | .set('Accept', 'application/json') | ||
143 | 108 | ||
144 | if (options.token) req.set('Authorization', 'Bearer ' + options.token) | 109 | return buildRequest(req, { accept: 'application/json', expectedStatus: HttpStatusCode.BAD_REQUEST_400, ...options }) |
145 | |||
146 | return req.send(options.fields) | ||
147 | .expect(options.statusCodeExpected) | ||
148 | } | 110 | } |
149 | 111 | ||
150 | function makeHTMLRequest (url: string, path: string) { | 112 | function decodeQueryString (path: string) { |
151 | return request(url) | 113 | return decode(path.split('?')[1]) |
152 | .get(path) | ||
153 | .set('Accept', 'text/html') | ||
154 | .expect(HttpStatusCode.OK_200) | ||
155 | } | 114 | } |
156 | 115 | ||
157 | function updateImageRequest (options: { | 116 | function unwrapBody <T> (test: request.Test): Promise<T> { |
158 | url: string | 117 | return test.then(res => res.body) |
159 | path: string | ||
160 | accessToken: string | ||
161 | fixture: string | ||
162 | fieldname: string | ||
163 | }) { | ||
164 | let filePath = '' | ||
165 | if (isAbsolute(options.fixture)) { | ||
166 | filePath = options.fixture | ||
167 | } else { | ||
168 | filePath = join(root(), 'server', 'tests', 'fixtures', options.fixture) | ||
169 | } | ||
170 | |||
171 | return makeUploadRequest({ | ||
172 | url: options.url, | ||
173 | path: options.path, | ||
174 | token: options.accessToken, | ||
175 | fields: {}, | ||
176 | attaches: { [options.fieldname]: filePath }, | ||
177 | statusCodeExpected: HttpStatusCode.OK_200 | ||
178 | }) | ||
179 | } | 118 | } |
180 | 119 | ||
181 | function decodeQueryString (path: string) { | 120 | function unwrapText (test: request.Test): Promise<string> { |
182 | return decode(path.split('?')[1]) | 121 | return test.then(res => res.text) |
183 | } | 122 | } |
184 | 123 | ||
185 | // --------------------------------------------------------------------------- | 124 | // --------------------------------------------------------------------------- |
186 | 125 | ||
187 | export { | 126 | export { |
188 | get4KFileUrl, | ||
189 | makeHTMLRequest, | 127 | makeHTMLRequest, |
190 | makeGetRequest, | 128 | makeGetRequest, |
191 | decodeQueryString, | 129 | decodeQueryString, |
@@ -194,5 +132,51 @@ export { | |||
194 | makePutBodyRequest, | 132 | makePutBodyRequest, |
195 | makeDeleteRequest, | 133 | makeDeleteRequest, |
196 | makeRawRequest, | 134 | makeRawRequest, |
197 | updateImageRequest | 135 | makeActivityPubGetRequest, |
136 | unwrapBody, | ||
137 | unwrapText | ||
138 | } | ||
139 | |||
140 | // --------------------------------------------------------------------------- | ||
141 | |||
142 | function buildRequest (req: request.Test, options: CommonRequestParams) { | ||
143 | if (options.contentType) req.set('Accept', options.contentType) | ||
144 | if (options.token) req.set('Authorization', 'Bearer ' + options.token) | ||
145 | if (options.range) req.set('Range', options.range) | ||
146 | if (options.accept) req.set('Accept', options.accept) | ||
147 | if (options.host) req.set('Host', options.host) | ||
148 | if (options.redirects) req.redirects(options.redirects) | ||
149 | if (options.expectedStatus) req.expect(options.expectedStatus) | ||
150 | if (options.xForwardedFor) req.set('X-Forwarded-For', options.xForwardedFor) | ||
151 | if (options.type) req.type(options.type) | ||
152 | |||
153 | Object.keys(options.headers || {}).forEach(name => { | ||
154 | req.set(name, options.headers[name]) | ||
155 | }) | ||
156 | |||
157 | return req | ||
158 | } | ||
159 | |||
160 | function buildFields (req: request.Test, fields: { [ fieldName: string ]: any }, namespace?: string) { | ||
161 | if (!fields) return | ||
162 | |||
163 | let formKey: string | ||
164 | |||
165 | for (const key of Object.keys(fields)) { | ||
166 | if (namespace) formKey = `${namespace}[${key}]` | ||
167 | else formKey = key | ||
168 | |||
169 | if (fields[key] === undefined) continue | ||
170 | |||
171 | if (Array.isArray(fields[key]) && fields[key].length === 0) { | ||
172 | req.field(key, null) | ||
173 | continue | ||
174 | } | ||
175 | |||
176 | if (fields[key] !== null && typeof fields[key] === 'object') { | ||
177 | buildFields(req, fields[key], formKey) | ||
178 | } else { | ||
179 | req.field(formKey, fields[key]) | ||
180 | } | ||
181 | } | ||
198 | } | 182 | } |