Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Apify Client v1 #146

Merged
merged 136 commits into from
Oct 22, 2020
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
136 commits
Select commit Hold shift + click to select a range
1ffc2fd
Reimplement first actor endpoints
mnmkng Sep 4, 2019
98dd092
Progress commit
mnmkng Dec 31, 2019
6fc594c
Completed tests for actors endpoint.
petrpatek Dec 31, 2019
3f0fa6c
Task endpoint partial rewrote.
petrpatek Dec 31, 2019
819012b
rewrote users.
petrpatek Dec 31, 2019
a86bd4d
rewrote logs.js
petrpatek Dec 31, 2019
2208ea5
started dataset rewrote.
petrpatek Jan 2, 2020
f66a4ba
keyValuesStores basic methods rewrote.
petrpatek Jan 6, 2020
a5ffb60
rewrote request_queues.js
petrpatek Jan 6, 2020
634af0e
rewrote webhooks.js
petrpatek Jan 6, 2020
cc7057d
rewrote webhook_dispatches.js.
petrpatek Jan 6, 2020
503def2
rewrote dataset tests. Removed Crawler.
petrpatek Jan 11, 2020
165520b
rewrote key_value_stores.js tests
petrpatek Jan 13, 2020
ec90fdf
xited tests that are not working, because of the client implementatio…
petrpatek Jan 13, 2020
5a1c71a
stuck in rewriting to async-retry.
petrpatek Jan 14, 2020
80d1ff6
fixed exponential backoff... I guess
petrpatek Jan 28, 2020
e8a55fa
make working browser POC.
petrpatek Jan 29, 2020
00c7638
make browser test work.
petrpatek Feb 14, 2020
ce86a42
used brotli compressor to reduce the bundle size.
petrpatek Feb 17, 2020
bebeb3b
Merge remote-tracking branch 'origin/master' into feature/isomorphic
petrpatek Feb 17, 2020
e96514c
introduced latest features from master.
petrpatek Feb 17, 2020
4887c61
refactored code. split lodash to functions because of the bundle size.
petrpatek Feb 17, 2020
3773cc4
removed deprecated method from task
petrpatek Feb 17, 2020
f953264
fixed lint err
petrpatek Feb 17, 2020
c72697b
added workflows.
petrpatek Feb 17, 2020
1622133
improved build script to build files for the test.
petrpatek Feb 17, 2020
9b96984
removed prepare script.
petrpatek Feb 17, 2020
f523460
split build to two parts.
petrpatek Feb 17, 2020
d24a225
improved webpack browser build.
petrpatek Feb 17, 2020
b9938df
removed unnecessary dependencies and engines.
petrpatek Feb 17, 2020
7eea3e4
migrated to jest due to performance
petrpatek Feb 17, 2020
24974f7
fixed lint
petrpatek Feb 17, 2020
a88d4a3
made tests green again,
petrpatek Feb 18, 2020
29a66fc
started abstract class.
petrpatek Feb 18, 2020
9d2cfe5
Comments from PR.
petrpatek Feb 20, 2020
81b3d74
renamed edpoint to resource. Deleted extra expbackoff logic from data…
petrpatek Feb 20, 2020
12208e0
changed webpack mode.
petrpatek Feb 20, 2020
af99d39
added defaults.
petrpatek Feb 20, 2020
9ab5568
improved text decoding.
petrpatek Feb 20, 2020
2bed4fa
documented listActors endpoint.
petrpatek Feb 20, 2020
d827a7d
updated docs for actor endpoint.
petrpatek Feb 21, 2020
d7fded2
improved dataset endpoint docs.
petrpatek Feb 21, 2020
2df3d35
documented dataset endpoint.
petrpatek Feb 21, 2020
f164895
improved KV stores docs.
petrpatek Feb 21, 2020
0f1082d
logs docs slightly improved.
petrpatek Feb 21, 2020
c39b07a
Request queue docs improved.
petrpatek Feb 21, 2020
4963927
improved tasks docs.
petrpatek Feb 22, 2020
a29bd00
documented user endpoint.
petrpatek Feb 22, 2020
d763168
documented webhook dispatches
petrpatek Feb 22, 2020
3ca1579
documented webhooks endpoint.
petrpatek Feb 22, 2020
89c8589
added build docs action.
petrpatek Feb 24, 2020
d2d3744
improved docs strategy.
petrpatek Feb 24, 2020
616b6c8
fixed yml
petrpatek Feb 24, 2020
8456f68
fix
petrpatek Feb 24, 2020
ea957b4
refactored actions.
petrpatek Feb 24, 2020
c7a0f07
refactored actions.
petrpatek Feb 24, 2020
b958310
renamed.
petrpatek Feb 24, 2020
02d4a68
on added
petrpatek Feb 24, 2020
0321852
increased version added my parameter to list actors endpoint.
petrpatek Feb 24, 2020
4ccc2b1
added dist
petrpatek Feb 24, 2020
20173bc
fix because of null from integration tests.
petrpatek Feb 24, 2020
972d5d6
renamed tilde separated to slash separated.
petrpatek Feb 25, 2020
dcded5e
added missing return.
petrpatek Feb 25, 2020
464d04b
Fixed docs. Improved README.md. Added CHANGELOG.md. Bump version.
petrpatek Mar 3, 2020
6f50096
Added possibility to remove content-type header with null and fixed J…
petrpatek Mar 22, 2020
1e1ea72
fixed getValue
petrpatek Mar 22, 2020
edb560d
Implementing schedules API
mtrunkat Mar 24, 2020
232d1cc
Adding tests and linting
mtrunkat Mar 24, 2020
2b60154
fixed header removal and log endpoint.
petrpatek Mar 24, 2020
d9c2e47
Merge pull request #139 from apifytech/feature/schedules
petrpatek Mar 24, 2020
f1b3881
improved log tests increased beta version.
petrpatek Mar 24, 2020
3642fe1
Fix typo in version
mnmkng Mar 25, 2020
c041286
Fix lint
mnmkng Mar 25, 2020
4f66e4f
Fixed tasks endpoint
petrpatek Mar 30, 2020
678c1d7
fixed signing url. task input handling.
petrpatek Mar 31, 2020
c33bc35
Merge branch 'feature/isomorphic' of github.com:apifytech/apify-clien…
petrpatek Mar 31, 2020
91eee0d
removed babel for nodeJs.
petrpatek Apr 1, 2020
32c97eb
improved browser header to don't trigger error.
petrpatek Apr 1, 2020
28cac26
fixed docs.
petrpatek Apr 1, 2020
5782a2a
bump version
petrpatek Apr 1, 2020
a6185ce
Some changes
mnmkng Apr 19, 2020
61f6223
Not great, not terrible
mnmkng Jun 9, 2020
16bd1a5
Add missing clients
mnmkng Jun 9, 2020
b4b6464
Add clients to index
mnmkng Jun 9, 2020
1eb64b6
Dataset tests work
mnmkng Jun 10, 2020
cabb896
Dataset tests pass
mnmkng Jun 12, 2020
926616a
Key value stores
mnmkng Jun 17, 2020
5099c2d
Key value stores finished
mnmkng Jun 22, 2020
cca7c8d
Logs finished
mnmkng Jun 22, 2020
0c067f6
Request queue finished
mnmkng Jun 22, 2020
61f14bc
Schedules finished
mnmkng Jun 22, 2020
ac7dd9e
Tasks finished
mnmkng Jun 23, 2020
8a17bad
Users finished
mnmkng Jun 23, 2020
c019b50
Webhooks finished
mnmkng Jun 23, 2020
8fa92f4
Webhook Dispatches finished
mnmkng Jun 23, 2020
2b15949
Remove old stuff, refactor, fix tests
mnmkng Jun 23, 2020
6baeee2
Test Statistics
mnmkng Jun 23, 2020
ad8027c
Bump version
mnmkng Jun 24, 2020
cc5452b
Minor improvements
mnmkng Jul 22, 2020
f25f65b
Add response parsing logic
mnmkng Jul 22, 2020
77f3094
Add status params to actor runs endpoint
mnmkng Aug 25, 2020
785e9ce
Fix axios defaults
mnmkng Sep 2, 2020
9700d2d
Bump version
mnmkng Sep 2, 2020
ba63a57
Update KVS, bump version
mnmkng Sep 5, 2020
27fb35b
Lint files
mnmkng Sep 5, 2020
dd46dcd
Some changes
mnmkng Sep 8, 2020
04cc125
Merge branch 'chore/refactoring' of github.com:apify/apify-client-js …
jbartadev Sep 9, 2020
6b4340c
Minor improvements
mnmkng Sep 18, 2020
adb0a71
Merge branch 'chore/refactoring' of github.com:apify/apify-client-js …
jbartadev Sep 21, 2020
52a90e3
Add waitForFinish function to Run and Build
mnmkng Oct 2, 2020
c3caee2
Add waitForFinish tests
mnmkng Oct 2, 2020
27ae925
Fix lint
mnmkng Oct 2, 2020
fbf0b01
Merge branch 'chore/refactoring' of github.com:apify/apify-client-js …
jbartadev Oct 2, 2020
3cc3b4a
Added call method to actor
jbartadev Oct 2, 2020
4f9f53b
Fix invalid return value in getRecord
mnmkng Oct 5, 2020
5724d28
Merge branch 'chore/refactoring' of github.com:apify/apify-client-js …
jbartadev Oct 5, 2020
b8ce886
Fix setRecord with stringified JSON, remove lodash
mnmkng Oct 6, 2020
2b07cc5
Added private method _call to resource client
jbartadev Oct 6, 2020
ad5d99c
Added call method to task client, update actor call method
jbartadev Oct 6, 2020
ca40649
Bump version
mnmkng Oct 6, 2020
e63d263
Added call method tests
jbartadev Oct 6, 2020
13a6ed8
Merge branch 'chore/refactoring' into refactor/v1
jbartadev Oct 6, 2020
3850a50
fixed condition
jbartadev Oct 6, 2020
2df33ef
further refactoring
jbartadev Oct 7, 2020
0c1064e
small changes & made tests green again
jbartadev Oct 9, 2020
cdb1e95
Merge pull request #144 from apify/refactor/v1
mnmkng Oct 9, 2020
a715a36
Bump version
mnmkng Oct 9, 2020
304500d
Change signature of start and call methods
mnmkng Oct 13, 2020
0c185cd
Update comment
mnmkng Oct 13, 2020
ff0dea9
Change metamorph signature
mnmkng Oct 13, 2020
631b1ee
Improve ApifyApiError
mnmkng Oct 14, 2020
a8be029
Fix trailing slash and add some improvements
mnmkng Oct 14, 2020
93db7fb
Bump version
mnmkng Oct 14, 2020
aea26b8
Fix v2 in baseUrl and tests
mnmkng Oct 14, 2020
ff19d9c
Improve error logging, fix 404 on redirects
mnmkng Oct 14, 2020
8be38ec
Add browser test for api error, test node 14 not 10
mnmkng Oct 22, 2020
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
stuck in rewriting to async-retry.
  • Loading branch information
