-
Notifications
You must be signed in to change notification settings - Fork 2.6k
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
Save Markdown and BlockNote in RICH_TEXT fields (#7613) #9279
Save Markdown and BlockNote in RICH_TEXT fields (#7613) #9279
Conversation
TODOs/FIXMEs:
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
PR Summary
This PR transforms RICH_TEXT fields into composite fields that store both BlockNote and Markdown formats, enabling dual-format storage and format choice during CSV imports.
- Adds new
rich-text.composite-type.ts
defining the composite structure with nullableblocknote
andmarkdown
TEXT fields - Modifies search functionality in task/note entities to use
bodyBlocknote
instead ofbody
, with a TODO to verify search behavior - Moves RICH_TEXT handling from
basicColumnActionFactory
tocompositeColumnActionFactory
for database migrations - Updates OpenAPI schema and Zapier integration to handle RICH_TEXT as a composite field with both formats
- Needs proper example values in
SettingsCompositeFieldTypeConfigs.ts
(currently marked as TODO)
22 file(s) reviewed, 11 comment(s)
Edit PR Review Bot Settings | Greptile
export const richTextSchema: z.ZodType<FieldRichTextValue> = z.object({ | ||
blocknote: z.string().nullable(), | ||
markdown: z.string().nullable(), | ||
}); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
style: consider adding validation to ensure at least one of blocknote or markdown is non-null to prevent empty objects
blocknote: 'TODO', // TODO | ||
markdown: 'TODO', |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
logic: example values are marked as TODO and need to be properly defined with realistic sample data
label: 'Rich Text', | ||
Icon: IllustrationIconText, | ||
subFields: ['blocknote', 'markdown'], | ||
filterableSubFields: [], |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
style: consider adding 'markdown' to filterableSubFields to enable searching/filtering by content
@@ -21,7 +21,7 @@ export class ActivityQueryResultGetterHandler | |||
return activity; | |||
} | |||
|
|||
const body: RichTextBody = JSON.parse(activity.body); | |||
const body: RichTextBody = JSON.parse(activity.body.blocknote); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
logic: add try/catch around JSON.parse to handle invalid JSON gracefully
} | ||
}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
logic: missing return type for undefined case - could cause runtime issues if field type is not handled
export type RichTextMetadata = { | ||
blocknote: string; | ||
markdown: string; | ||
}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
logic: both fields are defined as required in the type but marked as not required in the composite type definition above. This inconsistency could cause runtime issues
export class FieldMetadataDefaultValueRichText { | ||
@ValidateIf((_object, value) => value !== null) | ||
@IsString() | ||
value: string | null; | ||
@ValidateIf((object, value) => value !== null) | ||
@IsQuotedString() | ||
blocknote: string | null; | ||
|
||
@ValidateIf((object, value) => value !== null) | ||
@IsQuotedString() | ||
markdown: string | null; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
logic: Both blocknote and markdown fields can be null simultaneously. Consider adding validation to ensure at least one format is provided.
@@ -37,7 +38,7 @@ const BODY_FIELD_NAME = 'body'; | |||
|
|||
export const SEARCH_FIELDS_FOR_TASK: FieldTypeAndNameMetadata[] = [ | |||
{ name: TITLE_FIELD_NAME, type: FieldMetadataType.TEXT }, | |||
{ name: BODY_FIELD_NAME, type: FieldMetadataType.RICH_TEXT }, | |||
{ name: `${BODY_FIELD_NAME}Blocknote`, type: FieldMetadataType.RICH_TEXT }, // TODO: Check later if and how this works |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
logic: search field using 'bodyBlocknote' may break full-text search for existing tasks since the field name changed from 'body' to 'bodyBlocknote'
@@ -35,7 +36,7 @@ const BODY_FIELD_NAME = 'body'; | |||
|
|||
export const SEARCH_FIELDS_FOR_NOTES: FieldTypeAndNameMetadata[] = [ | |||
{ name: TITLE_FIELD_NAME, type: FieldMetadataType.TEXT }, | |||
{ name: BODY_FIELD_NAME, type: FieldMetadataType.RICH_TEXT }, | |||
{ name: `${BODY_FIELD_NAME}Blocknote`, type: FieldMetadataType.RICH_TEXT }, // TODO: Check later if and how this works |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
logic: search field name change from 'body' to 'bodyBlocknote' may break existing search functionality if not properly migrated
name: 'blocknote', | ||
label: 'Blocknote', | ||
description: 'Blocknote', |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
style: inconsistent casing between 'blocknote' and 'BlockNote' in label/description
@eliasylonen Could you link the original issue ? |
@lucasbordeau I edited the description to add the related issues. I think we should create a separate issue for the input in table views as this PR is going to be quite big already (it's still wip). Would you mind creating it if you can point to the right direction? I'm not sure what has been done/is left to be done |
Could we use deprecated instead of old ? |
Shouldn't we add the type RICH_TEXT next to the RICH_TEXT_OLD/DEPRECATED ? |
@lucasbordeau Yes 👍 will do! |
...enty-front/src/modules/settings/data-model/constants/SettingsNonCompositeFieldTypeConfigs.ts
Outdated
Show resolved
Hide resolved
Nice, @FelixMalfait Is it ok for you ? |
) | ||
) { | ||
// TODO: Convert back and forth from BlockNote. | ||
// Has to happen server-side, because API shouldn't require both fields? How / where? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you happen to have time for a quick call tomorrow @lucasbordeau? Shouldn't take long & needs no prep
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Exciting work!
I have to do some work related to the Rich Text field and stumbled upon this PR. I left a comment related to the blocknote library. I'm thrilled to see these changes live!
const { blocksToMarkdownLossy, tryParseMarkdownToBlocks } = | ||
ServerBlockNoteEditor.create(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think you can't use destructuring assignment because ServerBlockNoteEditor.create()
returns a new ServerBlockNoteEditor
instance. The blocksToMarkdownLossy
and tryParseMarkdownToBlocks
methods use this
, as seen here, which is unbound when calling a destructured function like this.
Instead, I would keep a reference to the editor and call the functions like so:
const editor = ServerBlockNoteEditor.create();
editor.tryParseMarkdownToBlocks();
I couldn't make the functions work like this, but I might have missed something. Feel free to tell me!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, works now! Thank you @Devessier!
Unrelated: there seems to be some issue in the latest version (0.22.0) of @blocknote/server-util
where importing fails with ERR_REQUIRE_ESM
. Downgrading to 0.17.1 helped.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The last versions of @blocknote/server-util
are ESM-only. It seems our backend doesn't support importing ESM-only packages. You can use dynamic import:
async function overrideDataByFieldMetadata() {
const { ServerBlockNoteEditor } = await import('@blocknote/server-util')
const editor = ServerBlockNoteEditor.create()
// etc...
}
TODO:
|
Elias this PR gets too big, once it's not WIP anymore, could you split it in multiple small PRs ? It would be easier to review. For example maybe the migration command could be done in a separate PR ? I'm here to review when the implementation is ok. |
I moved the upgrade command to PR #9534 |
@eliasylonen let me know when you feel this one is ready for review :) |
@charlesBochet Will do! Still need to make a few changes to make the upgrade process smooth, talked yesterday with @lucasbordeau! TODO:
Then it's ready! |
@charlesBochet Ready for a review! The upgrade command PR (#9534) is also ready |
@@ -36,7 +37,7 @@ const BODY_FIELD_NAME = 'body'; | |||
|
|||
export const SEARCH_FIELDS_FOR_NOTES: FieldTypeAndNameMetadata[] = [ | |||
{ name: TITLE_FIELD_NAME, type: FieldMetadataType.TEXT }, | |||
{ name: BODY_FIELD_NAME, type: FieldMetadataType.RICH_TEXT }, | |||
// { name: BODY_FIELD_NAME, type: FieldMetadataType.RICH_TEXT_V2 }, // TODO: Check later if and how this works |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This one is still missing, I'll work on it tomorrow
@lucasbordeau @Weiko it seems that there is still some left overs but let's re-review it to help unblocking it :) |
TODO: Create the following three PRs.
|
@eliasylonen I'm closing this PR as we try to keep a clean PR list, please re-open a new PR for step 1. |
RICH_TEXT
field composite: instead of saving only BlockNote also save a lossy copy of the value in Markdown format.Fixes #7613
Fixes #7444