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

[Spreadsheet] fix isValidAlias() #18567

Merged
merged 3 commits into from
Dec 23, 2024
Merged

[Spreadsheet] fix isValidAlias() #18567

merged 3 commits into from
Dec 23, 2024

Conversation

mwganson
Copy link
Contributor

Alias validation was broken in one of my previous PR's. This fixes the issue introduced in #17315

Incorrectly, I previously thought the re matching was only looking for valid address strings, but it was also looking for invalid characters, such as spaces or quote characters in the alias name. The relevant re search was moved back into isValidAlias().

I also added an alias named with spaces and another with a leading single quote character to the unit tests. I am not entirely sure these tests are actually being run. @chennes can you have a look? I tried adding one of the invalid names to the valid names list, but it did not trigger the expected test failure. Instead, all tests passed when that one should have failed. Method to run was from the linux terminal in the build folder:

./bin/FreeCAD -t0

As a sidenote there appear to be lots of exceptions getting thrown in the unit tests, but they aren't triggering test failures.

What lead me back here was this post on the forum:

https://forum.freecad.org/viewtopic.php?t=92952

It seems when copying the contents of a cell that has a string in it the leading singlequote is also copied, and when this is simply pasted into the Alias QLineEdit widget it can lead to a crash. This should resolve that crash because now the alias name with the leading quote is flagged as an invalid alias and so setAlias() isn't being called on it. I considered automatically removing the leading single quote, but then upon investigating found this bug that this PR is fixing.

@@ -150,13 +150,19 @@ bool PropertySheet::isValidCellAddressName(const std::string& candidate)

bool PropertySheet::isValidAlias(const std::string& candidate)
{
/* Ensure it only contains allowed characters */
static const boost::regex gen("^[A-Za-z][_A-Za-z0-9]*$");
Copy link
Member

Choose a reason for hiding this comment

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

This seems to eliminate the possibility of e.g. accented characters -- is that a concern? Do we support them now? And is there another method elsewhere in FC that handles this already?

Copy link
Member

@adrianinsaval adrianinsaval Dec 20, 2024

Choose a reason for hiding this comment

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

maybe

bool ExpressionParser::isTokenAnIndentifier(const std::string & str)

Since what we really want to check here is that it is a valid python/expression identifier

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Accented characters do not work for dynamic property names. I tried creating a dynamic property named 'Bä' and it failed.

obj.addProperty("App::PropertyFloat",'Bä')
Traceback (most recent call last):
File "", line 1, in
NameError: {'sclassname': 'class Base::NameError', 'sErrMsg': "Invalid property name 'Bä'", 'sfile': 'C:\bld\freecad_1734380974498\work\src\App\DynamicProperty.cpp', 'iline': 202, 'sfunction': 'class App::Property *__cdecl App::DynamicProperty::addDynamicProperty(class App::PropertyContainer &,const char *,const char *,const char *,const char *,short,bool,bool)', 'swhat': "Invalid property name 'Bä'", 'btranslatable': False, 'breported': False}

But note that Bä is a valid python identifier if you try to create a variable:

Bä = 42

42

Note also that the expression engine can handle this as an alias name. IMO, if it works with the expression engine, is a valid python identifier, doesn't conflict with a reserved unit name, could not be used as a cell address, and is not already in use, then it should be allowed. But it was never my intention with this or the previous PR to change alias validation.

Copy link
Contributor

Choose a reason for hiding this comment

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

Unfortunately, it appears that there's an assumption that identifiers will be represented by a subset of ASCII. The function performing validation is Base::Tools::getIdentifier() which uses comparisons to ASCII numeric values.

Copy link
Member

Choose a reason for hiding this comment

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

At any rate, the problem with the new structure is that we now have two copies of the exact same regex, something like ten lines apart. So setting aside the issue that we're being overly restrictive with our valid identifier checks (it turns out that's always been true), we have a new bit of code duplication. I think to fix the alias validation a bit more refactoring is required, to eliminate the doubled-up regex.

Copy link
Member

Choose a reason for hiding this comment

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

getIdentifier is not just validation though, it replaces characters that it doesn't consider valid, so maybe it's not appropiate for use here.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I think the re might not be needed in the cell address name validation. I will test and see if still works properly without it. If not, then it can be moved to a new function called by both of the others.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I removed the duplicate re from the cell address name validation function and it passed the c++ unit tests (and the other unit tests). The other re that remains will filter out unsupported characters for cell address names. The reason both were in there is a holdover from before the functions were separated.

Modifying permissiveness is beyond the scope of this PR. I just want to fix what is currently broken. If somebody wants to relax the restrictions, that can be done in a separate PR. But what is really needed is a bit of a bigger project -- a centralized validator that can be used to validate all user-generated names (aliases, constraints, and dynamic properties). This would make for a more consistent user experience. The same validator should work for all of these identifier types, but there might need to be additional restrictions handled locally in the spreadsheet or sketcher module, for example, "A1" is a good name for a constraint, but doesn't work well for a spreadsheet alias because it is also a valid cell address name. There might be similar names that work as aliases, but that create conflicts in sketcher, not sure on that. If a new validator is made, then it should also be accessible in python so workbenches and macros that create new aliases, constraints, or dynamic properties could also make use of it.

@adrianinsaval
Copy link
Member

Method to run was from the linux terminal in the build folder:

./bin/FreeCAD -t0

the c++ unit tests are compiled to a separate binary, you have to run

./tests/Tests_run

@adrianinsaval adrianinsaval merged commit c0210c2 into FreeCAD:main Dec 23, 2024
9 checks passed
@mwganson mwganson deleted the aliases branch December 24, 2024 17:33
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Mod: Spreadsheet Related to the Spreadsheet Workbench
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants