-
Notifications
You must be signed in to change notification settings - Fork 3k
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
Conversation
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.
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)) |
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.
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
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.
Would also allowing to change the icon make sense?
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 great point! Icon is similar to description
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.
Editing icon and description is now possible, but the sync switch should at the very least have a different label.
...ty-server/src/engine/metadata-modules/object-metadata/hooks/before-update-one-object.hook.ts
Outdated
Show resolved
Hide resolved
...ty-server/src/engine/metadata-modules/object-metadata/hooks/before-update-one-object.hook.ts
Outdated
Show resolved
Hide resolved
); | ||
} | ||
} | ||
} else { |
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.
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
...rc/modules/settings/data-model/objects/forms/components/SettingsDataModelObjectAboutForm.tsx
Outdated
Show resolved
Hide resolved
...rc/modules/settings/data-model/objects/forms/components/SettingsDataModelObjectAboutForm.tsx
Outdated
Show resolved
Hide resolved
...ty-server/src/engine/metadata-modules/object-metadata/hooks/before-update-one-object.hook.ts
Outdated
Show resolved
Hide resolved
if (value === true) { | ||
if ( | ||
value === true && | ||
(!objectMetadataItem || objectMetadataItem?.isCustom) |
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.
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
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.
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)
...les/settings/data-model/object-details/components/SettingsUpdateDataModelObjectAboutForm.tsx
Outdated
Show resolved
Hide resolved
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 |
...ty-server/src/engine/metadata-modules/object-metadata/hooks/before-update-one-object.hook.ts
Outdated
Show resolved
Hide resolved
|
||
const standardObjectMap = new Map<string, WorkspaceEntityMetadataArgs>(); | ||
|
||
standardObjectMetadataDefinitions.forEach((entity) => { |
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 does not feel right, this will perform a side effect when importing the utils
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 get why you wouldn't want side effects but I think this is efficient from a performance perspective. How would you do this?
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.
See #10922 (comment) ; it's my comment that Louis into this direction. I didn't like the previous solution
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.
Digging a bit more:
- I'm also not sure what is specific to standard vs custom. It feels that they should be handled the same way
- 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
- 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
- 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)
objectMetadata.standardId | ||
) { | ||
const standardObjectMetadata = | ||
getStandardObjectMetadataFromDefinition( |
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.
TBH, I would load the metadata from the cache here and filter on standards (actually we only need to filter with standardId)
Signed-off-by: AFCMS <[email protected]>
Do not send the label changes when switching `isLabelSyncedWithName` to false
033df7f
to
6838a85
Compare
@greptileai review |
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 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 tofieldMetadata
andobjectMetadata
tables via migration 1742736630054 to store overridden labels/descriptions - Modified
SettingsDataModelObjectAboutForm
andSettingsDataModelFieldIconLabelForm
to allow label editing whenisLabelSyncedWithName
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
toresolveOverridableString
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
if ( | ||
formValues.isLabelSyncedWithName === false && | ||
isLabelSyncedWithName === true | ||
) { |
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 check for when isLabelSyncedWithName changes from false to true - could leave labels in an inconsistent state
disabled={ | ||
disabled || (isLabelSyncedWithName ?? false) | ||
disabled || | ||
(isLabelSyncedWithName ?? false) || | ||
!fieldMetadataItem?.isCustom | ||
} |
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: API name field's disabled state should be simplified to avoid potential bugs. Consider extracting this logic into a separate function for better maintainability.
if ( | ||
value === true && | ||
(!objectMetadataItem || objectMetadataItem?.isCustom) | ||
) { |
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: Redundant condition check - !objectMetadataItem || objectMetadataItem?.isCustom
. The optional chaining is unnecessary since we already check if objectMetadataItem is undefined
if ( | |
value === true && | |
(!objectMetadataItem || objectMetadataItem?.isCustom) | |
) { | |
if ( | |
value === true && | |
(!objectMetadataItem || objectMetadataItem.isCustom) | |
) { |
packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.entity.ts
Outdated
Show resolved
Hide resolved
packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.entity.ts
Outdated
Show resolved
Hide resolved
packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.service.ts
Show resolved
Hide resolved
packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.entity.ts
Outdated
Show resolved
Hide resolved
const isUpdatingLabelsWhenSynced = | ||
(instance.update.labelSingular || instance.update.labelPlural) && | ||
objectMetadata.isLabelSyncedWithName && | ||
instance.update.isLabelSyncedWithName !== false && | ||
(instance.update.labelSingular !== objectMetadata.labelSingular || | ||
instance.update.labelPlural !== objectMetadata.labelPlural); |
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: This condition may miss edge cases where labelSingular/Plural are set to empty strings or other falsy values. Consider using isDefined() here for consistency.
...ty-server/src/engine/metadata-modules/object-metadata/hooks/before-update-one-object.hook.ts
Show resolved
Hide resolved
...ty-server/src/engine/metadata-modules/object-metadata/hooks/before-update-one-object.hook.ts
Show resolved
Hide resolved
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.
Will resume review when you ping me @FelixMalfait !
.../modules/settings/data-model/fields/forms/components/SettingsDataModelFieldIconLabelForm.tsx
Show resolved
Hide resolved
if ( | ||
formValues.isLabelSyncedWithName === false && | ||
isLabelSyncedWithName === true | ||
) { |
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.
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 ( |
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.
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
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.
@FelixMalfait did you create a ticket for this one in the end ? or was it fixed ?
packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.resolver.ts
Show resolved
Hide resolved
packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.service.ts
Outdated
Show resolved
Hide resolved
fieldMetadata: FieldMetadataEntity, | ||
): UpdateOneInputType<T> { | ||
const update: StandardFieldUpdate = {}; | ||
const allowedFields = ['isActive', 'isLabelSyncedWithName']; |
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.
updatableFields
? fieldsAllowedForUpdate
? (trying to find a more precise naming)
update: StandardFieldUpdate, | ||
): void { | ||
if (!isDefined(instance.update.description)) { | ||
return; |
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.
can we not update to an empty description ? Maybe we jsut want to evaluate
if(!('description' in instance.update))
?
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
(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
andObjectStandardOverridesDTO
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} |
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: 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 }); |
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: Calling reset with keepValues: true after a successful update may prevent the form from reflecting server-side changes
if (!objectMetadataItem.isCustom) { | ||
const { | ||
nameSingular: _, | ||
namePlural: __, | ||
...payloadWithoutNames | ||
} = updatePayload; | ||
|
||
return updateOneObjectMetadataItem({ | ||
idToUpdate: objectMetadataItem.id, | ||
updatePayload: payloadWithoutNames, | ||
}); | ||
} |
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: Destructuring with _ and __ is not a clear naming convention. Consider using more descriptive names like 'unusedSingular' and 'unusedPlural'
{ | ||
keepDirty: true, | ||
}, |
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: Keeping dirty state during reset could lead to unexpected form behavior and validation issues
packages/twenty-server/src/engine/metadata-modules/field-metadata/dtos/field-metadata.dto.ts
Outdated
Show resolved
Hide resolved
packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.module.ts
Show resolved
Hide resolved
…request
…request
0fd497c
to
fc02d6a
Compare
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.
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); |
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.
was this intended or part of local debug?
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 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 ( |
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.
@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'], |
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.
does icon also vary depending on locale ? I don't see why it should
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 indeed the icon should remain the same for all locales (cf next PR)
Fixes #10793
This PR is a work in progress.
Still left to fix:
isLabelSyncedWithName
change. Could be an easy fix.SettingsDataModelObjectAboutForm
have adisableEdition
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?What should work: