Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow to edit labels of standard objects #10922

Merged
merged 36 commits into from
Mar 24, 2025

Conversation

AFCMS
Copy link
Contributor

@AFCMS AFCMS commented Mar 15, 2025

Fixes #10793

This PR is a work in progress.

Still left to fix:

  • When disabling synchronization of labels / api names, the edited labels should be set to the English version. Currently the client just send the localized versions together with the isLabelSyncedWithName change. Could be an easy fix.
  • Sometimes flipping the switch don't trigger the update function, may be a regression as it seems to affect the custom objects too.
  • There is a frontend problem where the labels inputs don't reflect the changes made. When enabling back synchronisation after editing labels, they are correctly back to their base values (backend, navigation breadcrumb, etc) but the label inputs still have the old values (switching pages will put them back to normal). I suspect this could be linked to the above problem.
  • API names are still displayed for standard objects per (kept them for debugging, trivial fix)
  • SettingsDataModelObjectAboutForm have a disableEdition parameter which is now used only for a few fields, not sure if it's worth keeping because it's a bit misleading since it doesn't "disable" much?
  • I don't know what these do, but I have seen "Remote" object types. Not sure if they work with my patch or not (I don't know how to test them)
  • Make it work with metadata synchronisation

What should work:

  • Disabling synchronization of standard objects should work, label inputs should no longer be disabled
  • Modifying labels should work
  • Enabling back synchronization should reset back the labels to the base value and disable the label inputs again (minus the mentioned display bug)
  • The synchronisation switch should still work as expected for custom objects
  • Creating custom objects should still work (it uses the same form)

Sorry, something went wrong.

Copy link
Member

@FelixMalfait FelixMalfait left a comment

Choose a reason for hiding this comment

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

Good start!!! Now what happens if I run yarn command:prod workspace:sync-metadata, will it get erased? What should we do next?

I don't know what these do, but I have seen "Remote" object types. Not sure if they work with my patch or not (I don't know how to test them)

Remote objects are foreign tables that live on a different server. In that case we shouldn't be able to rename the API. I don't have a strong opinion on whether label name should be editable or not in that case, do what makes the code simpler (as long as you cannot edit api name)

I haven't tested the code yet, will test more soon after you make updates. Thank you!

