Merge pull request #1076 from wrightia/v4-connection-google-sheets

Connection Google Sheets V4
This commit is contained in:
Sam 2022-02-01 12:52:47 +02:00 committed by GitHub
commit 01b6fa769a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
53 changed files with 715 additions and 660 deletions

1
.pnp.cjs generated
View File

@ -5207,6 +5207,7 @@ function $$SETUP_STATE(hydrateRuntimeState, basePath) {
["@lowdefy/blocks-markdown", "workspace:packages/plugins/blocks/blocks-markdown"],
["@lowdefy/connection-axios-http", "workspace:packages/plugins/connections/connection-axios-http"],
["@lowdefy/connection-elasticsearch", "workspace:packages/plugins/connections/connection-elasticsearch"],
["@lowdefy/connection-google-sheets", "workspace:packages/plugins/connections/connection-google-sheets"],
["@lowdefy/connection-knex", "workspace:packages/plugins/connections/connection-knex"],
["@lowdefy/connection-mongodb", "workspace:packages/plugins/connections/connection-mongodb"],
["@lowdefy/connection-redis", "workspace:packages/plugins/connections/connection-redis"],

View File

@ -64,6 +64,7 @@
"@lowdefy/blocks-markdown": "4.0.0-alpha.6",
"@lowdefy/connection-axios-http": "4.0.0-alpha.6",
"@lowdefy/connection-elasticsearch": "4.0.0-alpha.6",
"@lowdefy/connection-google-sheets": "4.0.0-alpha.6",
"@lowdefy/connection-knex": "4.0.0-alpha.6",
"@lowdefy/connection-mongodb": "4.0.0-alpha.6",
"@lowdefy/connection-redis": "4.0.0-alpha.6",

View File

@ -28,6 +28,7 @@ const defaultPackages = [
'@lowdefy/blocks-markdown',
'@lowdefy/connection-axios-http',
'@lowdefy/connection-elasticsearch',
'@lowdefy/connection-google-sheets',
'@lowdefy/connection-knex',
'@lowdefy/connection-mongodb',
'@lowdefy/connection-redis',

View File

@ -27,8 +27,8 @@
},
"type": "module",
"exports": {
".": "./dist/index.js",
"./connections/*": "./dist/connections/*"
"./connections": "./dist/connections.js",
"./types": "./dist/types.js"
},
"files": [
"dist/*"

View File

@ -14,13 +14,4 @@
limitations under the License.
*/
export default {
import: {
path: 'connections/GoogleSheet/GoogleSheetGetOne/GoogleSheetGetOne.js',
schema: 'connections/GoogleSheet/GoogleSheetGetOne/GoogleSheetGetOneSchema.json',
},
meta: {
checkRead: true,
checkWrite: false,
},
};
export { default as GoogleSheet } from './connections/GoogleSheet/GoogleSheet.js';

View File

@ -21,11 +21,10 @@ import GoogleSheetGetMany from './GoogleSheetGetMany/GoogleSheetGetMany.js';
import GoogleSheetGetOne from './GoogleSheetGetOne/GoogleSheetGetOne.js';
import GoogleSheetUpdateOne from './GoogleSheetUpdateOne/GoogleSheetUpdateOne.js';
import GoogleSheetUpdateMany from './GoogleSheetUpdateMany/GoogleSheetUpdateMany.js';
import schema from './schema.js';
export default {
import: {
schema: 'connections/GoogleSheet/GoogleSheetSchema.json',
},
schema,
requests: {
GoogleSheetAppendMany,
GoogleSheetAppendOne,

View File

@ -16,7 +16,8 @@
import { validate } from '@lowdefy/ajv';
import GoogleSheet from './GoogleSheet.js';
import schema from './GoogleSheetSchema.json';
const schema = GoogleSheet.schema;
test('All requests are present', () => {
expect(GoogleSheet.requests.GoogleSheetAppendOne).toBeDefined();

View File

@ -16,8 +16,9 @@
import getSheet from '../getSheet.js';
import { transformWrite } from '../transformTypes.js';
import schema from './schema.js';
async function googleSheetAppendMany({ request, connection }) {
async function GoogleSheetAppendMany({ request, connection }) {
const { rows, options = {} } = request;
const { raw } = options;
const sheet = await getSheet({ connection });
@ -27,4 +28,10 @@ async function googleSheetAppendMany({ request, connection }) {
};
}
export default googleSheetAppendMany;
GoogleSheetAppendMany.schema = schema;
GoogleSheetAppendMany.meta = {
checkRead: false,
checkWrite: true,
};
export default GoogleSheetAppendMany;

View File

@ -15,11 +15,11 @@
*/
import { validate } from '@lowdefy/ajv';
import googleSheetAppendMany from './GoogleSheetAppendMany.js';
import requestIndex from './index.js';
import schema from './GoogleSheetAppendManySchema.json';
import GoogleSheetAppendMany from './GoogleSheetAppendMany.js';
const { checkRead, checkWrite } = GoogleSheetAppendMany.meta;
const schema = GoogleSheetAppendMany.schema;
const { checkRead, checkWrite } = requestIndex.meta;
const mockAddRows = jest.fn();
jest.mock('../getSheet', () => () => ({
addRows: mockAddRows,
@ -29,7 +29,7 @@ const mockAddRowsDefaultImp = (rows) => rows.map((row) => ({ ...row, _sheet: {}
test('googleSheetAppendMany, one row', async () => {
mockAddRows.mockImplementation(mockAddRowsDefaultImp);
const res = await googleSheetAppendMany({
const res = await GoogleSheetAppendMany({
request: {
rows: [{ id: '1', name: 'John', age: '34', birth_date: '2020/04/26', married: 'TRUE' }],
},
@ -58,7 +58,7 @@ test('googleSheetAppendMany, one row', async () => {
test('googleSheetAppendMany, two rows', async () => {
mockAddRows.mockImplementation(mockAddRowsDefaultImp);
const res = await googleSheetAppendMany({
const res = await GoogleSheetAppendMany({
request: {
rows: [
{ id: '1', name: 'John', age: '34', birth_date: '2020/04/26', married: 'TRUE' },
@ -97,7 +97,7 @@ test('googleSheetAppendMany, two rows', async () => {
test('googleSheetAppendMany, rows empty array', async () => {
mockAddRows.mockImplementation(mockAddRowsDefaultImp);
const res = await googleSheetAppendMany({
const res = await GoogleSheetAppendMany({
request: {
rows: [],
},
@ -118,7 +118,7 @@ test('googleSheetAppendMany, rows empty array', async () => {
test('googleSheetAppendMany, transform types', async () => {
mockAddRows.mockImplementation(mockAddRowsDefaultImp);
const res = await googleSheetAppendMany({
const res = await GoogleSheetAppendMany({
request: {
rows: [
{
@ -162,7 +162,7 @@ test('googleSheetAppendMany, transform types', async () => {
test('googleSheetAppendMany, one row, raw true', async () => {
mockAddRows.mockImplementation(mockAddRowsDefaultImp);
const res = await googleSheetAppendMany({
const res = await GoogleSheetAppendMany({
request: {
rows: [{ id: '1', name: 'John', age: '34', birth_date: '2020/04/26', married: 'TRUE' }],
options: {

View File

@ -1,40 +0,0 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "Lowdefy Request Schema - GoogleSheetAppendMany",
"type": "object",
"required": ["rows"],
"properties": {
"rows": {
"type": "array",
"description": "The rows to insert into the sheet. An an array of objects where keys are the column names and values are the values to insert.",
"errorMessage": {
"type": "GoogleSheetAppendMany request property \"rows\" should be an array."
},
"items": {
"type": "object",
"description": "The row to insert into the sheet. An object where keys are the column names and values are the values to insert.",
"errorMessage": {
"type": "GoogleSheetAppendMany request property \"rows\" should be an array of objects."
}
}
},
"options": {
"type": "object",
"properties": {
"raw": {
"type": "boolean",
"description": "Store raw values instead of converting as if typed into the sheets UI.",
"errorMessage": {
"type": "GoogleSheetAppendMany request property \"options.raw\" should be a boolean."
}
}
}
}
},
"errorMessage": {
"type": "GoogleSheetAppendMany request properties should be an object.",
"required": {
"rows": "GoogleSheetAppendMany request should have required property \"rows\"."
}
}
}

View File

@ -1,26 +0,0 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
import: {
path: 'connections/GoogleSheet/GoogleSheetAppendMany/GoogleSheetAppendMany.js',
schema: 'connections/GoogleSheet/GoogleSheetAppendMany/GoogleSheetAppendManySchema.json',
},
meta: {
checkRead: false,
checkWrite: true,
},
};

View File

@ -0,0 +1,58 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
$schema: 'http://json-schema.org/draft-07/schema#',
title: 'Lowdefy Request Schema - GoogleSheetAppendMany',
type: 'object',
required: ['rows'],
properties: {
rows: {
type: 'array',
description:
'The rows to insert into the sheet. An an array of objects where keys are the column names and values are the values to insert.',
errorMessage: {
type: 'GoogleSheetAppendMany request property "rows" should be an array.',
},
items: {
type: 'object',
description:
'The row to insert into the sheet. An object where keys are the column names and values are the values to insert.',
errorMessage: {
type: 'GoogleSheetAppendMany request property "rows" should be an array of objects.',
},
},
},
options: {
type: 'object',
properties: {
raw: {
type: 'boolean',
description: 'Store raw values instead of converting as if typed into the sheets UI.',
errorMessage: {
type: 'GoogleSheetAppendMany request property "options.raw" should be a boolean.',
},
},
},
},
},
errorMessage: {
type: 'GoogleSheetAppendMany request properties should be an object.',
required: {
rows: 'GoogleSheetAppendMany request should have required property "rows".',
},
},
};

View File

@ -17,8 +17,9 @@
import getSheet from '../getSheet.js';
import cleanRows from '../cleanRows.js';
import { transformWrite } from '../transformTypes.js';
import schema from './schema.js';
async function googleSheetAppendOne({ request, connection }) {
async function GoogleSheetAppendOne({ request, connection }) {
const { row, options = {} } = request;
const { raw } = options;
const sheet = await getSheet({ connection });
@ -32,4 +33,10 @@ async function googleSheetAppendOne({ request, connection }) {
};
}
export default googleSheetAppendOne;
GoogleSheetAppendOne.schema = schema;
GoogleSheetAppendOne.meta = {
checkRead: false,
checkWrite: true,
};
export default GoogleSheetAppendOne;

View File

@ -15,11 +15,10 @@
*/
import { validate } from '@lowdefy/ajv';
import googleSheetAppendOne from './GoogleSheetAppendOne.js';
import requestIndex from './index.js';
import schema from './GoogleSheetAppendOneSchema.json';
import GoogleSheetAppendOne from './GoogleSheetAppendOne.js';
const { checkRead, checkWrite } = requestIndex.meta;
const { checkRead, checkWrite } = GoogleSheetAppendOne.meta;
const schema = GoogleSheetAppendOne.schema;
const mockAddRow = jest.fn();
jest.mock('../getSheet', () => () => ({
@ -30,7 +29,7 @@ const mockAddRowDefaultImp = (row) => ({ ...row, _sheet: {} });
test('googleSheetAppendOne', async () => {
mockAddRow.mockImplementation(mockAddRowDefaultImp);
const res = await googleSheetAppendOne({
const res = await GoogleSheetAppendOne({
request: {
row: { id: '1', name: 'John', age: '34', birth_date: '2020/04/26', married: 'TRUE' },
},
@ -66,7 +65,7 @@ test('googleSheetAppendOne', async () => {
test('googleSheetAppendOne, transform types', async () => {
mockAddRow.mockImplementation(mockAddRowDefaultImp);
const res = await googleSheetAppendOne({
const res = await GoogleSheetAppendOne({
request: {
row: {
id: '1',
@ -115,7 +114,7 @@ test('googleSheetAppendOne, transform types', async () => {
test('googleSheetAppendOne, raw true', async () => {
mockAddRow.mockImplementation(mockAddRowDefaultImp);
const res = await googleSheetAppendOne({
const res = await GoogleSheetAppendOne({
request: {
row: { id: '1', name: 'John', age: '34', birth_date: '2020/04/26', married: 'TRUE' },
options: {

View File

@ -1,33 +0,0 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "Lowdefy Request Schema - GoogleSheetAppendOne",
"type": "object",
"required": ["row"],
"properties": {
"row": {
"type": "object",
"description": "The row to insert into the sheet. An object where keys are the column names and values are the values to insert.",
"errorMessage": {
"type": "GoogleSheetAppendOne request property \"row\" should be an object."
}
},
"options": {
"type": "object",
"properties": {
"raw": {
"type": "boolean",
"description": "Store raw values instead of converting as if typed into the sheets UI.",
"errorMessage": {
"type": "GoogleSheetAppendOne request property \"options.raw\" should be a boolean."
}
}
}
}
},
"errorMessage": {
"type": "GoogleSheetAppendOne request properties should be an object.",
"required": {
"row": "GoogleSheetAppendOne request should have required property \"row\"."
}
}
}

View File

@ -1,26 +0,0 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
import: {
path: 'connections/GoogleSheet/GoogleSheetAppendOneSchema/GoogleSheetAppendOne.js',
schema: 'connections/GoogleSheet/GoogleSheetAppendOneSchema/GoogleSheetAppendOneSchema.json',
},
meta: {
checkRead: false,
checkWrite: true,
},
};

View File

@ -0,0 +1,50 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
$schema: 'http://json-schema.org/draft-07/schema#',
title: 'Lowdefy Request Schema - GoogleSheetAppendOne',
type: 'object',
required: ['row'],
properties: {
row: {
type: 'object',
description:
'The row to insert into the sheet. An object where keys are the column names and values are the values to insert.',
errorMessage: {
type: 'GoogleSheetAppendOne request property "row" should be an object.',
},
},
options: {
type: 'object',
properties: {
raw: {
type: 'boolean',
description: 'Store raw values instead of converting as if typed into the sheets UI.',
errorMessage: {
type: 'GoogleSheetAppendOne request property "options.raw" should be a boolean.',
},
},
},
},
},
errorMessage: {
type: 'GoogleSheetAppendOne request properties should be an object.',
required: {
row: 'GoogleSheetAppendOne request should have required property "row".',
},
},
};

View File

@ -18,8 +18,9 @@ import cleanRows from '../cleanRows.js';
import getSheet from '../getSheet.js';
import { transformRead } from '../transformTypes.js';
import mingoFilter from '../mingoFilter.js';
import schema from './schema.js';
async function googleSheetDeleteOne({ request, connection }) {
async function GoogleSheetDeleteOne({ request, connection }) {
const { filter, options = {} } = request;
const { limit, skip } = options;
const sheet = await getSheet({ connection });
@ -39,4 +40,10 @@ async function googleSheetDeleteOne({ request, connection }) {
};
}
export default googleSheetDeleteOne;
GoogleSheetDeleteOne.schema = schema;
GoogleSheetDeleteOne.meta = {
checkRead: false,
checkWrite: true,
};
export default GoogleSheetDeleteOne;

View File

@ -15,11 +15,10 @@
*/
import { validate } from '@lowdefy/ajv';
import googleSheetDeleteOne from './GoogleSheetDeleteOne.js';
import requestIndex from './index.js';
import schema from './GoogleSheetDeleteOneSchema.json';
import GoogleSheetDeleteOne from './GoogleSheetDeleteOne.js';
const { checkRead, checkWrite } = requestIndex.meta;
const { checkRead, checkWrite } = GoogleSheetDeleteOne.meta;
const schema = GoogleSheetDeleteOne.schema;
const mockGetRows = jest.fn();
const mockDelete = jest.fn();
@ -79,7 +78,7 @@ const mockGetRowsDefaultImp = ({ limit, offset }) => {
test('googleSheetDeleteMany, match one', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetDeleteOne({
const res = await GoogleSheetDeleteOne({
request: {
filter: { id: '1' },
},
@ -103,7 +102,7 @@ test('googleSheetDeleteMany, match one', async () => {
test('googleSheetDeleteMany, match nothing', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetDeleteOne({
const res = await GoogleSheetDeleteOne({
request: {
filter: { id: 'does_not_exist' },
},
@ -117,7 +116,7 @@ test('googleSheetDeleteMany, match nothing', async () => {
test('googleSheetDeleteMany, match more than one', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetDeleteOne({
const res = await GoogleSheetDeleteOne({
request: {
filter: { _rowNumber: { $gt: 3 } },
},

View File

@ -1,40 +0,0 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "Lowdefy Request Schema - GoogleSheetDeleteOne",
"type": "object",
"required": ["filter"],
"properties": {
"filter": {
"type": "object",
"description": "A MongoDB query expression to filter the data. The first row matched by the filter will be deleted.",
"errorMessage": {
"type": "GoogleSheetDeleteOne request property \"filter\" should be an object."
}
},
"options": {
"type": "object",
"properties": {
"limit": {
"type": "number",
"description": "The maximum number of rows to fetch.",
"errorMessage": {
"type": "GoogleSheetDeleteOne request property \"options.limit\" should be a number."
}
},
"skip": {
"type": "number",
"description": "The number of rows to skip from the top of the sheet.",
"errorMessage": {
"type": "GoogleSheetDeleteOne request property \"options.skip\" should be a number."
}
}
}
}
},
"errorMessage": {
"type": "GoogleSheetDeleteOne request properties should be an object.",
"required": {
"filter": "GoogleSheetDeleteOne request should have required property \"filter\"."
}
}
}

View File

@ -1,26 +0,0 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
import: {
path: 'connections/GoogleSheet/GoogleSheetDeleteOneSchema/GoogleSheetDeleteOne.js',
schema: 'connections/GoogleSheet/GoogleSheetDeleteOneSchema/GoogleSheetDeleteOneSchema.json',
},
meta: {
checkRead: false,
checkWrite: true,
},
};

View File

@ -0,0 +1,57 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
$schema: 'http://json-schema.org/draft-07/schema#',
title: 'Lowdefy Request Schema - GoogleSheetDeleteOne',
type: 'object',
required: ['filter'],
properties: {
filter: {
type: 'object',
description:
'A MongoDB query expression to filter the data. The first row matched by the filter will be deleted.',
errorMessage: {
type: 'GoogleSheetDeleteOne request property "filter" should be an object.',
},
},
options: {
type: 'object',
properties: {
limit: {
type: 'number',
description: 'The maximum number of rows to fetch.',
errorMessage: {
type: 'GoogleSheetDeleteOne request property "options.limit" should be a number.',
},
},
skip: {
type: 'number',
description: 'The number of rows to skip from the top of the sheet.',
errorMessage: {
type: 'GoogleSheetDeleteOne request property "options.skip" should be a number.',
},
},
},
},
},
errorMessage: {
type: 'GoogleSheetDeleteOne request properties should be an object.',
required: {
filter: 'GoogleSheetDeleteOne request should have required property "filter".',
},
},
};

View File

@ -19,8 +19,9 @@ import getSheet from '../getSheet.js';
import { transformRead } from '../transformTypes.js';
import mingoAggregation from '../mingoAggregation.js';
import mingoFilter from '../mingoFilter.js';
import schema from './schema.js';
async function googleSheetGetMany({ request, connection }) {
async function GoogleSheetGetMany({ request, connection }) {
const { filter, pipeline, options = {} } = request;
const { limit, skip } = options;
const sheet = await getSheet({ connection });
@ -36,4 +37,10 @@ async function googleSheetGetMany({ request, connection }) {
return rows;
}
export default googleSheetGetMany;
GoogleSheetGetMany.schema = schema;
GoogleSheetGetMany.meta = {
checkRead: true,
checkWrite: false,
};
export default GoogleSheetGetMany;

View File

@ -15,11 +15,10 @@
*/
import { validate } from '@lowdefy/ajv';
import googleSheetGetMany from './GoogleSheetGetMany.js';
import requestIndex from './index.js';
import schema from './GoogleSheetGetManySchema.json';
import GoogleSheetGetMany from './GoogleSheetGetMany.js';
const { checkRead, checkWrite } = requestIndex.meta;
const { checkRead, checkWrite } = GoogleSheetGetMany.meta;
const schema = GoogleSheetGetMany.schema;
const mockGetRows = jest.fn();
jest.mock('../getSheet', () => () => ({
@ -74,7 +73,7 @@ const mockGetRowsDefaultImp = ({ limit, offset }) => {
test('googleSheetGetMany, all rows', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({ request: {}, connection: {} });
const res = await GoogleSheetGetMany({ request: {}, connection: {} });
expect(res).toEqual([
{
_rowNumber: 2,
@ -117,13 +116,13 @@ test('googleSheetGetMany, all rows', async () => {
test('googleSheetGetMany, empty rows returned', async () => {
mockGetRows.mockImplementation(() => []);
const res = await googleSheetGetMany({ request: {}, connection: {} });
const res = await GoogleSheetGetMany({ request: {}, connection: {} });
expect(res).toEqual([]);
});
test('googleSheetGetMany, limit', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({ request: { options: { limit: 2 } }, connection: {} });
const res = await GoogleSheetGetMany({ request: { options: { limit: 2 } }, connection: {} });
expect(res).toEqual([
{
_rowNumber: 2,
@ -148,7 +147,7 @@ test('googleSheetGetMany, limit', async () => {
test('googleSheetGetMany, skip', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({ request: { options: { skip: 2 } }, connection: {} });
const res = await GoogleSheetGetMany({ request: { options: { skip: 2 } }, connection: {} });
expect(res).toEqual([
{
_rowNumber: 4,
@ -173,7 +172,7 @@ test('googleSheetGetMany, skip', async () => {
test('googleSheetGetMany, skip and limit', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({
const res = await GoogleSheetGetMany({
request: { options: { skip: 2, limit: 1 } },
connection: {},
});
@ -192,7 +191,7 @@ test('googleSheetGetMany, skip and limit', async () => {
test('googleSheetGetMany, filter', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({ request: { filter: { name: 'Tim' } }, connection: {} });
const res = await GoogleSheetGetMany({ request: { filter: { name: 'Tim' } }, connection: {} });
expect(res).toEqual([
{
_rowNumber: 4,
@ -208,13 +207,13 @@ test('googleSheetGetMany, filter', async () => {
test('googleSheetGetMany, filter filters all', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({ request: { filter: { name: 'Nobody' } }, connection: {} });
const res = await GoogleSheetGetMany({ request: { filter: { name: 'Nobody' } }, connection: {} });
expect(res).toEqual([]);
});
test('googleSheetGetMany, filter _rowNumber', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({
const res = await GoogleSheetGetMany({
request: { filter: { _rowNumber: { $gt: 3 } } },
connection: {},
});
@ -242,7 +241,7 @@ test('googleSheetGetMany, filter _rowNumber', async () => {
test('googleSheetGetMany, pipeline count', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({
const res = await GoogleSheetGetMany({
request: { pipeline: [{ $group: { _id: 0, count: { $sum: 1 } } }] },
connection: {},
});
@ -256,7 +255,7 @@ test('googleSheetGetMany, pipeline count', async () => {
test('googleSheetGetMany, columnTypes', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetMany({
const res = await GoogleSheetGetMany({
request: {},
connection: {
columnTypes: {

View File

@ -1,43 +0,0 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "Lowdefy Request Schema - GoogleSheetGetMany",
"type": "object",
"properties": {
"filter": {
"type": "object",
"description": "A MongoDB query expression to filter the data.",
"errorMessage": {
"type": "GoogleSheetGetMany request property \"filter\" should be an object."
}
},
"pipeline": {
"type": "array",
"description": "A MongoDB aggregation pipeline to transform the data.",
"errorMessage": {
"type": "GoogleSheetGetMany request property \"pipeline\" should be an array."
}
},
"options": {
"type": "object",
"properties": {
"limit": {
"type": "number",
"description": "The maximum number of rows to fetch.",
"errorMessage": {
"type": "GoogleSheetGetMany request property \"options.limit\" should be a number."
}
},
"skip": {
"type": "number",
"description": "The number of rows to skip from the top of the sheet.",
"errorMessage": {
"type": "GoogleSheetGetMany request property \"options.skip\" should be a number."
}
}
}
}
},
"errorMessage": {
"type": "GoogleSheetGetMany request properties should be an object."
}
}

View File

@ -0,0 +1,59 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
$schema: 'http://json-schema.org/draft-07/schema#',
title: 'Lowdefy Request Schema - GoogleSheetGetMany',
type: 'object',
properties: {
filter: {
type: 'object',
description: 'A MongoDB query expression to filter the data.',
errorMessage: {
type: 'GoogleSheetGetMany request property "filter" should be an object.',
},
},
pipeline: {
type: 'array',
description: 'A MongoDB aggregation pipeline to transform the data.',
errorMessage: {
type: 'GoogleSheetGetMany request property "pipeline" should be an array.',
},
},
options: {
type: 'object',
properties: {
limit: {
type: 'number',
description: 'The maximum number of rows to fetch.',
errorMessage: {
type: 'GoogleSheetGetMany request property "options.limit" should be a number.',
},
},
skip: {
type: 'number',
description: 'The number of rows to skip from the top of the sheet.',
errorMessage: {
type: 'GoogleSheetGetMany request property "options.skip" should be a number.',
},
},
},
},
},
errorMessage: {
type: 'GoogleSheetGetMany request properties should be an object.',
},
};

View File

@ -18,8 +18,9 @@ import cleanRows from '../cleanRows.js';
import getSheet from '../getSheet.js';
import { transformRead } from '../transformTypes.js';
import mingoFilter from '../mingoFilter.js';
import schema from './schema.js';
async function googleSheetGetOne({ request, connection }) {
async function GoogleSheetGetOne({ request, connection }) {
const { filter, options = {} } = request;
const { limit, skip } = options;
const sheet = await getSheet({ connection });
@ -32,4 +33,10 @@ async function googleSheetGetOne({ request, connection }) {
return rows[0] || null;
}
export default googleSheetGetOne;
GoogleSheetGetOne.schema = schema;
GoogleSheetGetOne.meta = {
checkRead: true,
checkWrite: false,
};
export default GoogleSheetGetOne;

View File

@ -15,11 +15,10 @@
*/
import { validate } from '@lowdefy/ajv';
import googleSheetGetOne from './GoogleSheetGetOne';
import requestIndex from './index.js';
import schema from './GoogleSheetGetOneSchema.json';
import GoogleSheetGetOne from './GoogleSheetGetOne.js';
const { checkRead, checkWrite } = requestIndex.meta;
const { checkRead, checkWrite } = GoogleSheetGetOne.meta;
const schema = GoogleSheetGetOne.schema;
const mockGetRows = jest.fn();
jest.mock('../getSheet', () => () => ({
@ -74,7 +73,7 @@ const mockGetRowsDefaultImp = ({ limit, offset }) => {
test('googleSheetGetOne, first row is returned', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({ request: {}, connection: {} });
const res = await GoogleSheetGetOne({ request: {}, connection: {} });
expect(res).toEqual({
_rowNumber: 2,
_rawData: ['1', 'John', '34', '2020/04/26', 'TRUE'],
@ -88,13 +87,13 @@ test('googleSheetGetOne, first row is returned', async () => {
test('googleSheetGetOne, empty rows returned', async () => {
mockGetRows.mockImplementation(() => []);
const res = await googleSheetGetOne({ request: {}, connection: {} });
const res = await GoogleSheetGetOne({ request: {}, connection: {} });
expect(res).toEqual(null);
});
test('googleSheetGetOne, limit', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({ request: { options: { limit: 2 } }, connection: {} });
const res = await GoogleSheetGetOne({ request: { options: { limit: 2 } }, connection: {} });
expect(res).toEqual({
_rowNumber: 2,
_rawData: ['1', 'John', '34', '2020/04/26', 'TRUE'],
@ -108,7 +107,7 @@ test('googleSheetGetOne, limit', async () => {
test('googleSheetGetOne, skip', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({ request: { options: { skip: 2 } }, connection: {} });
const res = await GoogleSheetGetOne({ request: { options: { skip: 2 } }, connection: {} });
expect(res).toEqual({
_rowNumber: 4,
_rawData: ['3', 'Tim', '34', '2020/04/28', 'FALSE'],
@ -122,7 +121,7 @@ test('googleSheetGetOne, skip', async () => {
test('googleSheetGetOne, skip and limit', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({
const res = await GoogleSheetGetOne({
request: { options: { skip: 2, limit: 1 } },
connection: {},
});
@ -139,7 +138,7 @@ test('googleSheetGetOne, skip and limit', async () => {
test('googleSheetGetOne, filter', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({ request: { filter: { name: 'Tim' } }, connection: {} });
const res = await GoogleSheetGetOne({ request: { filter: { name: 'Tim' } }, connection: {} });
expect(res).toEqual({
_rowNumber: 4,
_rawData: ['3', 'Tim', '34', '2020/04/28', 'FALSE'],
@ -153,7 +152,7 @@ test('googleSheetGetOne, filter', async () => {
test('googleSheetGetOne, limit before filter', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({
const res = await GoogleSheetGetOne({
request: { filter: { name: 'Tim' }, options: { limit: 2 } },
connection: {},
});
@ -162,7 +161,7 @@ test('googleSheetGetOne, limit before filter', async () => {
test('googleSheetGetOne, skip before filter', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({
const res = await GoogleSheetGetOne({
request: { filter: { married: 'TRUE' }, options: { skip: 2 } },
connection: {},
});
@ -179,13 +178,13 @@ test('googleSheetGetOne, skip before filter', async () => {
test('googleSheetGetOne, filter filters all', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({ request: { filter: { name: 'Nobody' } }, connection: {} });
const res = await GoogleSheetGetOne({ request: { filter: { name: 'Nobody' } }, connection: {} });
expect(res).toEqual(null);
});
test('googleSheetGetOne, filter _rowNumber', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({
const res = await GoogleSheetGetOne({
request: { filter: { _rowNumber: { $gt: 3 } } },
connection: {},
});
@ -202,7 +201,7 @@ test('googleSheetGetOne, filter _rowNumber', async () => {
test('googleSheetGetOne, columnTypes', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetGetOne({
const res = await GoogleSheetGetOne({
request: {},
connection: {
columnTypes: {

View File

@ -1,36 +0,0 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "Lowdefy Request Schema - GoogleSheetGetOne",
"type": "object",
"properties": {
"filter": {
"type": "object",
"description": "A MongoDB query expression to filter the data.",
"errorMessage": {
"type": "GoogleSheetGetOne request property \"filter\" should be an object."
}
},
"options": {
"type": "object",
"properties": {
"limit": {
"type": "number",
"description": "The maximum number of rows to fetch.",
"errorMessage": {
"type": "GoogleSheetGetOne request property \"options.limit\" should be a number."
}
},
"skip": {
"type": "number",
"description": "The number of rows to skip from the top of the sheet.",
"errorMessage": {
"type": "GoogleSheetGetOne request property \"options.skip\" should be a number."
}
}
}
}
},
"errorMessage": {
"type": "GoogleSheetGetOne request properties should be an object."
}
}

View File

@ -0,0 +1,52 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
$schema: 'http://json-schema.org/draft-07/schema#',
title: 'Lowdefy Request Schema - GoogleSheetGetOne',
type: 'object',
properties: {
filter: {
type: 'object',
description: 'A MongoDB query expression to filter the data.',
errorMessage: {
type: 'GoogleSheetGetOne request property "filter" should be an object.',
},
},
options: {
type: 'object',
properties: {
limit: {
type: 'number',
description: 'The maximum number of rows to fetch.',
errorMessage: {
type: 'GoogleSheetGetOne request property "options.limit" should be a number.',
},
},
skip: {
type: 'number',
description: 'The number of rows to skip from the top of the sheet.',
errorMessage: {
type: 'GoogleSheetGetOne request property "options.skip" should be a number.',
},
},
},
},
},
errorMessage: {
type: 'GoogleSheetGetOne request properties should be an object.',
},
};

View File

@ -1,88 +0,0 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "Lowdefy Connection Schema - GoogleSheet",
"type": "object",
"required": ["spreadsheetId"],
"properties": {
"apiKey": {
"type": "string",
"description": "API key for your google project.",
"errorMessage": {
"type": "GoogleSheet connection property \"apiKey\" should be a string."
}
},
"client_email": {
"type": "string",
"description": "The email of your service account.",
"errorMessage": {
"type": "GoogleSheet connection property \"client_email\" should be a string."
}
},
"private_key": {
"type": "string",
"description": "The private key for your service account.",
"errorMessage": {
"type": "GoogleSheet connection property \"private_key\" should be a string."
}
},
"sheetId": {
"type": "string",
"description": "The ID of the worksheet. Can be found in the URL as the \"gid\" parameter. One of \"sheetId\" or \"sheetIndex\" is required.",
"errorMessage": {
"type": "GoogleSheet connection property \"sheetId\" should be a string."
}
},
"sheetIndex": {
"type": "number",
"description": "The position of the worksheet as they appear in the Google sheets UI. Starts from 0. One of \"sheetId\" or \"sheetIndex\" is required.",
"errorMessage": {
"type": "GoogleSheet connection property \"sheetIndex\" should be a number."
}
},
"spreadsheetId": {
"type": "string",
"description": "document ID from the URL of the spreadsheet.",
"errorMessage": {
"type": "GoogleSheet connection property \"spreadsheetId\" should be a string."
}
},
"columnTypes": {
"type": "object",
"description": "Define types for columns in the spreadsheet.",
"patternProperties": {
"^.*$": {
"type": "string",
"enum": ["string", "number", "boolean", "date", "json"],
"errorMessage": {
"enum": "GoogleSheet connection property \"{{ instancePath }}\" should be one of \"string\", \"number\", \"boolean\", \"date\", or \"json\"."
}
}
},
"errorMessage": {
"type": "GoogleSheet connection property \"columnTypes\" should be an object."
}
},
"read": {
"type": "boolean",
"default": true,
"description": "Allow reads from the spreadsheet.",
"errorMessage": {
"type": "GoogleSheet connection property \"read\" should be a boolean."
}
},
"write": {
"type": "boolean",
"default": false,
"description": "Allow writes to the spreadsheet.",
"errorMessage": {
"type": "GoogleSheet connection property \"write\" should be a boolean."
}
}
},
"errorMessage": {
"type": "GoogleSheet connection properties should be an object.",
"required": {
"spreadsheetId": "GoogleSheet connection should have required property \"spreadsheetId\"."
}
}
}

View File

@ -17,8 +17,9 @@
import getSheet from '../getSheet.js';
import { transformRead, transformWrite } from '../transformTypes.js';
import mingoFilter from '../mingoFilter.js';
import schema from './schema.js';
async function googleSheetUpdateMany({ request, connection }) {
async function GoogleSheetUpdateMany({ request, connection }) {
const { filter, update, options = {} } = request;
const { limit, skip, raw } = options;
const sheet = await getSheet({ connection });
@ -41,4 +42,10 @@ async function googleSheetUpdateMany({ request, connection }) {
};
}
export default googleSheetUpdateMany;
GoogleSheetUpdateMany.schema = schema;
GoogleSheetUpdateMany.meta = {
checkRead: false,
checkWrite: true,
};
export default GoogleSheetUpdateMany;

View File

@ -15,11 +15,10 @@
*/
import { validate } from '@lowdefy/ajv';
import googleSheetUpdateMany from './GoogleSheetUpdateMany';
import requestIndex from './index.js';
import schema from './GoogleSheetUpdateManySchema.json';
import GoogleSheetUpdateMany from './GoogleSheetUpdateMany.js';
const { checkRead, checkWrite } = requestIndex.meta;
const { checkRead, checkWrite } = GoogleSheetUpdateMany.meta;
const schema = GoogleSheetUpdateMany.schema;
const mockGetRows = jest.fn();
const mockSave = jest.fn();
@ -79,7 +78,7 @@ const mockGetRowsDefaultImp = ({ limit, offset }) => {
test('googleSheetUpdateMany, match one', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetUpdateMany({
const res = await GoogleSheetUpdateMany({
request: {
filter: { id: '1' },
update: {
@ -104,7 +103,7 @@ test('googleSheetUpdateMany, match one', async () => {
test('googleSheetUpdateMany, match one, raw true', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetUpdateMany({
const res = await GoogleSheetUpdateMany({
request: {
filter: { id: '1' },
update: {
@ -132,7 +131,7 @@ test('googleSheetUpdateMany, match one, raw true', async () => {
test('googleSheetUpdateMany, match nothing', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetUpdateMany({
const res = await GoogleSheetUpdateMany({
request: {
filter: { id: 'does_not_exist' },
update: {
@ -149,7 +148,7 @@ test('googleSheetUpdateMany, match nothing', async () => {
test('googleSheetUpdateMany, match more than one', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetUpdateMany({
const res = await GoogleSheetUpdateMany({
request: {
filter: { _rowNumber: { $gt: 3 } },
update: {

View File

@ -1,55 +0,0 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "Lowdefy Request Schema - GoogleSheetUpdateMany",
"type": "object",
"required": ["update", "filter"],
"properties": {
"filter": {
"type": "object",
"description": "A MongoDB query expression to filter the data. All rows matched by the filter will be updated.",
"errorMessage": {
"type": "GoogleSheetUpdateMany request property \"filter\" should be an object."
}
},
"update": {
"type": "object",
"description": "The update to apply to the row. An object where keys are the column names and values are the updated values.",
"errorMessage": {
"type": "GoogleSheetUpdateMany request property \"update\" should be an object."
}
},
"options": {
"type": "object",
"properties": {
"limit": {
"type": "number",
"description": "The maximum number of rows to fetch.",
"errorMessage": {
"type": "GoogleSheetUpdateMany request property \"options.limit\" should be a number."
}
},
"raw": {
"type": "boolean",
"description": "Store raw values instead of converting as if typed into the sheets UI.",
"errorMessage": {
"type": "GoogleSheetUpdateMany request property \"options.raw\" should be a boolean."
}
},
"skip": {
"type": "number",
"description": "The number of rows to skip from the top of the sheet.",
"errorMessage": {
"type": "GoogleSheetUpdateMany request property \"options.skip\" should be a number."
}
}
}
}
},
"errorMessage": {
"type": "GoogleSheetUpdateMany request properties should be an object.",
"required": {
"filter": "GoogleSheetUpdateMany request should have required property \"filter\".",
"update": "GoogleSheetUpdateMany request should have required property \"update\"."
}
}
}

View File

@ -1,26 +0,0 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
import: {
path: 'connections/GoogleSheet/GoogleSheetUpdateMany/GoogleSheetUpdateMany.js',
schema: 'connections/GoogleSheet/GoogleSheetUpdateMany/GoogleSheetUpdateManySchema.json',
},
meta: {
checkRead: false,
checkWrite: true,
},
};

View File

@ -0,0 +1,73 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
$schema: 'http://json-schema.org/draft-07/schema#',
title: 'Lowdefy Request Schema - GoogleSheetUpdateMany',
type: 'object',
required: ['update', 'filter'],
properties: {
filter: {
type: 'object',
description:
'A MongoDB query expression to filter the data. All rows matched by the filter will be updated.',
errorMessage: {
type: 'GoogleSheetUpdateMany request property "filter" should be an object.',
},
},
update: {
type: 'object',
description:
'The update to apply to the row. An object where keys are the column names and values are the updated values.',
errorMessage: {
type: 'GoogleSheetUpdateMany request property "update" should be an object.',
},
},
options: {
type: 'object',
properties: {
limit: {
type: 'number',
description: 'The maximum number of rows to fetch.',
errorMessage: {
type: 'GoogleSheetUpdateMany request property "options.limit" should be a number.',
},
},
raw: {
type: 'boolean',
description: 'Store raw values instead of converting as if typed into the sheets UI.',
errorMessage: {
type: 'GoogleSheetUpdateMany request property "options.raw" should be a boolean.',
},
},
skip: {
type: 'number',
description: 'The number of rows to skip from the top of the sheet.',
errorMessage: {
type: 'GoogleSheetUpdateMany request property "options.skip" should be a number.',
},
},
},
},
},
errorMessage: {
type: 'GoogleSheetUpdateMany request properties should be an object.',
required: {
filter: 'GoogleSheetUpdateMany request should have required property "filter".',
update: 'GoogleSheetUpdateMany request should have required property "update".',
},
},
};

View File

@ -18,8 +18,9 @@ import cleanRows from '../cleanRows.js';
import getSheet from '../getSheet.js';
import { transformRead, transformWrite } from '../transformTypes.js';
import mingoFilter from '../mingoFilter.js';
import schema from './schema.js';
async function googleSheetUpdateOne({ request, connection }) {
async function GoogleSheetUpdateOne({ request, connection }) {
const { filter, update, options = {} } = request;
const { limit, skip, upsert, raw } = options;
const sheet = await getSheet({ connection });
@ -51,4 +52,10 @@ async function googleSheetUpdateOne({ request, connection }) {
};
}
export default googleSheetUpdateOne;
GoogleSheetUpdateOne.schema = schema;
GoogleSheetUpdateOne.meta = {
checkRead: false,
checkWrite: true,
};
export default GoogleSheetUpdateOne;

View File

@ -15,11 +15,10 @@
*/
import { validate } from '@lowdefy/ajv';
import googleSheetUpdateOne from './GoogleSheetUpdateOne';
import requestIndex from './index.js';
import schema from './GoogleSheetUpdateOneSchema.json';
import GoogleSheetUpdateOne from './GoogleSheetUpdateOne.js';
const { checkRead, checkWrite } = requestIndex.meta;
const { checkRead, checkWrite } = GoogleSheetUpdateOne.meta;
const schema = GoogleSheetUpdateOne.schema;
const mockGetRows = jest.fn();
const mockAddRow = jest.fn();
@ -83,7 +82,7 @@ const mockGetRowsDefaultImp = ({ limit, offset }) => {
test('googleSheetUpdateOne, match one', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetUpdateOne({
const res = await GoogleSheetUpdateOne({
request: {
filter: { id: '1' },
update: {
@ -119,7 +118,7 @@ test('googleSheetUpdateOne, match one', async () => {
test('googleSheetUpdateOne, match one, raw true', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetUpdateOne({
const res = await GoogleSheetUpdateOne({
request: {
filter: { id: '1' },
update: {
@ -158,7 +157,7 @@ test('googleSheetUpdateOne, match one, raw true', async () => {
test('googleSheetUpdateOne, match nothing', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetUpdateOne({
const res = await GoogleSheetUpdateOne({
request: {
filter: { id: 'does_not_exist' },
update: {
@ -177,7 +176,7 @@ test('googleSheetUpdateOne, match nothing', async () => {
test('googleSheetUpdateOne, match nothing, upsert true', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
mockAddRow.mockImplementation(mockAddRowDefaultImp);
const res = await googleSheetUpdateOne({
const res = await GoogleSheetUpdateOne({
request: {
filter: { id: 'does_not_exist' },
update: {
@ -212,7 +211,7 @@ test('googleSheetUpdateOne, match nothing, upsert true', async () => {
test('googleSheetUpdateOne, match more than one', async () => {
mockGetRows.mockImplementation(mockGetRowsDefaultImp);
const res = await googleSheetUpdateOne({
const res = await GoogleSheetUpdateOne({
request: {
filter: { _rowNumber: { $gt: 3 } },
update: {

View File

@ -1,62 +0,0 @@
{
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "Lowdefy Request Schema - GoogleSheetUpdateOne",
"type": "object",
"required": ["update", "filter"],
"properties": {
"filter": {
"type": "object",
"description": "A MongoDB query expression to filter the data. The first row matched by the filter will be updated.",
"errorMessage": {
"type": "GoogleSheetUpdateOne request property \"filter\" should be an object."
}
},
"update": {
"type": "object",
"description": "The update to apply to the row. An object where keys are the column names and values are the updated values.",
"errorMessage": {
"type": "GoogleSheetUpdateOne request property \"update\" should be an object."
}
},
"options": {
"type": "object",
"properties": {
"limit": {
"type": "number",
"description": "The maximum number of rows to fetch.",
"errorMessage": {
"type": "GoogleSheetUpdateOne request property \"options.limit\" should be a number."
}
},
"raw": {
"type": "boolean",
"description": "Store raw values instead of converting as if typed into the sheets UI.",
"errorMessage": {
"type": "GoogleSheetUpdateOne request property \"options.raw\" should be a boolean."
}
},
"skip": {
"type": "number",
"description": "The number of rows to skip from the top of the sheet.",
"errorMessage": {
"type": "GoogleSheetUpdateOne request property \"options.skip\" should be a number."
}
},
"upsert": {
"type": "boolean",
"description": "Insert the row if no rows are matched by the filter.",
"errorMessage": {
"type": "GoogleSheetUpdateOne request property \"options.upsert\" should be a boolean."
}
}
}
}
},
"errorMessage": {
"type": "GoogleSheetUpdateOne request properties should be an object.",
"required": {
"filter": "GoogleSheetUpdateOne request should have required property \"filter\".",
"update": "GoogleSheetUpdateOne request should have required property \"update\"."
}
}
}

View File

@ -1,26 +0,0 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
import: {
path: 'connections/GoogleSheet/GoogleSheetUpdateOne/GoogleSheetUpdateOne.js',
schema: 'connections/GoogleSheet/GoogleSheetUpdateOne/GoogleSheetUpdateOneSchema.json',
},
meta: {
checkRead: false,
checkWrite: true,
},
};

View File

@ -0,0 +1,80 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
$schema: 'http://json-schema.org/draft-07/schema#',
title: 'Lowdefy Request Schema - GoogleSheetUpdateOne',
type: 'object',
required: ['update', 'filter'],
properties: {
filter: {
type: 'object',
description:
'A MongoDB query expression to filter the data. The first row matched by the filter will be updated.',
errorMessage: {
type: 'GoogleSheetUpdateOne request property "filter" should be an object.',
},
},
update: {
type: 'object',
description:
'The update to apply to the row. An object where keys are the column names and values are the updated values.',
errorMessage: {
type: 'GoogleSheetUpdateOne request property "update" should be an object.',
},
},
options: {
type: 'object',
properties: {
limit: {
type: 'number',
description: 'The maximum number of rows to fetch.',
errorMessage: {
type: 'GoogleSheetUpdateOne request property "options.limit" should be a number.',
},
},
raw: {
type: 'boolean',
description: 'Store raw values instead of converting as if typed into the sheets UI.',
errorMessage: {
type: 'GoogleSheetUpdateOne request property "options.raw" should be a boolean.',
},
},
skip: {
type: 'number',
description: 'The number of rows to skip from the top of the sheet.',
errorMessage: {
type: 'GoogleSheetUpdateOne request property "options.skip" should be a number.',
},
},
upsert: {
type: 'boolean',
description: 'Insert the row if no rows are matched by the filter.',
errorMessage: {
type: 'GoogleSheetUpdateOne request property "options.upsert" should be a boolean.',
},
},
},
},
},
errorMessage: {
type: 'GoogleSheetUpdateOne request properties should be an object.',
required: {
filter: 'GoogleSheetUpdateOne request should have required property "filter".',
update: 'GoogleSheetUpdateOne request should have required property "update".',
},
},
};

View File

@ -14,7 +14,7 @@
limitations under the License.
*/
import cleanRows from './cleanRows';
import cleanRows from './cleanRows.js';
test('cleanRows removes objects with key _sheet from an array of rows', () => {
expect(

View File

@ -17,7 +17,7 @@
import { wait } from '@lowdefy/helpers';
// eslint-disable-next-line no-unused-vars
import { GoogleSpreadsheet } from 'google-spreadsheet';
import getSheet from './getSheet';
import getSheet from './getSheet.js';
// Not testing if spreadsheetId is given to GoogleSpreadsheet class in
// const doc = new GoogleSpreadsheet(spreadsheetId);

View File

@ -16,18 +16,22 @@
import { type } from '@lowdefy/helpers';
import mingo from 'mingo';
import { useOperators, OperatorType } from 'mingo/core';
import * as accumulatorOperators from 'mingo/operators/accumulator';
import * as expressionOperators from 'mingo/operators/expression';
import * as pipelineOperators from 'mingo/operators/pipeline';
import * as queryOperators from 'mingo/operators/query';
import * as projectionOperators from 'mingo/operators/projection';
import { useOperators, OperatorType } from 'mingo/core.js';
import * as accumulatorOperators from 'mingo/operators/accumulator/index.js';
import * as expressionOperators from 'mingo/operators/expression/index.js';
import * as pipelineOperators from 'mingo/operators/pipeline/index.js';
import * as queryOperators from 'mingo/operators/query/index.js';
import * as projectionOperators from 'mingo/operators/projection/index.js';
useOperators(OperatorType.ACCUMULATOR, accumulatorOperators);
useOperators(OperatorType.EXPRESSION, expressionOperators);
useOperators(OperatorType.PIPELINE, pipelineOperators);
useOperators(OperatorType.PROJECTION, queryOperators);
useOperators(OperatorType.QUERY, projectionOperators);
// "import * as" is returning different object structures when the connection is being
// imported in the build or when running the tests.
// So we check for the a default object with all the named exports, otherwise the
// returned object has all the named exports.
useOperators(OperatorType.ACCUMULATOR, accumulatorOperators.default || accumulatorOperators);
useOperators(OperatorType.EXPRESSION, expressionOperators.default || expressionOperators);
useOperators(OperatorType.PIPELINE, pipelineOperators.default || pipelineOperators);
useOperators(OperatorType.QUERY, queryOperators.default || queryOperators);
useOperators(OperatorType.PROJECTION, projectionOperators.default || projectionOperators);
function mingoAggregation({ input = [], pipeline = [] }) {
if (!type.isArray(input)) {

View File

@ -14,7 +14,7 @@
limitations under the License.
*/
import mingoAggregation from './mingoAggregation';
import mingoAggregation from './mingoAggregation.js';
test('mingoAggregation sort', () => {
const pipeline = [

View File

@ -15,7 +15,7 @@
*/
import { type } from '@lowdefy/helpers';
import mingoAggregation from './mingoAggregation';
import mingoAggregation from './mingoAggregation.js';
function mingoFilter({ input = [], filter = {} }) {
if (!type.isObject(filter)) {

View File

@ -14,7 +14,7 @@
limitations under the License.
*/
import mingoFilter from './mingoFilter';
import mingoFilter from './mingoFilter.js';
test('mingoFilter equals shorthand', () => {
const filter = { id: 1 };

View File

@ -0,0 +1,106 @@
/*
Copyright 2020-2021 Lowdefy, Inc
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
export default {
$schema: 'http://json-schema.org/draft-07/schema#',
title: 'Lowdefy Connection Schema - GoogleSheet',
type: 'object',
required: ['spreadsheetId'],
properties: {
apiKey: {
type: 'string',
description: 'API key for your google project.',
errorMessage: {
type: 'GoogleSheet connection property "apiKey" should be a string.',
},
},
client_email: {
type: 'string',
description: 'The email of your service account.',
errorMessage: {
type: 'GoogleSheet connection property "client_email" should be a string.',
},
},
private_key: {
type: 'string',
description: 'The private key for your service account.',
errorMessage: {
type: 'GoogleSheet connection property "private_key" should be a string.',
},
},
sheetId: {
type: 'string',
description:
'The ID of the worksheet. Can be found in the URL as the "gid" parameter. One of "sheetId" or "sheetIndex" is required.',
errorMessage: {
type: 'GoogleSheet connection property "sheetId" should be a string.',
},
},
sheetIndex: {
type: 'number',
description:
'The position of the worksheet as they appear in the Google sheets UI. Starts from 0. One of "sheetId" or "sheetIndex" is required.',
errorMessage: {
type: 'GoogleSheet connection property "sheetIndex" should be a number.',
},
},
spreadsheetId: {
type: 'string',
description: 'document ID from the URL of the spreadsheet.',
errorMessage: {
type: 'GoogleSheet connection property "spreadsheetId" should be a string.',
},
},
columnTypes: {
type: 'object',
description: 'Define types for columns in the spreadsheet.',
patternProperties: {
'^.*$': {
type: 'string',
enum: ['string', 'number', 'boolean', 'date', 'json'],
errorMessage: {
enum: 'GoogleSheet connection property "{{ instancePath }}" should be one of "string", "number", "boolean", "date", or "json".',
},
},
},
errorMessage: {
type: 'GoogleSheet connection property "columnTypes" should be an object.',
},
},
read: {
type: 'boolean',
default: true,
description: 'Allow reads from the spreadsheet.',
errorMessage: {
type: 'GoogleSheet connection property "read" should be a boolean.',
},
},
write: {
type: 'boolean',
default: false,
description: 'Allow writes to the spreadsheet.',
errorMessage: {
type: 'GoogleSheet connection property "write" should be a boolean.',
},
},
},
errorMessage: {
type: 'GoogleSheet connection properties should be an object.',
required: {
spreadsheetId: 'GoogleSheet connection should have required property "spreadsheetId".',
},
},
};

View File

@ -14,7 +14,7 @@
limitations under the License.
*/
import { transformRead } from './transformTypes';
import { transformRead } from './transformTypes.js';
test('transformRead invalid input', () => {
expect(() => transformRead({ input: 1 })).toThrow(

View File

@ -14,7 +14,7 @@
limitations under the License.
*/
import { transformWrite } from './transformTypes';
import { transformWrite } from './transformTypes.js';
test('transformWrite invalid input', () => {
expect(() => transformWrite({ input: 1 })).toThrow(

View File

@ -1,7 +0,0 @@
import GoogleSheet from './connections/GoogleSheet/GoogleSheet.js';
export const connections = {
GoogleSheet,
};
export default { connections };

View File

@ -1,3 +1,4 @@
/* eslint-disable import/namespace */
/*
Copyright 2020-2021 Lowdefy, Inc
@ -14,13 +15,24 @@
limitations under the License.
*/
import * as connections from './connections.js';
export default {
import: {
path: 'connections/GoogleSheet/GoogleSheetGetManySchema/GoogleSheetGetMany.js',
schema: 'connections/GoogleSheet/GoogleSheetGetManySchema/GoogleSheetGetManySchema.json',
},
meta: {
checkRead: true,
checkWrite: false,
},
connections: Object.keys(connections),
requests: Object.keys(connections)
.map((connection) => Object.keys(connections[connection].requests))
.flat(),
};
// export default {
// connections: ['GoogleSheet'],
// requests: [
// GoogleSheetAppendMany,
// GoogleSheetAppendOne,
// GoogleSheetDeleteOne,
// GoogleSheetGetMany,
// GoogleSheetGetOne,
// GoogleSheetUpdateOne,
// GoogleSheetUpdateMany,
// ],
// };

View File

@ -3503,6 +3503,7 @@ __metadata:
"@lowdefy/blocks-markdown": 4.0.0-alpha.6
"@lowdefy/connection-axios-http": 4.0.0-alpha.6
"@lowdefy/connection-elasticsearch": 4.0.0-alpha.6
"@lowdefy/connection-google-sheets": 4.0.0-alpha.6
"@lowdefy/connection-knex": 4.0.0-alpha.6
"@lowdefy/connection-mongodb": 4.0.0-alpha.6
"@lowdefy/connection-redis": 4.0.0-alpha.6
@ -3560,7 +3561,7 @@ __metadata:
languageName: unknown
linkType: soft
"@lowdefy/connection-google-sheets@workspace:packages/plugins/connections/connection-google-sheets":
"@lowdefy/connection-google-sheets@4.0.0-alpha.6, @lowdefy/connection-google-sheets@workspace:packages/plugins/connections/connection-google-sheets":
version: 0.0.0-use.local
resolution: "@lowdefy/connection-google-sheets@workspace:packages/plugins/connections/connection-google-sheets"
dependencies: