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

feat(users): Add users collection to SDK #52

Merged
merged 35 commits into from
Jan 23, 2019
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
3f3c501
Add docs related to adding the users namespace to the API
kwelch Dec 28, 2018
89530c7
updates to users doc based on review
kwelch Dec 31, 2018
174d8b8
more code review updates
kwelch Jan 2, 2019
3f2d2d7
update doc to match emailLookup name
kwelch Jan 2, 2019
6026411
update getById example to use a string for id
kwelch Jan 2, 2019
698b878
Add implementation for users.me
kwelch Jan 2, 2019
ab23e91
add keys to UserObject interface to fix lint issue
kwelch Jan 2, 2019
a3fd6e1
remove unnecessary typing
kwelch Jan 2, 2019
3c16eaa
update users test to use request instead of full sdk
kwelch Jan 2, 2019
5288f3f
add index test for users collection
kwelch Jan 2, 2019
a1f11d4
updated to use instance of instead of any
kwelch Jan 2, 2019
57a0e5c
update doc on each method to remove extra words
kwelch Jan 3, 2019
22b978b
Update type name from collection to methods
kwelch Jan 3, 2019
90b597a
Update type from RequestHelper to JSONRequest
kwelch Jan 3, 2019
c232d53
Add Email type and rename UserObject to User
kwelch Jan 3, 2019
24aae2b
Add transformation of snakeToCamel for objects keys
kwelch Jan 3, 2019
926c305
fixed tests for transformed data
kwelch Jan 3, 2019
ac627ec
fixed test for node@6
kwelch Jan 3, 2019
a71616a
replace Object.entries with keys for node@6 support
kwelch Jan 3, 2019
ab65a28
added rejection case to users.me
kwelch Jan 4, 2019
f1e5574
Added more test and methods to userMethods
kwelch Jan 4, 2019
fcdb247
swap functions for classes
jcreamer898 Jan 7, 2019
28437e5
clean up a bit
jcreamer898 Jan 7, 2019
e05a3b2
a bit more cleanup
jcreamer898 Jan 7, 2019
dfca786
rename interfaces while I make up my mind
jcreamer898 Jan 7, 2019
392ac6d
Merge pull request #1 from jcreamer898/jc-add-users
kwelch Jan 8, 2019
9b6dabb
minor updates to make jsonRequest generic
kwelch Jan 8, 2019
95c2b15
fix(types): update generics in request
jcreamer898 Jan 8, 2019
54dd2b3
add default api response
jcreamer898 Jan 8, 2019
2cb171e
Merge pull request #2 from jcreamer898/jc-add-users
kwelch Jan 9, 2019
817fa36
remove support for emailLookup since it is deprecated
kwelch Jan 10, 2019
7afe721
update test to deconstruct to specific methods
kwelch Jan 10, 2019
2a8c562
avoid async/await for compile purposes
kwelch Jan 10, 2019
1170cc6
fix binding issue with users api functions
kwelch Jan 10, 2019
4c18869
revert arrow functions and fix tests
kwelch Jan 10, 2019
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
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ The official JavaScript SDK for the [Eventbrite v3 API](https://www.eventbrite.c

> NOTE: This library is still in **beta** as we flesh out the API of the SDK.

## ToC
## Table of Contents

* [Installation](#installation)
* [Quick Usage](#quick-usage)
Expand Down
3 changes: 2 additions & 1 deletion docs/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,12 @@

This SDK interface closely mirors the [Eventbrite v3 REST API](https://www.eventbrite.com/developer/v3/) endpoints that it wraps. The SDK provides many conveniences for making requests and processing responses to make it easier to use in the JavaScript environment.

## ToC
## Table of Contents

- [Including the package](#including-the-package)
- [Configuring a SDK object](#configuring-a-sdk-object)
- [`request()`](./request.md)
- [Users](./users.md)

## Including the package

Expand Down
93 changes: 93 additions & 0 deletions docs/users.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,93 @@
# Users
kwelch-eb marked this conversation as resolved.
Show resolved Hide resolved

This is a collection of method that are intended to be helpful wrappers around the [Users API endpoints](user-api-docs).

View the [User response object](user-object-reference) for details on the properties you'll get back with each response.

## Table on Contents

- [`sdk.users.me()`](#me)
- [`sdk.users.get(id)`](#getById)
- [`sdk.users.emailLookup(email)`](#lookByEmail)

<a id="me"></a>

## `sdk.users.me()`
Gets details about the current logged in user.

**Read [`/users/me` documentation](user-get-me) for more details.**

### API
```js
sdk.users.me(): Promise<User>
```

### Example

```js
const eventbrite = require('eventbrite');

// Create configured Eventbrite SDK
const sdk = eventbrite({token: 'OATH_TOKEN_HERE'});

sdk.users.me().then((user) => {
console.log(`Hi ${user.name}!`);
});
```

<a id="getById"></a>

## `sdk.users.get(id)`
Gets the details for a specific user by their user id.

**Read [`/users/:id` documentation](user-get-me) for more details.**

### API
```js
sdk.users.get(id: string): Promise<User>
```

### Example

```js
const eventbrite = require('eventbrite');

// Create configured Eventbrite SDK
const sdk = eventbrite({token: 'OATH_TOKEN_HERE'});

sdk.users.get('1234567890').then((user) => {
console.log(`Hi ${user.name}!`);
});
```

<a id="lookByEmail"></a>

## `sdk.users.emailLookup(email)`
kwelch marked this conversation as resolved.
Show resolved Hide resolved
Gets the details for a specific user using their emaill address.

**_Currently, no public documentation page._**

### API
```js
sdk.users.emailLookup(string>): Promise<User>
```

### Example

```js
const eventbrite = require('eventbrite');

// Create configured Eventbrite SDK
const sdk = eventbrite({token: 'OATH_TOKEN_HERE'});

sdk.users.emailLookup('[email protected]').then((user) => {
console.log(`Hi ${user.name}!`);
});
```


<!-- link reference section -->
[user-api-docs]: https://www.eventbrite.com/platform/api#/reference/user
[user-object-reference]: https://www.eventbrite.com/platform/api#/reference/user/retrieve-a-user
[user-by-id]: https://www.eventbrite.com/platform/api#/reference/user/retrieve-a-user
[user-get-me]: https://www.eventbrite.com/platform/api#/reference/user/retrieve/retrieve-your-user
16 changes: 16 additions & 0 deletions src/__tests__/__fixtures__/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,22 @@ export const MOCK_USERS_ME_RESPONSE_DATA = {
image_id: null as string,
};

export const MOCK_TRANSFORMED_USERS_ME_RESPONSE_DATA = {
emails: [
{
email: '[email protected]',
verified: true,
primary: true,
},
],
id: '142429416488',
name: 'Eventbrite Engineer',
firstName: 'Eventbrite',
lastName: 'Engineer',
isPublic: false,
imageId: null as string,
};

export const MOCK_INTERNAL_ERROR_RESPONSE_DATA = {
status_code: 500,
error: 'INTERNAL_ERROR',
Expand Down
40 changes: 39 additions & 1 deletion src/__tests__/index.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,10 @@ import {
restoreMockFetch,
getMockResponse,
} from './utils';
import {MOCK_USERS_ME_RESPONSE_DATA} from './__fixtures__';
import {
MOCK_USERS_ME_RESPONSE_DATA,
MOCK_TRANSFORMED_USERS_ME_RESPONSE_DATA,
} from './__fixtures__';

describe('configurations', () => {
it('does not error when creating sdk object w/o configuration', () => {
Expand Down Expand Up @@ -90,4 +93,39 @@ describe('request', () => {
})
);
});

describe('users collection', () => {
it('should return an object of functions', () => {
const {users} = eventbrite({
token: MOCK_TOKEN,
baseUrl: MOCK_BASE_URL,
});

expect(users).toBeDefined();
for (const value of Object.values(users)) {
expect(value).toBeInstanceOf(Function);
}
});

it('makes request to API base url override w/ specified token', async() => {
const {users} = eventbrite({
token: MOCK_TOKEN,
baseUrl: MOCK_BASE_URL,
});

await expect(users.me()).resolves.toEqual(
MOCK_TRANSFORMED_USERS_ME_RESPONSE_DATA
);

expect(getMockFetch()).toHaveBeenCalledTimes(1);
expect(getMockFetch()).toHaveBeenCalledWith(
`${MOCK_BASE_URL}/users/me/`,
expect.objectContaining({
headers: expect.objectContaining({
Authorization: `Bearer ${MOCK_TOKEN}`,
}),
})
);
});
});
});
33 changes: 33 additions & 0 deletions src/__tests__/users.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
import {
mockFetch,
getMockFetch,
getMockResponse,
restoreMockFetch,
} from './utils';
import {
MOCK_USERS_ME_RESPONSE_DATA,
MOCK_TRANSFORMED_USERS_ME_RESPONSE_DATA,
} from './__fixtures__';

import request from '../request';
import usersMethods from '../users';

const users = usersMethods(request);
kwelch marked this conversation as resolved.
Show resolved Hide resolved

describe('users.me()', () => {
kwelch marked this conversation as resolved.
Show resolved Hide resolved
it('calls fetch and calls fetch with appropriate defaults', async() => {
mockFetch(getMockResponse(MOCK_USERS_ME_RESPONSE_DATA));

await expect(users.me()).resolves.toEqual(
MOCK_TRANSFORMED_USERS_ME_RESPONSE_DATA
);

expect(getMockFetch()).toHaveBeenCalledTimes(1);
expect(getMockFetch()).toHaveBeenCalledWith(
'/users/me/',
expect.objectContaining({})
);

restoreMockFetch();
});
});
16 changes: 11 additions & 5 deletions src/index.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import {Sdk, SdkConfig} from './types';
import {Sdk, SdkConfig, JSONRequest} from './types';
import request from './request';
import userMethods from './users';

export * from './constants';

Expand All @@ -8,8 +9,8 @@ const DEFAULT_API_URL = 'https://www.eventbriteapi.com/v3';
const eventbrite = ({
baseUrl = DEFAULT_API_URL,
token,
}: SdkConfig = {}): Sdk => ({
request: (endpoint, options = {}) => {
}: SdkConfig = {}): Sdk => {
const requestHelper: JSONRequest = (endpoint, options = {}) => {
kwelch marked this conversation as resolved.
Show resolved Hide resolved
const url = `${baseUrl}${endpoint}`;
let requestOptions = options;

Expand All @@ -24,7 +25,12 @@ const eventbrite = ({
}

return request(url, requestOptions);
},
});
};

return {
request: requestHelper,
users: userMethods(requestHelper),
};
};

export default eventbrite;
6 changes: 4 additions & 2 deletions src/request.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import {JSONResponseData, ParsedResponseError} from './types';
import {JSONRequest, JSONResponseData, ParsedResponseError} from './types';
import 'isomorphic-fetch';

/**
Expand Down Expand Up @@ -145,5 +145,7 @@ const _catchStatusError = (res: Response): Promise<any> =>
* Low-level method that makes fetch requests, returning the response formatted as JSON.
* It parses errors from API v3 and throws exceptions with those errors
*/
export default (url: string, options?: RequestInit): Promise<{}> =>
const jsonRequest: JSONRequest = (url: string, options?: RequestInit) =>
_fetchJSON(url, options).catch(_catchStatusError);

export default jsonRequest;
11 changes: 10 additions & 1 deletion src/types.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,18 @@
import {UserMethods} from './users';

export interface SdkConfig {
token?: string;
baseUrl?: string;
}

export type JSONRequest = (
apiPath: string,
options?: RequestInit
) => Promise<{}>;

export interface Sdk {
request: (apiPath: string, options?: RequestInit) => Promise<{}>;
request: JSONRequest;
users: UserMethods;
}

export interface ArgumentErrors {
Expand Down
50 changes: 50 additions & 0 deletions src/users.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
import {JSONRequest} from './types';

export type Email = {
email: string;
primary: boolean;
};

export interface User {
id: string;
firstName: string;
lastName: string;
imageId: string;
email: Email[];
}

export interface UserMethods {
me: () => Promise<User>;
}

const SNAKE_CASE_MATCH = /_\w/g;
const snakeToCamel = (str: string) =>
str.replace(SNAKE_CASE_MATCH, (chars: string) => chars[1].toUpperCase());

const transformKeysSnakeToCamel = (obj: {}): {} =>
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do you wanna use a lib like camelize? https://yarnpkg.com/en/package/camelize. Cuz now in theory we need tests for transformKeysSnakeToCamel

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will look into a package, however that one will not work because it also converts dotCase to camel, which breaks our API. That is what led us to making this function internally.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is turning out to be difficult to find something for since it is very generic, but all of them convert expand.destination_event to expandDestinationEvent, which is undesired.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What do you think about leaving this and adding tests? It is that or I make a package of it 😄

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I moved this around and added tests. Also working on adding something to camel-keys to allow us to be able to remove this.

Object.entries(obj).reduce((memo, [key, value]) => {
const camelKey = snakeToCamel(key);
let newValue = value;

if (
newValue &&
typeof newValue === 'object' &&
!Array.isArray(newValue)
) {
newValue = transformKeysSnakeToCamel(newValue);
}

return {
...memo,
[camelKey]: newValue,
};
}, {});

export default (request: JSONRequest): UserMethods => {
const me = () =>
request('/users/me/').then(transformKeysSnakeToCamel) as Promise<User>;

return {
kwelch marked this conversation as resolved.
Show resolved Hide resolved
me,
};
};
2 changes: 1 addition & 1 deletion tsconfig.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"compilerOptions": {
"lib": ["es2015", "dom"],
"lib": ["es2017", "dom"],
"noImplicitAny": true,
"noEmit": true
},
Expand Down