if (
objectMetadata.isCustom ||
(objectMetadata.isLabelSyncedWithName === false &&
['labelPlural', 'labelSingular'].includes(labelKey))
Copy link
Member

Choose a reason for hiding this comment

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

We should probably let people edit the description all of the time too. Sorry this should have been in the initial spec. There is a product flaw here in the way we define what gets overwritten by new standard metadata changes, I think it's fine to put it under the same toggle but this is something we might want to reconsider in the future

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Would also allowing to change the icon make sense?

Copy link
Member

@FelixMalfait FelixMalfait Mar 17, 2025

Choose a reason for hiding this comment

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

Yes great point! Icon is similar to description

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Editing icon and description is now possible, but the sync switch should at the very least have a different label.

);
}
}
} else {
Copy link
Member

Choose a reason for hiding this comment

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

We most often try avoid using elseor elseif and favor early returns which makes the code more readable (if x return y; if z return d;). We also avoid nesting if structures within another

if (value === true) {
if (
value === true &&
(!objectMetadataItem || objectMetadataItem?.isCustom)
Copy link
Member

Choose a reason for hiding this comment

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

If I set the value to true for a standard object, then the Label should be updated back to its value as you rightly pointed out on Discord. Is that done purely server-side and then it's updated with the response? I'm reviewing the code but haven't tested it yet

Copy link
Contributor Author

Choose a reason for hiding this comment

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

It should be purely server-side, since translations of labels are server-side. Label inputs should updated with the response (currently the update are propagated to the rest of the app, but not the label inputs, as I mentioned in the PR description)

@AFCMS
Copy link
Contributor Author

AFCMS commented Mar 17, 2025

About the form problem mentioned in the PR description:

I have determined that when you flip the switch two times in a row, the second time when the handleSave function is called, it doesn't find any "dirty" fields and logically don't save anything. No idea why there are no dirty fields yet.


const standardObjectMap = new Map<string, WorkspaceEntityMetadataArgs>();

standardObjectMetadataDefinitions.forEach((entity) => {
Copy link
Member

Choose a reason for hiding this comment

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

this does not feel right, this will perform a side effect when importing the utils

Copy link
Member

Choose a reason for hiding this comment

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

I get why you wouldn't want side effects but I think this is efficient from a performance perspective. How would you do this?

Copy link
Member

Choose a reason for hiding this comment

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

See #10922 (comment) ; it's my comment that Louis into this direction. I didn't like the previous solution

Copy link
Member

Choose a reason for hiding this comment

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

Digging a bit more:

  1. I'm also not sure what is specific to standard vs custom. It feels that they should be handled the same way
  2. We would usually load this kind of information from the database metadata (or actually the redis metadata) as this will differ from what we have in the "definitions" (aka "workspace-entities") as long as the sync metadata has not been run
  3. If we really want to parse the workspace entity (which I'm not convinced we shoud!), I would have a dedicated singleton service handled by nest container to do that. This way it's only instanciated once
  4. If for some reason we don't want to use a nest js service, and want to do raw javascript, I would introduce another util with a very explicit naming: initStandardObjectMetadataFromDefMap instead of a getStandardObjectMetadataFromDefinition which I don't expect to create a new Map ==> this is likely to end up in a memory leak IMO!

I would prefer to go with 1) and 2), then if we can't 3) and then last solution 4)

@FelixMalfait
Copy link
Member

Screenshot 2025-03-20 at 12 36 47

Also lake sure ton configure your linter in VSCode so you don't break the CI :)

objectMetadata.standardId
) {
const standardObjectMetadata =
getStandardObjectMetadataFromDefinition(
Copy link
Member

Choose a reason for hiding this comment

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

TBH, I would load the metadata from the cache here and filter on standards (actually we only need to filter with standardId)

AFCMS added 9 commits March 22, 2025 15:53

Verified

This commit was signed with the committer’s verified signature.
Signed-off-by: AFCMS <[email protected]>

Verified

This commit was signed with the committer’s verified signature.
Do not send the label changes when switching `isLabelSyncedWithName` to false

Verified

This commit was signed with the committer’s verified signature.

Verified

This commit was signed with the committer’s verified signature.
…nition`

Verified

This commit was signed with the committer’s verified signature.

Verified

This commit was signed with the committer’s verified signature.

Verified

This commit was signed with the committer’s verified signature.

Verified

This commit was signed with the committer’s verified signature.

Verified

This commit was signed with the committer’s verified signature.
@AFCMS AFCMS force-pushed the standard_obj_label_edit branch from 033df7f to 6838a85 Compare March 22, 2025 14:53
AFCMS and others added 12 commits March 22, 2025 16:47

Verified

This commit was signed with the committer’s verified signature.

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature.
@FelixMalfait
Copy link
Member

@greptileai review

Copy link
Contributor

@greptile-apps greptile-apps bot left a 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 adds support for editing labels of standard objects while preserving API names through a new standardOverrides mechanism in both frontend and backend.

  • Added standardOverrides JSONB column to fieldMetadata and objectMetadata tables via migration 1742736630054 to store overridden labels/descriptions
  • Modified SettingsDataModelObjectAboutForm and SettingsDataModelFieldIconLabelForm to allow label editing when isLabelSyncedWithName is false
  • Added validation in before-update-one-object.hook.ts to prevent API name changes for standard objects while allowing label updates
  • Renamed resolveTranslatableString to resolveOverridableString in services to handle overridden values
  • Needs testing with workspace:sync-metadata command to ensure overrides persist during synchronization

💡 (1/5) You can manually trigger the bot by mentioning @greptileai in a comment!

19 file(s) reviewed, 15 comment(s)
Edit PR Review Bot Settings | Greptile

Comment on lines 84 to 87
if (
formValues.isLabelSyncedWithName === false &&
isLabelSyncedWithName === true
) {
Copy link
Contributor

Choose a reason for hiding this comment

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

logic: Missing check for when isLabelSyncedWithName changes from false to true - could leave labels in an inconsistent state

Comment on lines 170 to 174
disabled={
disabled || (isLabelSyncedWithName ?? false)
disabled ||
(isLabelSyncedWithName ?? false) ||
!fieldMetadataItem?.isCustom
}
Copy link
Contributor

Choose a reason for hiding this comment

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

style: API name field's disabled state should be simplified to avoid potential bugs. Consider extracting this logic into a separate function for better maintainability.

Comment on lines 323 to 326
if (
value === true &&
(!objectMetadataItem || objectMetadataItem?.isCustom)
) {
Copy link
Contributor

Choose a reason for hiding this comment

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

logic: Redundant condition check - !objectMetadataItem || objectMetadataItem?.isCustom. The optional chaining is unnecessary since we already check if objectMetadataItem is undefined

Suggested change
if (
value === true &&
(!objectMetadataItem || objectMetadataItem?.isCustom)
) {
if (
value === true &&
(!objectMetadataItem || objectMetadataItem.isCustom)
) {

Comment on lines +97 to +102
const isUpdatingLabelsWhenSynced =
(instance.update.labelSingular || instance.update.labelPlural) &&
objectMetadata.isLabelSyncedWithName &&
instance.update.isLabelSyncedWithName !== false &&
(instance.update.labelSingular !== objectMetadata.labelSingular ||
instance.update.labelPlural !== objectMetadata.labelPlural);
Copy link
Contributor

Choose a reason for hiding this comment

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

logic: This condition may miss edge cases where labelSingular/Plural are set to empty strings or other falsy values. Consider using isDefined() here for consistency.

Copy link
Collaborator

@ijreilly ijreilly left a comment

Choose a reason for hiding this comment

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

Will resume review when you ping me @FelixMalfait !

if (
formValues.isLabelSyncedWithName === false &&
isLabelSyncedWithName === true
) {
Copy link
Collaborator

Choose a reason for hiding this comment

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

what if in the same form we update isLabelSyncedWithName value from true to false + the label value ? from this code I feel we would only update isLabelSyncedWithName then return?

fillNamePluralFromLabelPlural(labelPlural);
fillNameSingularFromLabelSingular(labelSingular);
}
onNewDirtyField?.();

// Server-side side effect when isLabelSyncedWithName is changed
if (
Copy link
Collaborator

Choose a reason for hiding this comment

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

No call is sent when we are only updating isLabelSyncedWithName value
(recording does not show Network tab but I checked and no call was sent)

Enregistrement.de.l.ecran.2025-03-24.a.11.28.15.mov

Copy link
Collaborator

Choose a reason for hiding this comment

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

@FelixMalfait did you create a ticket for this one in the end ? or was it fixed ?

fieldMetadata: FieldMetadataEntity,
): UpdateOneInputType<T> {
const update: StandardFieldUpdate = {};
const allowedFields = ['isActive', 'isLabelSyncedWithName'];
Copy link
Collaborator

@ijreilly ijreilly Mar 24, 2025

Choose a reason for hiding this comment

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

updatableFields? fieldsAllowedForUpdate? (trying to find a more precise naming)

update: StandardFieldUpdate,
): void {
if (!isDefined(instance.update.description)) {
return;
Copy link
Collaborator

Choose a reason for hiding this comment

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

can we not update to an empty description ? Maybe we jsut want to evaluate
if(!('description' in instance.update)) ?

FelixMalfait and others added 9 commits March 24, 2025 11:58

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature.
@FelixMalfait FelixMalfait marked this pull request as ready for review March 24, 2025 14:16
Copy link
Contributor

@greptile-apps greptile-apps bot left a comment

Choose a reason for hiding this comment

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

PR Summary

(updates since last review)

Based on the recent changes and previous reviews, I'll focus on new developments and unaddressed points:

This PR continues work on standard object label editing, with significant progress on the standardOverrides implementation and validation logic.

  • Added proper validation in before-update-one-field.hook.ts to handle label synchronization and overrides for standard fields
  • Introduced type-safe FieldStandardOverridesDTO and ObjectStandardOverridesDTO for managing override data
  • Potential SQL injection vulnerability in field-metadata.service.ts createViewAndViewFields needs attention
  • The any type casting for standardOverrides in multiple services should be replaced with proper types
  • Transaction handling in deleteOneField needs review for multiple queryRunner safety

21 file(s) reviewed, 7 comment(s)
Edit PR Review Bot Settings | Greptile

@@ -143,7 +141,7 @@ export const SettingsDataModelFieldIconLabelForm = ({
}
}}
error={getErrorMessageFromError(errors.label?.message)}
disabled={disabled}
disabled={isLabelSyncedWithName === true}
Copy link
Contributor

Choose a reason for hiding this comment

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

logic: Label input's disabled state is incorrect - should be isLabelSyncedWithName === true && !fieldMetadataItem?.isCustom to allow editing standard object labels when sync is disabled

idToUpdate: objectMetadataItem.id,
updatePayload: formValues,
});
await updateObjectMetadata(formValues);

formConfig.reset(undefined, { keepValues: true });
Copy link
Contributor

Choose a reason for hiding this comment

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

logic: Calling reset with keepValues: true after a successful update may prevent the form from reflecting server-side changes

Comment on lines 83 to 94
if (!objectMetadataItem.isCustom) {
const {
nameSingular: _,
namePlural: __,
...payloadWithoutNames
} = updatePayload;

return updateOneObjectMetadataItem({
idToUpdate: objectMetadataItem.id,
updatePayload: payloadWithoutNames,
});
}
Copy link
Contributor

Choose a reason for hiding this comment

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

style: Destructuring with _ and __ is not a clear naming convention. Consider using more descriptive names like 'unusedSingular' and 'unusedPlural'

Comment on lines 342 to 344
{
keepDirty: true,
},
Copy link
Contributor

Choose a reason for hiding this comment

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

style: Keeping dirty state during reset could lead to unexpected form behavior and validation issues

@FelixMalfait FelixMalfait force-pushed the standard_obj_label_edit branch from 0fd497c to fc02d6a Compare March 24, 2025 19:06
@FelixMalfait FelixMalfait merged commit 52cf6f4 into twentyhq:main Mar 24, 2025
45 checks passed
@AFCMS AFCMS deleted the standard_obj_label_edit branch March 24, 2025 20:07
Copy link
Collaborator

@ijreilly ijreilly left a comment

Choose a reason for hiding this comment

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

huge work done here! :)
the update logic is not the simplest between isLabelSyncedWithName, overrides, etc. I think it would be better protected if covered by integration tests.
We already have integration tests for custom object renaming (rename-custom-object.integration-spec.ts), I think we should add more !


const handleError = (error: unknown) => {
// eslint-disable-next-line no-console
console.error(error);
Copy link
Collaborator

Choose a reason for hiding this comment

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

was this intended or part of local debug?

Copy link
Member

Choose a reason for hiding this comment

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

I wasn't sure but it was already there so I left it. Might not be needed indeed...

fillNamePluralFromLabelPlural(labelPlural);
fillNameSingularFromLabelSingular(labelSingular);
}
onNewDirtyField?.();

// Server-side side effect when isLabelSyncedWithName is changed
if (
Copy link
Collaborator

Choose a reason for hiding this comment

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

@FelixMalfait did you create a ticket for this one in the end ? or was it fixed ?

return this.objectMetadataService.resolveOverridableString(
objectMetadata,
'icon',
context.req.headers['x-locale'],
Copy link
Collaborator

Choose a reason for hiding this comment

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

does icon also vary depending on locale ? I don't see why it should

Copy link
Member

Choose a reason for hiding this comment

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

Yes indeed the icon should remain the same for all locales (cf next PR)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

We should be able to change the label of standard objects
4 participants