mirror of
https://github.com/n8n-io/n8n.git
synced 2024-12-25 20:54:07 -08:00
✨ Add Oura node (#1609)
* ✨ Add Oura node * 🔨 Make it work * ⚡ Improvements * ⚡ Improvements * 🎨 Fix SVG size and position * ⚡ Fix parameter error & other improvements Co-authored-by: Iván Ovejero <ivov.src@gmail.com> Co-authored-by: ricardo <ricardoespinoza105@gmail.com> Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
This commit is contained in:
parent
bbf875cd08
commit
45c0d6598f
18
packages/nodes-base/credentials/OuraApi.credentials.ts
Normal file
18
packages/nodes-base/credentials/OuraApi.credentials.ts
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
import {
|
||||||
|
ICredentialType,
|
||||||
|
NodePropertyTypes,
|
||||||
|
} from 'n8n-workflow';
|
||||||
|
|
||||||
|
export class OuraApi implements ICredentialType {
|
||||||
|
name = 'ouraApi';
|
||||||
|
displayName = 'Oura API';
|
||||||
|
documentationUrl = 'oura';
|
||||||
|
properties = [
|
||||||
|
{
|
||||||
|
displayName: 'Personal Access Token',
|
||||||
|
name: 'accessToken',
|
||||||
|
type: 'string' as NodePropertyTypes,
|
||||||
|
default: '',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
63
packages/nodes-base/nodes/Oura/GenericFunctions.ts
Normal file
63
packages/nodes-base/nodes/Oura/GenericFunctions.ts
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
import {
|
||||||
|
OptionsWithUri,
|
||||||
|
} from 'request';
|
||||||
|
|
||||||
|
import {
|
||||||
|
IExecuteFunctions,
|
||||||
|
IExecuteSingleFunctions,
|
||||||
|
IHookFunctions,
|
||||||
|
ILoadOptionsFunctions,
|
||||||
|
} from 'n8n-core';
|
||||||
|
|
||||||
|
import {
|
||||||
|
IDataObject,
|
||||||
|
} from 'n8n-workflow';
|
||||||
|
|
||||||
|
export async function ouraApiRequest(
|
||||||
|
this: IHookFunctions | IExecuteFunctions | IExecuteSingleFunctions | ILoadOptionsFunctions,
|
||||||
|
method: string,
|
||||||
|
resource: string,
|
||||||
|
body: IDataObject = {},
|
||||||
|
qs: IDataObject = {},
|
||||||
|
uri?: string,
|
||||||
|
option: IDataObject = {},
|
||||||
|
) {
|
||||||
|
|
||||||
|
const credentials = this.getCredentials('ouraApi');
|
||||||
|
if (credentials === undefined) {
|
||||||
|
throw new Error('No credentials got returned!');
|
||||||
|
}
|
||||||
|
let options: OptionsWithUri = {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${credentials.accessToken}`,
|
||||||
|
},
|
||||||
|
method,
|
||||||
|
qs,
|
||||||
|
body,
|
||||||
|
uri: uri || `https://api.ouraring.com/v1${resource}`,
|
||||||
|
json: true,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!Object.keys(body).length) {
|
||||||
|
delete options.body;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Object.keys(qs).length) {
|
||||||
|
delete options.qs;
|
||||||
|
}
|
||||||
|
|
||||||
|
options = Object.assign({}, options, option);
|
||||||
|
|
||||||
|
try {
|
||||||
|
return await this.helpers.request!(options);
|
||||||
|
} catch (error) {
|
||||||
|
|
||||||
|
const errorMessage = error?.response?.body?.message;
|
||||||
|
|
||||||
|
if (errorMessage) {
|
||||||
|
throw new Error(`Oura error response [${error.statusCode}]: ${errorMessage}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
178
packages/nodes-base/nodes/Oura/Oura.node.ts
Normal file
178
packages/nodes-base/nodes/Oura/Oura.node.ts
Normal file
|
@ -0,0 +1,178 @@
|
||||||
|
import {
|
||||||
|
IExecuteFunctions,
|
||||||
|
} from 'n8n-core';
|
||||||
|
|
||||||
|
import {
|
||||||
|
IDataObject,
|
||||||
|
INodeExecutionData,
|
||||||
|
INodeType,
|
||||||
|
INodeTypeDescription,
|
||||||
|
} from 'n8n-workflow';
|
||||||
|
|
||||||
|
import {
|
||||||
|
ouraApiRequest,
|
||||||
|
} from './GenericFunctions';
|
||||||
|
|
||||||
|
import {
|
||||||
|
profileOperations,
|
||||||
|
} from './ProfileDescription';
|
||||||
|
|
||||||
|
import {
|
||||||
|
summaryFields,
|
||||||
|
summaryOperations,
|
||||||
|
} from './SummaryDescription';
|
||||||
|
|
||||||
|
import * as moment from 'moment';
|
||||||
|
|
||||||
|
export class Oura implements INodeType {
|
||||||
|
description: INodeTypeDescription = {
|
||||||
|
displayName: 'Oura',
|
||||||
|
name: 'oura',
|
||||||
|
icon: 'file:oura.svg',
|
||||||
|
group: ['output'],
|
||||||
|
version: 1,
|
||||||
|
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
|
||||||
|
description: 'Consume Oura API',
|
||||||
|
defaults: {
|
||||||
|
name: 'Oura',
|
||||||
|
color: '#2f4a73',
|
||||||
|
},
|
||||||
|
inputs: ['main'],
|
||||||
|
outputs: ['main'],
|
||||||
|
credentials: [
|
||||||
|
{
|
||||||
|
name: 'ouraApi',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
properties: [
|
||||||
|
{
|
||||||
|
displayName: 'Resource',
|
||||||
|
name: 'resource',
|
||||||
|
type: 'options',
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'Profile',
|
||||||
|
value: 'profile',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Summary',
|
||||||
|
value: 'summary',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default: 'summary',
|
||||||
|
description: 'Resource to consume.',
|
||||||
|
},
|
||||||
|
...profileOperations,
|
||||||
|
...summaryOperations,
|
||||||
|
...summaryFields,
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
||||||
|
const items = this.getInputData();
|
||||||
|
const length = items.length;
|
||||||
|
|
||||||
|
let responseData;
|
||||||
|
const returnData: IDataObject[] = [];
|
||||||
|
|
||||||
|
const resource = this.getNodeParameter('resource', 0) as string;
|
||||||
|
const operation = this.getNodeParameter('operation', 0) as string;
|
||||||
|
|
||||||
|
for (let i = 0; i < length; i++) {
|
||||||
|
|
||||||
|
if (resource === 'profile') {
|
||||||
|
|
||||||
|
// *********************************************************************
|
||||||
|
// profile
|
||||||
|
// *********************************************************************
|
||||||
|
|
||||||
|
// https://cloud.ouraring.com/docs/personal-info
|
||||||
|
|
||||||
|
if (operation === 'get') {
|
||||||
|
|
||||||
|
// ----------------------------------
|
||||||
|
// profile: get
|
||||||
|
// ----------------------------------
|
||||||
|
|
||||||
|
responseData = await ouraApiRequest.call(this, 'GET', '/userinfo');
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (resource === 'summary') {
|
||||||
|
|
||||||
|
// *********************************************************************
|
||||||
|
// summary
|
||||||
|
// *********************************************************************
|
||||||
|
|
||||||
|
// https://cloud.ouraring.com/docs/daily-summaries
|
||||||
|
|
||||||
|
const qs: IDataObject = {};
|
||||||
|
|
||||||
|
const { start, end } = this.getNodeParameter('filters', i) as { start: string; end: string; };
|
||||||
|
|
||||||
|
const returnAll = this.getNodeParameter('returnAll', 0) as boolean;
|
||||||
|
|
||||||
|
if (start) {
|
||||||
|
qs.start = moment(start).format('YYYY-MM-DD');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (end) {
|
||||||
|
qs.end = moment(end).format('YYYY-MM-DD');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (operation === 'getActivity') {
|
||||||
|
|
||||||
|
// ----------------------------------
|
||||||
|
// profile: getActivity
|
||||||
|
// ----------------------------------
|
||||||
|
|
||||||
|
responseData = await ouraApiRequest.call(this, 'GET', '/activity', {}, qs);
|
||||||
|
responseData = responseData.activity;
|
||||||
|
|
||||||
|
if (returnAll === false) {
|
||||||
|
const limit = this.getNodeParameter('limit', 0) as number;
|
||||||
|
responseData = responseData.splice(0, limit);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (operation === 'getReadiness') {
|
||||||
|
|
||||||
|
// ----------------------------------
|
||||||
|
// profile: getReadiness
|
||||||
|
// ----------------------------------
|
||||||
|
|
||||||
|
responseData = await ouraApiRequest.call(this, 'GET', '/readiness', {}, qs);
|
||||||
|
responseData = responseData.readiness;
|
||||||
|
|
||||||
|
if (returnAll === false) {
|
||||||
|
const limit = this.getNodeParameter('limit', 0) as number;
|
||||||
|
responseData = responseData.splice(0, limit);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (operation === 'getSleep') {
|
||||||
|
|
||||||
|
// ----------------------------------
|
||||||
|
// profile: getSleep
|
||||||
|
// ----------------------------------
|
||||||
|
|
||||||
|
responseData = await ouraApiRequest.call(this, 'GET', '/sleep', {}, qs);
|
||||||
|
responseData = responseData.sleep;
|
||||||
|
|
||||||
|
if (returnAll === false) {
|
||||||
|
const limit = this.getNodeParameter('limit', 0) as number;
|
||||||
|
responseData = responseData.splice(0, limit);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
Array.isArray(responseData)
|
||||||
|
? returnData.push(...responseData)
|
||||||
|
: returnData.push(responseData);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return [this.helpers.returnJsonArray(returnData)];
|
||||||
|
}
|
||||||
|
}
|
27
packages/nodes-base/nodes/Oura/ProfileDescription.ts
Normal file
27
packages/nodes-base/nodes/Oura/ProfileDescription.ts
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
import {
|
||||||
|
INodeProperties,
|
||||||
|
} from 'n8n-workflow';
|
||||||
|
|
||||||
|
export const profileOperations = [
|
||||||
|
{
|
||||||
|
displayName: 'Operation',
|
||||||
|
name: 'operation',
|
||||||
|
type: 'options',
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'profile',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'Get',
|
||||||
|
value: 'get',
|
||||||
|
description: 'Get the user\'s personal information.',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default: 'get',
|
||||||
|
description: 'Operation to perform.',
|
||||||
|
},
|
||||||
|
] as INodeProperties[];
|
105
packages/nodes-base/nodes/Oura/SummaryDescription.ts
Normal file
105
packages/nodes-base/nodes/Oura/SummaryDescription.ts
Normal file
|
@ -0,0 +1,105 @@
|
||||||
|
import {
|
||||||
|
INodeProperties,
|
||||||
|
} from 'n8n-workflow';
|
||||||
|
|
||||||
|
export const summaryOperations = [
|
||||||
|
{
|
||||||
|
displayName: 'Operation',
|
||||||
|
name: 'operation',
|
||||||
|
type: 'options',
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'summary',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'Get Activity Summary',
|
||||||
|
value: 'getActivity',
|
||||||
|
description: 'Get the user\'s activity summary.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Get Readiness Summary',
|
||||||
|
value: 'getReadiness',
|
||||||
|
description: 'Get the user\'s readiness summary.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Get Sleep Periods',
|
||||||
|
value: 'getSleep',
|
||||||
|
description: 'Get the user\'s sleep summary.',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default: 'getSleep',
|
||||||
|
description: 'Operation to perform.',
|
||||||
|
},
|
||||||
|
] as INodeProperties[];
|
||||||
|
|
||||||
|
export const summaryFields = [
|
||||||
|
{
|
||||||
|
displayName: 'Return All',
|
||||||
|
name: 'returnAll',
|
||||||
|
type: 'boolean',
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'summary',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
default: false,
|
||||||
|
description: 'If all results should be returned or only up to a given limit.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Limit',
|
||||||
|
name: 'limit',
|
||||||
|
type: 'number',
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'summary',
|
||||||
|
],
|
||||||
|
returnAll: [
|
||||||
|
false,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
typeOptions: {
|
||||||
|
minValue: 1,
|
||||||
|
maxValue: 10,
|
||||||
|
},
|
||||||
|
default: 5,
|
||||||
|
description: 'How many results to return.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Filters',
|
||||||
|
name: 'filters',
|
||||||
|
type: 'collection',
|
||||||
|
placeholder: 'Add Filter',
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'summary',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
default: {},
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
displayName: 'End Date',
|
||||||
|
name: 'end',
|
||||||
|
type: 'dateTime',
|
||||||
|
default: '',
|
||||||
|
description: 'End date for the summary retrieval. If omitted, it defaults to the current day.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Start Date',
|
||||||
|
name: 'start',
|
||||||
|
type: 'dateTime',
|
||||||
|
default: '',
|
||||||
|
description: 'Start date for the summary retrieval. If omitted, it defaults to a week ago.',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
] as INodeProperties[];
|
16
packages/nodes-base/nodes/Oura/oura.svg
Normal file
16
packages/nodes-base/nodes/Oura/oura.svg
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
<?xml version="1.0" standalone="no"?>
|
||||||
|
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN"
|
||||||
|
"http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd">
|
||||||
|
<svg version="1.0" xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="12 12 35.000000 35.000000"
|
||||||
|
preserveAspectRatio="xMidYMid meet">
|
||||||
|
|
||||||
|
<g transform="translate(0.000000,60.000000) scale(0.100000,-0.100000)"
|
||||||
|
fill="#000000" stroke="none">
|
||||||
|
<path d="M240 435 c0 -12 13 -15 60 -15 47 0 60 3 60 15 0 12 -13 15 -60 15
|
||||||
|
-47 0 -60 -3 -60 -15z"/>
|
||||||
|
<path d="M260 387 c-41 -14 -80 -68 -80 -111 0 -77 79 -138 151 -117 98 29
|
||||||
|
121 154 39 212 -35 24 -69 29 -110 16z m100 -44 c49 -43 43 -121 -10 -148 -42
|
||||||
|
-22 -74 -18 -109 11 -43 36 -44 93 -2 135 38 38 80 39 121 2z"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 705 B |
|
@ -169,6 +169,7 @@
|
||||||
"dist/credentials/OAuth2Api.credentials.js",
|
"dist/credentials/OAuth2Api.credentials.js",
|
||||||
"dist/credentials/OpenWeatherMapApi.credentials.js",
|
"dist/credentials/OpenWeatherMapApi.credentials.js",
|
||||||
"dist/credentials/OrbitApi.credentials.js",
|
"dist/credentials/OrbitApi.credentials.js",
|
||||||
|
"dist/credentials/OuraApi.credentials.js",
|
||||||
"dist/credentials/PaddleApi.credentials.js",
|
"dist/credentials/PaddleApi.credentials.js",
|
||||||
"dist/credentials/PagerDutyApi.credentials.js",
|
"dist/credentials/PagerDutyApi.credentials.js",
|
||||||
"dist/credentials/PagerDutyOAuth2Api.credentials.js",
|
"dist/credentials/PagerDutyOAuth2Api.credentials.js",
|
||||||
|
@ -436,6 +437,7 @@
|
||||||
"dist/nodes/OpenThesaurus/OpenThesaurus.node.js",
|
"dist/nodes/OpenThesaurus/OpenThesaurus.node.js",
|
||||||
"dist/nodes/OpenWeatherMap.node.js",
|
"dist/nodes/OpenWeatherMap.node.js",
|
||||||
"dist/nodes/Orbit/Orbit.node.js",
|
"dist/nodes/Orbit/Orbit.node.js",
|
||||||
|
"dist/nodes/Oura/Oura.node.js",
|
||||||
"dist/nodes/Paddle/Paddle.node.js",
|
"dist/nodes/Paddle/Paddle.node.js",
|
||||||
"dist/nodes/PagerDuty/PagerDuty.node.js",
|
"dist/nodes/PagerDuty/PagerDuty.node.js",
|
||||||
"dist/nodes/PayPal/PayPal.node.js",
|
"dist/nodes/PayPal/PayPal.node.js",
|
||||||
|
|
Loading…
Reference in a new issue