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

Enabling SOQL functions ( closes #163 ) #258

Open
wants to merge 3 commits into
base: master
Choose a base branch
from

Conversation

foxysolutions
Copy link
Contributor

@foxysolutions foxysolutions commented Dec 26, 2019

Enrichment to the fflib Query Factory to allow soql methods (like COUNT(), toLabel, and such) to be specified. Note, they can be added via selectFields( String(s) ) methods, as is illustrated in the provided test methods.


This change is Reviewable

rvandenassum and others added 3 commits December 26, 2019 13:40
…nd re-adding after matched token

* fflib_QueryFactory.cls: Added matching Pattern for all SOQL Functions (COUNT_DISTINCT, toLabel, MIN, MAX, etc.)
* fflib_QueryFactoryTest.cls: Adding test methods to verify correct behavior
Merging apex-common back into personal fork
@foxysolutions foxysolutions changed the title Enabling SOQL functions (#163) Enabling SOQL functions ( closes #163 ) Dec 26, 2019
return soqlFunctionPattern;
} set; }

private String getFieldPath( String fieldName ){
Copy link
Contributor

Choose a reason for hiding this comment

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

By adding this logic inside this method the complexity suddenly becomes very high. I would advise to split this logic into a number of private methods to keep a level of abstraction here in his method.

Copy link
Contributor Author

@foxysolutions foxysolutions Jan 7, 2020

Choose a reason for hiding this comment

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

@wimvelzeboer I do agree with you on the increased complexity and length of the method. I've thought of this before, but in my perspective this will only become more difficult to read, since beforeField and afterField are used throughout the function. The methods should then return a list/map or an innerclass to support. That's why I've put emphasize on the comments.

If you have any suggestion how to split and make sure all data is properly fed back to be used, please let me know!

Copy link
Contributor

@wimvelzeboer wimvelzeboer Mar 12, 2020

Choose a reason for hiding this comment

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

I would maybe split up this method into three parts:
private String getFieldPathForMethod(String fieldName)
private String getFieldPathForRelatedField(String fieldName)
private String getFieldPathForField(String fieldName)

private String getFieldPath( String fieldName )
{
  if (fieldName.contains( '(' )) return getFieldPathForMethod(fieldName);
  if (fieldName.contains( '.' ) ) return getFieldPathForRelatedField(fieldName);
  return getFieldPathForField(fieldName);
}

private getFieldPathForMethod(String methodCall)
{
  Matcher match = this.soqlFunctionPattern.matcher( methodCall );
  if (!match.matches())
  {
    throw new InvalidFieldException( 'The method '+ methodCall + 'does specify a ( function opening, but doesn\'t match the pattern' );
  }
  
  String beforeField = match.group( 1 );
  String fieldName = match.group( 2 );
  String afterField = match.group( 3 );

  // When no fieldName is specified (e.g. COUNT() ), return directly, since nothing to check
  if (String.isBlank( fieldName ))
  {
    return beforeField + afterField;
  }
  return beforeField + getFieldPath(fieldName) + afterField;
}

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.

4 participants