petrpatek committed Jan 14, 2020
commit 5a1c71a0bd044777a5fb870397558e0811772489
10 changes: 8 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -45,11 +45,13 @@
"prepare": "npm run build-toc && npm run build",
"prepublishOnly": "(test $RUNNING_FROM_SCRIPT || (echo \"You must use publish.sh instead of 'npm publish' directly!\"; exit 1)) && npm test && npm run lint",
"clean": "rm -rf build && rm -rf docs",
"lint": "npm run build && eslint src test"
"lint": "npm run build && eslint src test",
"webpack": "webpack"
},
"dependencies": {
"agentkeepalive": "^4.0.2",
"apify-shared": "^0.1.45",
"async-retry": "^1.3.1",
"axios": "^0.19.0",
"content-type": "^1.0.4",
"lodash": "^4.17.15",
Expand All @@ -67,18 +69,22 @@
"@types/mocha": "^5.2.7",
"@types/nock": "^10.0.3",
"apify-jsdoc-template": "github:apifytech/apify-jsdoc-template",
"babel-loader": "^8.0.6",
"chai": "^4.1.2",
"compression": "^1.7.4",
"deploy-web-to-s3": "^1.3.0",
"eslint": "^5.16.0",
"express": "^4.17.1",
"jsdoc": "^3.4.3",
"jsdoc-export-default-interop": "^0.3.1",
"json-loader": "^0.5.7",
"markdown-toc": "^1.2.0",
"mocha": "^6.2.0",
"morgan": "^1.9.1",
"nock": "^11.3.2",
"nyc": "^14.1.1",
"sinon": "^7.4.1",
"nock": "^11.3.2"
"webpack": "^4.41.5",
"webpack-cli": "^3.3.10"
}
}
28 changes: 15 additions & 13 deletions src/datasets.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
import { retryWithExpBackoff, RetryableError } from 'apify-shared/exponential_backoff';
import _ from 'lodash';
import {
checkParamOrThrow,
Expand All @@ -10,7 +9,8 @@ import {
} from './utils'; // eslint-disable-line import/no-duplicates
import * as Utils from './utils'; // eslint-disable-line import/no-duplicates


export const RETRIES = 5;
export const BACKOFF_MILLIS = 200;
/**
* Datasets
* @memberOf ApifyClient
Expand Down Expand Up @@ -360,11 +360,13 @@ export default class Datasets {
encoding: null,
};

return retryWithExpBackoff({
func: () => getDatasetItems(() => this._call(options, endpointOptions), disableBodyParser),
expBackoffMillis: 200,
expBackoffMaxRepeats: 5,
});
return Utils.retryWithExpBackoff(
bail => getDatasetItems(() => this._call(options, endpointOptions), disableBodyParser, bail),
{
retry: RETRIES,
minTimeout: BACKOFF_MILLIS,
},
);
}

/**
Expand Down Expand Up @@ -404,25 +406,25 @@ export default class Datasets {
}
}

export function parseDatasetItemsResponse(response, disableBodyParser) {
export function parseDatasetItemsResponse(response, disableBodyParser, bail) {
const contentType = response.headers['content-type'];
const wrappedItems = wrapArray(response);
try {
if (!disableBodyParser) wrappedItems.items = Utils.parseBody(wrappedItems.items, contentType);
} catch (e) {
if (e.message.includes('Unexpected end of JSON input')) {
if (!e.message.includes('Unexpected end of JSON input')) {
// Getting invalid JSON error should be retried, because it is similar to getting 500 response code.
throw new RetryableError(e);
bail(e);
}
throw e;
}
return wrappedItems;
}

export async function getDatasetItems(call, disableBodyParser) {
export async function getDatasetItems(call, disableBodyParser, bail) {
try {
const response = await call();
return parseDatasetItemsResponse(response, disableBodyParser);
console.dir(response.body.toString());
return parseDatasetItemsResponse(response, disableBodyParser, bail);
} catch (err) {
return catchNotFoundOrThrow(err);
}
Expand Down
19 changes: 9 additions & 10 deletions src/http-client.js
Original file line number Diff line number Diff line change
@@ -1,14 +1,13 @@
import axios from 'axios';
import KeepAliveAgent from 'agentkeepalive';
import { RetryableError, retryWithExpBackoff } from 'apify-shared/exponential_backoff';
import os from 'os';
import ApifyClientError, {
INVALID_PARAMETER_ERROR_TYPE_V1,
INVALID_PARAMETER_ERROR_TYPE_V2,
REQUEST_FAILED_ERROR_TYPE_V1, REQUEST_FAILED_ERROR_TYPE_V2,
} from './apify_error';
import { checkParamOrThrow, CONTENT_TYPE_JSON, newApifyClientErrorFromResponse } from './utils';
import { version } from '../package';
import { checkParamOrThrow, CONTENT_TYPE_JSON, newApifyClientErrorFromResponse, retryWithExpBackoff } from './utils';
import { version } from '../package.json';

export const RATE_LIMIT_EXCEEDED_STATUS_CODE = 429;
export const EXP_BACKOFF_MILLIS = 500;
Expand Down Expand Up @@ -132,7 +131,7 @@ export class HttpClient {

const axiosConfig = this._getAxiosConfig(options);

const makeRequest = async () => {
const makeRequest = async (bail) => {
iteration += 1;
let statusCode;
let response;
Expand Down Expand Up @@ -168,11 +167,13 @@ export class HttpClient {
// For status codes 300-499 except options.retryOnStatusCodes we immediately rejects the promise
// since it's probably caused by invalid url (redirect 3xx) or invalid user input (4xx).
if (
statusCode >= 300
!(statusCode >= 300
&& statusCode < 500
&& !retryOnStatusCodes.includes(statusCode)
)
) {
throw newApifyClientErrorFromResponse(response.body, isApiV1, { statusCode, url: options.url, method: options.method });
bail(newApifyClientErrorFromResponse(response.body, isApiV1, { statusCode, url: options.url, method: options.method }));

}

const errorDetails = {
Expand All @@ -194,10 +195,8 @@ export class HttpClient {
? 'API request failed on the first try'
: `API request failed on retry number ${iteration - 1}`;
const REQUEST_FAILED_ERROR_TYPE = isApiV1 ? REQUEST_FAILED_ERROR_TYPE_V1 : REQUEST_FAILED_ERROR_TYPE_V2;
const originalError = new ApifyClientError(REQUEST_FAILED_ERROR_TYPE, errorMsg, errorDetails, error);
throw new RetryableError(originalError);
throw new ApifyClientError(REQUEST_FAILED_ERROR_TYPE, errorMsg, errorDetails, error);
};

return retryWithExpBackoff({ func: makeRequest, expBackoffMaxRepeats, expBackoffMillis });
return retryWithExpBackoff(makeRequest, { retries: expBackoffMaxRepeats, minTimeout: expBackoffMillis });
}
}
25 changes: 25 additions & 0 deletions src/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@ import _ from 'lodash';
import contentTypeParser from 'content-type';
import { parseType, parsedTypeCheck } from 'type-check';
import { gzip } from 'zlib';
import log from 'apify-shared/log';
import retry from 'async-retry';
import ApifyClientError, {
INVALID_PARAMETER_ERROR_TYPE_V1,
INVALID_PARAMETER_ERROR_TYPE_V2,
Expand Down Expand Up @@ -183,3 +185,26 @@ export function stringifyWebhooksToBase64(webhooks) {
export function replaceSlashWithTilde(stringWithSlash) {
return stringWithSlash.replace('/', '~');
}

// TODO: Move to the apify-shared
/**
*
* @param func
* @param opts
* @return {Promise | Promise<any>}
*/
export const retryWithExpBackoff = (func, opts) => {
let retryCount = 0;
const onRetry = (error) => {
retryCount += 1;
if (retryCount === Math.round(opts.retries / 2)) {
log.warning(`Retry failed ${retryCount} times and will be repeated later`, {
originalError: error.error.message,
errorDetails: error.error.details,
});
}
};
const options = Object.assign({}, { onRetry }, opts);

return retry(func, options);
};
2 changes: 2 additions & 0 deletions test/_helper.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import retry from 'async-retry';
import log from 'apify-shared/log';
import * as httpClient from '../build/http-client';
import { REQUEST_ENDPOINTS_EXP_BACKOFF_MAX_REPEATS } from '../build/request_queues';

Expand Down
23 changes: 11 additions & 12 deletions test/datasets.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { gzipSync } from 'zlib';
import sinon from 'sinon';
import * as exponentialBackoff from 'apify-shared/exponential_backoff';
import * as utils from '../build/utils';
import { getDatasetItems, parseDatasetItemsResponse, BASE_PATH } from '../build/datasets';
import { getDatasetItems, parseDatasetItemsResponse, RETRIES, BACKOFF_MILLIS } from '../build/datasets';
import ApifyClientError, { NOT_FOUND_STATUS_CODE } from '../src/apify_error';
import ApifyClient from '../build';
import mockServer from './mock_server/server';
Expand Down Expand Up @@ -272,19 +272,17 @@ describe('Dataset methods', () => {

it('getItems should retry with exponentialBackoff', async () => {
const datasetId = 'some-id';
let run = false;
const stub = sinon.stub(utils, 'parseBody');
const stubRetryWithExpBackoff = sinon.stub(exponentialBackoff, 'retryWithExpBackoff');
stub.callsFake(() => {
throw new exponentialBackoff.RetryableError(new Error(message));
});
stubRetryWithExpBackoff.callsFake(() => {
run = true;
const originalRetry = utils.retryWithExpBackoff;
const stub = sinon.stub(utils, 'retryWithExpBackoff');
stub.callsFake((func, opts) => {
console.log(func, opts)
expect(typeof func).to.be.eql('function');
expect(opts.retry).to.be.eql(RETRIES);
expect(opts.minTimeout).to.be.eql(BACKOFF_MILLIS);
return originalRetry(func, opts);
});
await client.datasets.getItems({ datasetId, limit: 1, offset: 1 });
expect(run).to.be.eql(true);
utils.parseBody.restore();
exponentialBackoff.retryWithExpBackoff.restore();
stub.restore();
});

it('getItems() limit and offset work', async () => {
Expand Down Expand Up @@ -358,6 +356,7 @@ describe('Dataset methods', () => {
mockServer.setResponse({ body: expected.items, headers });

const res = await client.datasets.getItems({ datasetId });
console.dir(res);
expect(res.toString()).to.be.eql(expected.toString());
validateRequest({}, { datasetId });
});
Expand Down
10 changes: 5 additions & 5 deletions test/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -118,7 +118,7 @@ describe('ApifyClient', () => {
});

xit('should be possible to use with promises', () => {
//TODO: Remove?
// TODO: Remove?
const apifyClient = new ApifyClient({
protocol: 'http',
host: 'myhost',
Expand All @@ -136,7 +136,7 @@ describe('ApifyClient', () => {
});

xit('should be possible handle errors when used with promises', () => {
//TODO: Remove?
// TODO: Remove?
const apifyClient = new ApifyClient({
protocol: 'http',
host: 'myhost',
Expand Down Expand Up @@ -179,7 +179,7 @@ describe('ApifyClient', () => {
});

xit('should be possible handle errors when used with callbacks', (done) => {
//TODO: Remove?
// TODO: Remove?
const apifyClient = new ApifyClient({
protocol: 'http',
host: 'myhost',
Expand All @@ -200,7 +200,7 @@ describe('ApifyClient', () => {
});

xit('should passed preconfigured utils.requestPromise to each method', () => {
//TODO: Remove?
// TODO: Remove?

const requestPromiseMock = sinon.mock(utils, 'requestPromise');
const expected = {
Expand Down Expand Up @@ -235,7 +235,7 @@ describe('ApifyClient', () => {
});

xit('should remove trailing forward slash from baseUrl', () => {
//TODO: Remove?
// TODO: Remove?

const apifyClient = new ApifyClient({
baseUrl: 'something/',
Expand Down
21 changes: 21 additions & 0 deletions webpack.config.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
module.exports = {
entry: './src/index.js',
module: {
rules: [
{
test: /\.(js)$/,
exclude: /node_modules/,
use: ['babel-loader'],
},
],
},
resolve: {
extensions: ['*', '.js'],
},
output: {
path: `${__dirname}/dist`,
publicPath: '/',
filename: 'bundle.js',
},
mode: 'production',
};