fix(Notion Node): Regex for block id (#8860)

This commit is contained in:
Michael Kret 2024-03-13 13:57:17 +02:00 committed by GitHub
parent e78cc2d8d2
commit a1f6c570d6
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
5 changed files with 124 additions and 10 deletions

View file

@ -13,13 +13,14 @@ export class Notion extends VersionedNodeType {
group: ['output'],
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
description: 'Consume Notion API',
defaultVersion: 2.1,
defaultVersion: 2.2,
};
const nodeVersions: IVersionedNodeType['nodeVersions'] = {
1: new NotionV1(baseDescription),
2: new NotionV2(baseDescription),
2.1: new NotionV2(baseDescription),
2.2: new NotionV2(baseDescription),
};
super(nodeVersions, baseDescription);

View file

@ -41,6 +41,7 @@ const apiVersion: { [key: number]: string } = {
1: '2021-05-13',
2: '2021-08-16',
2.1: '2021-08-16',
2.2: '2021-08-16',
};
export async function notionApiRequest(
@ -1128,3 +1129,44 @@ export function simplifyBlocksOutput(blocks: IDataObject[], rootId: string) {
return blocks;
}
export function extractBlockId(this: IExecuteFunctions, nodeVersion: number, itemIndex: number) {
let blockId: string;
if (nodeVersion < 2.2) {
blockId = extractPageId(
this.getNodeParameter('blockId', itemIndex, '', { extractValue: true }) as string,
);
} else {
const blockIdRLCData = this.getNodeParameter('blockId', itemIndex, {}) as IDataObject;
if (blockIdRLCData.mode === 'id') {
blockId = blockIdRLCData.value as string;
} else {
const blockRegex = /https:\/\/www\.notion\.so\/.+\?pvs=[0-9]+#([a-f0-9]{2,})/;
const match = (blockIdRLCData.value as string).match(blockRegex);
if (match === null) {
const pageRegex =
/(?:https|http):\/\/www\.notion\.so\/(?:[a-z0-9-]{2,}\/)?(?:[a-zA-Z0-9-]{2,}-)?([0-9a-f]{8}[0-9a-f]{4}4[0-9a-f]{3}[89ab][0-9a-f]{3}[0-9a-f]{12})/;
const pageMatch = (blockIdRLCData.value as string).match(pageRegex);
if (pageMatch === null) {
throw new NodeOperationError(
this.getNode(),
'Invalid URL, could not find block ID or page ID',
{
itemIndex,
},
);
} else {
blockId = extractPageId(pageMatch[1]);
}
} else {
blockId = match[1];
}
}
}
return blockId;
}

View file

@ -2,6 +2,55 @@ import type { INodeProperties } from 'n8n-workflow';
import { blocks } from './Blocks';
//RLC with fixed regex for blockId
const blockIdRLC: INodeProperties = {
displayName: 'Block',
name: 'blockId',
type: 'resourceLocator',
default: { mode: 'url', value: '' },
required: true,
modes: [
{
displayName: 'Link',
name: 'url',
type: 'string',
placeholder:
'e.g. https://www.notion.so/Block-Test-88888ccc303e4f44847f27d24bd7ad8e?pvs=4#c44444444444bbbbb4d32fdfdd84e',
validation: [
{
type: 'regex',
properties: {
regex:
'(?:https|http)://www.notion.so/(?:[a-z0-9-]{2,}/)?(?:[a-zA-Z0-9-]{2,}-)?([0-9a-f]{8}[0-9a-f]{4}4[0-9a-f]{3}[89ab][0-9a-f]{3}[0-9a-f]{12}).*',
errorMessage: 'Not a valid Notion Block URL',
},
},
],
// extractValue: {
// type: 'regex',
// regex: 'https:\\/\\/www\\.notion\\.so\\/.+\\?pvs=[0-9]+#([a-f0-9]{2,})',
// },
},
{
displayName: 'ID',
name: 'id',
type: 'string',
placeholder: 'e.g. ab1545b247fb49fa92d6f4b49f4d8116',
validation: [
{
type: 'regex',
properties: {
regex: '[a-f0-9]{2,}',
errorMessage: 'Not a valid Notion Block ID',
},
},
],
},
],
description:
"The Notion Block to get all children from, when using 'By URL' mode make sure to use the URL of the block itself, you can find it in block parameters in Notion under 'Copy link to block'",
};
export const blockOperations: INodeProperties[] = [
{
displayName: 'Operation',
@ -91,9 +140,22 @@ export const blockFields: INodeProperties[] = [
resource: ['block'],
operation: ['append'],
},
hide: {
'@version': [{ _cnd: { gte: 2.2 } }],
},
},
description: 'The Notion Block to append blocks to',
},
{
...blockIdRLC,
displayOptions: {
show: {
resource: ['block'],
operation: ['append'],
'@version': [{ _cnd: { gte: 2.2 } }],
},
},
},
...blocks('block', 'append'),
/* -------------------------------------------------------------------------- */
/* block:getAll */
@ -153,9 +215,22 @@ export const blockFields: INodeProperties[] = [
resource: ['block'],
operation: ['getAll'],
},
hide: {
'@version': [{ _cnd: { gte: 2.2 } }],
},
},
description: 'The Notion Block to get all children from',
},
{
...blockIdRLC,
displayOptions: {
show: {
resource: ['block'],
operation: ['getAll'],
'@version': [{ _cnd: { gte: 2.2 } }],
},
},
},
{
displayName: 'Return All',
name: 'returnAll',

View file

@ -11,6 +11,7 @@ import { jsonParse, NodeApiError } from 'n8n-workflow';
import type { SortData, FileRecord } from '../shared/GenericFunctions';
import {
downloadFiles,
extractBlockId,
extractDatabaseId,
extractDatabaseMentionRLC,
extractPageId,
@ -45,6 +46,7 @@ export class NotionV2 implements INodeType {
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
const items = this.getInputData();
const nodeVersion = this.getNode().typeVersion;
const resource = this.getNodeParameter('resource', 0);
const operation = this.getNodeParameter('operation', 0);
@ -60,9 +62,7 @@ export class NotionV2 implements INodeType {
if (operation === 'append') {
for (let i = 0; i < itemsLength; i++) {
try {
const blockId = extractPageId(
this.getNodeParameter('blockId', i, '', { extractValue: true }) as string,
);
const blockId = extractBlockId.call(this, nodeVersion, i);
const blockValues = this.getNodeParameter(
'blockUi.blockValues',
i,
@ -100,9 +100,7 @@ export class NotionV2 implements INodeType {
if (operation === 'getAll') {
for (let i = 0; i < itemsLength; i++) {
try {
const blockId = extractPageId(
this.getNodeParameter('blockId', i, '', { extractValue: true }) as string,
);
const blockId = extractBlockId.call(this, nodeVersion, i);
const returnAll = this.getNodeParameter('returnAll', i);
const fetchNestedBlocks = this.getNodeParameter('fetchNestedBlocks', i) as boolean;
@ -148,8 +146,6 @@ export class NotionV2 implements INodeType {
..._data,
}));
const nodeVersion = this.getNode().typeVersion;
if (nodeVersion > 2) {
const simplifyOutput = this.getNodeParameter('simplifyOutput', i) as boolean;

View file

@ -18,7 +18,7 @@ export const versionDescription: INodeTypeDescription = {
name: 'notion',
icon: 'file:notion.svg',
group: ['output'],
version: [2, 2.1],
version: [2, 2.1, 2.2],
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
description: 'Consume Notion API',
defaults: {