Mar 21, 2026
- Fixed error recovery dropping subsequent clauses - When using
ignoreParseErrors: true, if a clause failed to parse, all subsequent clauses (ORDER BY, LIMIT, OFFSET, etc.) were silently dropped. The parser now correctly synchronizes forward and continues parsing remaining clauses. - Standardized
NULLvalue casing in array expressions -NULLvalues insideIN/NOT IN/INCLUDES/EXCLUDESarrays now use uppercase'NULL'in the parsedvaluefield, consistent with scalarNULLcomparisons. The composer still outputs lowercasenullin SOQL strings. If you were comparingcondition.valueitems with a case-sensitive=== 'null'check, update to=== 'NULL'. - Removed incorrect
DOTtoken name alias -tokenTypeName(TokenKind.DOT)now returns'DOT'instead of the incorrect'DECIMAL'. This only affects consumers using the low-leveltokenTypeNameAPI.
- Removed singleton parser pattern —
parseQuerynow creates a fresh parser instance per call, eliminating shared mutable state and making the parser safe for concurrent use in worker threads.
Mar 14, 2026
🚀🚀🚀🚀🚀 Performed a complete re-write using a hand-rolled parser, complements of Claude Code which did 100% of the work.
The new parser is ~4,600x faster and has a 75% smaller bundle size since there are no longer any external dependencies.
| Old Parser (Chevrotain) | New Parser (Hand-rolled) | |
|---|---|---|
| Parse time per iteration | 3,224 ms | 0.7 ms |
| Bundle size (ESM, minified) | 194 KB | 48 KB |
| External dependencies | 2 (chevrotain, lodash) | 0 |
In addition, we re-wrote the CLI to drop all 3rd party dependencies since the CLI is very simple and doesn't need a library to manage it.
This library no longer exports some of the types used by the previous parser, shown below. These types were only exported as they were used in some public APIs.
This is very unlikely to impact most users, and if it does, the required changes should be very minimal.
export type { CstNode, CstParser, ILexingError, IRecognitionException } from 'chevrotain';Dec 15, 2025
- Fix indent for query format
Nov 27, 2025
- Add support for
mruinUSING SCOPE. (#268)- Thank you @jimmyhogerty
Aug 18, 2025
- Ensure all types are emitted during the build process (#266)
- Some types were missing from the build process which caused build errors for some typescript users
July 8, 2025
- Ensure types are referenced in exports. (#263)
July 7, 2025
- Updated build scripts to ensure esm build works correctly. (#259)
May 8, 2025
- Updated build tooling and dependencies, no externally facing changes.
June 16, 2024
- Add support for negative currency values (#244)
June 16, 2024
- Package has been moved to
@jetstreamappnpm organization
Jan 18, 2024
- Nested NOT negation WHERE clauses were not properly formed (#242)
Jan 13, 2024
💥 Breaking Changes
Fixed a bug where with typescript types to properly represent that WhereClause can have a null value for left in the case of a negation operator.
This was always the case, but prior to enabling strict typescript types, this went under the radar.
For Typescript consumers that have strict null checks enabled, they may need to make code changes depending on usage.
💥 Did not publish correct assets - package was marked as deprecated on npm.
Jan 13, 2024
Revert accidental breaking change to types. WhereClause left can have null in the negation case, but the types did not represent this.
Updating types to match reality is a breaking change for consumers, so worked around issue and will publish version 5 with breaking change.
💥 Use 4.10.1, this version was marked as deprecated because of accidental breaking change with is reverted in 4.10.1
Jan 13, 2024
- Fixed where clause's that have a field name beginning with
Inpreceded by theNOToperator. These were parsed asNOT INinstead ofNOTfollowed by a field name, example:NOT Invoice__c - Fixed queries that have two consecutive
NOToperators (#237) - Enabled Typescript strict mode and made a number of minor fixes related to this.
- When using
getFieldwhich returnFieldFunctionExpressionwill now always return an empty array even if no parameters are provided.
July 24, 2023
Ensure getFlattenedFields does not throw exception if query does not have fields property.
May 29, 2023
Fixed bug with composeQuery when some of the WHERE clause values were not strings.
May 23, 2023
Upgraded Chevrotain to version 10. Chevrotain dropped support for older browsers, but since this library has a build step, the target output from the processed build should remain in the same target format that was previously available.
May 22, 2023
- Included
srcfolder in npm package
May 22, 2023
- Fix output files to ensure sourcemaps are included in npm package for esm_build. (#227)
- There was a typo in one of the output paths.
April 9, 2023
- Fix output files to ensure sourcemaps are included in npm package. (#227)
April 9, 2023
- Fix
isAggregateFn(#228)- Date functions, such as
HOUR_IN_DAY(CreatedDate)did not properly have theisAggregateFnproperty set to true for the field. - As a result,
getFlattenedFieldswould produce incorrect results for these fields if they were not aliased.
- Date functions, such as
- Upgraded webpack, migrated from Travis CI to Github Actions
- Update dependencies to resolve reported security issues
- Update build process to fix invalid sourcemap mapping (#227)
- Removed non-minified version of application which had incorrect path to sourcemaps
- Ensure webpack generates a sourcemap for the minified code output
January 23rd, 2023
The repository was moved from paustint to jetstreamapp. No code changes.
October 6, 2022
- Ensure the
literalTypeis populated on subqueries that are part of a WHERE clause.
July 17, 2022
- Ensure boolean return from some utility functions.
- Re-designed the documentation website.
July 7, 2022
- Converted test framework from mocha to jest.
- Fixed bug where non-string values passed in to a where clause would throw an exception. (#121)
- Fixed bug where in some cases a soql query would be improperly formed if the operator was an array type, such as
IN, but the value and literal types were not arrays. (#107) - Added additional test-cases for stand-alone compose functions to test non-standard usage.
June 21, 2022
- Added support for accessLevel parameters in a
WITHclause (#193)- Examples:
SELECT Id FROM Account WITH USER_MODESELECT Id FROM Account WITH SYSTEM_MODE
- Thank you @ghingis
- Examples:
- Patch release - changed property from
accessLeveltowithAccessLevel
June 11, 2022
- Fixed parser error where
'mi'and'km'could not be used in a WHERE clause because they were being parsed as a GeoLocationUnit (#188)- Thank you @divijklenty for reporting this.
March 11, 2022
- Add support for partial parse and compose #182
- Added support for parsing and composing partial queries. When parsing, the new option
allowPartialQueryenables this functionality. - Added a third argument for
formatQuery, allowingParseQueryConfigoptions to be provided. - Some types on the
Queryinterface were made optional to support partial queries - Updated CLI to include additional commands
- Added support for parsing and composing partial queries. When parsing, the new option
September 19, 2021
- Added cli support, check out the readme for usage details
August 1, 2021
#160 - getFlattenedFields() Did not return correct results if a normal field used an alias, such as SELECT Count(Id), Name account_name FROM Account GROUP BY Name
June 18, 2021
#157 - getFlattenedFields() Did not return correct results if the aggregate function was nested in another function, such as FORMAT(MAX(CreatedDate)).
This bug only applied if there was not a field alias defined.
June 8, 2021
#155 - Apex bind variable support is improved to allow parsing of more complex Apex.
Review test cases 112 - 117 for examples of supported apex bind variables.
June 6, 2021
#153 - A new parser option has been added named ignoreParseErrors, which will remove invalid parts of a query if there are parsing errors.
The general structure of the query must be valid and the SELECT and WHERE clauses must both be valid, but any other clause may be removed from the parsed output if there are errors parsing the query and ignoreParseErrors is set to true.
This option has been added to the documentation application.
April 13, 20201
💥 Breaking Changes 💥
Release 4.x has changed the way the groupBy and having clauses are parsed. (#149)
Previously, the groupBy clause only allowed multiple entries for fields, but not functions.
The groupBy and orderBy are now always returned as arrays from parsed queries to normalize the returned data structure.
For backwards compatibility, a single groupBy or orderBy object is allowed to be passed in to composeQuery(), but a parsed query will always return an array.
The Query object now has
- A list of group by clauses (a single groupBy clause is allowed if you build the data structure yourself)
- A Having clause (this was previously nested in the groupBy clause)
- A list of orderBy clauses (a single orderBy clause is allowed if you build the data structure yourself)
-groupBy?: GroupByClause;
+groupBy?: GroupByClause | GroupByClause[]; // a parsed query will always be undefined or an array
+having?: HavingClause;
orderBy?: OrderByClause | OrderByClause[]; // a parsed query will always be undefined or an arrayEach groupBy clause
- No longer has a nested having clause
- Is an object with a single
fieldorfnproperty
type GroupByClause = GroupByFieldClause | GroupByFnClause;
-interface GroupByOptionalFieldsClause {
- having?: HavingClause;
-}
-interface GroupByFieldClause extends GroupByOptionalFieldsClause {
+interface GroupByFieldClause {
- field: string | string[];
+ field: string;
}
-interface GroupByFnClause extends GroupByOptionalFieldsClause {
+interface GroupByFnClause {
fn: FunctionExp;
}Here are a few examples of how the groupBy is parsed or expected when composing a query:
SELECT UserId, CALENDAR_MONTH(LoginTime) month FROM LoginHistory WHERE NetworkId != NULL GROUP BY UserId, CALENDAR_MONTH(LoginTime)
{
fields: [
{
type: 'Field',
field: 'UserId',
},
{
type: 'FieldFunctionExpression',
functionName: 'CALENDAR_MONTH',
rawValue: 'CALENDAR_MONTH(LoginTime)',
parameters: ['LoginTime'],
alias: 'month',
},
],
sObject: 'LoginHistory',
where: {
left: {
field: 'NetworkId',
operator: '!=',
literalType: 'NULL',
value: 'NULL',
},
},
groupBy: [
{ field: 'UserId' },
{
fn: {
functionName: 'CALENDAR_MONTH',
rawValue: 'CALENDAR_MONTH(LoginTime)',
parameters: ['LoginTime'],
},
},
],
}SELECT ProductCode FROM Product2 GROUP BY ProductCode HAVING COUNT(Id) > 1 ORDER BY COUNT(Id) DESC
{
fields: [{ type: 'Field', field: 'ProductCode' }],
sObject: 'Product2',
groupBy: [{
field: 'ProductCode',
}],
having: {
left: {
operator: '>',
value: '1',
literalType: 'INTEGER',
fn: { rawValue: 'COUNT(Id)', functionName: 'COUNT', parameters: ['Id'] },
},
},
orderBy: [{
fn: { rawValue: 'COUNT(Id)', functionName: 'COUNT', parameters: ['Id'] },
order: 'DESC',
}],
}SELECT SBQQ__Product__r.Name foo, SBQQ__Quote__c foo1 FROM SBQQ__Quoteline__c GROUP BY SBQQ__Quote__c, SBQQ__Product__r.Name
{
fields: [
{
type: 'FieldRelationship',
field: 'Name',
relationships: ['SBQQ__Product__r'],
rawValue: 'SBQQ__Product__r.Name',
alias: 'foo',
},
{
type: 'Field',
field: 'SBQQ__Quote__c',
alias: 'foo1',
},
],
sObject: 'SBQQ__Quoteline__c',
groupBy: [{ field: 'SBQQ__Quote__c' }, { field: 'SBQQ__Product__r.Name' }],
}March 27, 2021
A number of improvements to the formatter have been made with this release.
- The formatter option
whereClauseOperatorsIndentedhas been deprecated and will always be applied. - A new boolean formatter option named
newLineAfterKeywordshas been added and will ensure that there is always a new line after any keyword. (#137) TYPEOFfields will now always be included on their own line be default, or will span multiple lines, split by keywords ifnewLineAfterKeywordsis set to true. (#135)
SELECT Id, TYPEOF What WHEN Account THEN Phone, NumberOfEmployees WHEN Opportunity THEN Amount, CloseDate ELSE Name, Email END, Name FROM Event
formatOptions: { newLineAfterKeywords: true, fieldMaxLineLength: 1 },
SELECT
Id,
TYPEOF What
WHEN
Account
THEN
Phone, NumberOfEmployees
WHEN
Opportunity
THEN
Amount, CloseDate
ELSE
Name, Email
END,
Name
FROM
EventMarch 27, 2021
- Added support for the
FIELDS()function
March 6, 2021
- Date functions were not properly parsed when used in order by clauses. (#139)
- Modified names of functions / types (internal)
- Removed improper import of
isStringfrom node utils
Changes also released to 2.5.6
January 7, 20201
getFlattenedFieldsdid not properly handle the alias for an aggregate function within an aggregate query. (#131)
October 14, 2020
🔥 Breaking Changes 🔥
This version changes the WHERE clause structure when using the NOT operator t0 fix issue #122, and has implemented stricter type definitions.
The NOT operator is now treated as a LogicalOperator and will be set in the operator field between left and right.
In cases where this is populated, the preceding left condition will either be set to null or will at most have the openParens field populated.
The logicalPrefix property has been removed from Condition.
Example of the change in structure for queries using NOT - SELECT Id FROM Account WHERE NOT Id = '2'
{
"fields": [
{
"type": "Field",
"field": "Id"
}
],
"sObject": "Account",
"where": {
- "left": {
- "logicalPrefix": "NOT",
- "field": "Id",
- "operator": "=",
- "value": "'2'",
- "literalType": "STRING"
- }
+ "left": null
+ "operator": "NOT",
+ "right": {
+ "left": {
+ "field": "Id",
+ "operator": "=",
+ "value": "'2'",
+ "literalType": "STRING"
+ }
}
}If you are using Typescript in strict mode, you may encounter some breaking changes to your types depending on how you pre-checked for the presence of fields.
Field and FieldRelationship are now made up of two types, one with and one without alias.
Condition is now made up of multiple individual interfaces that represent different data types based on what data is populated.
OrderByClause is now made up of multiple individual interfaces that represent different data types based on what data is populated.
GroupByClause is now made up of multiple individual interfaces that represent different data types based on what data is populated.
HavingClause is now made up of multiple individual interfaces that represent different data types based on what data is populated.
Previously you could have just done null/undefined checks in Typescript strict mode.
Now, to avoid using the any type, you can use the newly introduced utility methods that provide type detection and type narrowing.
hasAlias()isFieldSubquery()isGroupByField()isGroupByFn()isHavingClauseWithRightCondition()bisNegationCondition()isOrderByField()isOrderByFn()isString()isSubquery()isValueCondition()isValueFunctionCondition()isValueQueryCondition()isValueWithDateLiteralCondition()isValueWithDateNLiteralCondition()isWhereClauseWithRightCondition()isWhereOrHavingClauseWithRightCondition()
Here is a summary of the core changes, view the Readme for the comprehensive types.
export type FieldType =
| Field
+ | FieldWithAlias
| FieldFunctionExpression
| FieldRelationship
+ | FieldRelationshipWithAlias
| FieldSubquery
| FieldTypeOf;-export interface WhereClause {
- left: Condition & ValueQuery;
- right?: WhereClause;
- operator?: LogicalOperator;
-}
+export type WhereClause = WhereClauseWithoutOperator | WhereClauseWithRightCondition;
-export interface Condition {
- openParen?: number;
- closeParen?: number;
- logicalPrefix?: LogicalPrefix;
- field?: string;
- fn?: FunctionExp;
- operator: Operator;
- value?: string | string[];
- literalType?: LiteralType | LiteralType[]; // If populated with STRING on compose, the value(s) will be wrapped in "'" if they are not already. - All other values ignored
- dateLiteralVariable?: number | number[]; // not required for compose, will be populated if SOQL is parsed
-}
+export type Condition =
+ | ValueCondition
+ | ValueWithDateLiteralCondition
+ | ValueWithDateNLiteralCondition
+ | ValueFunctionCondition
+ | NegationCondition;-export interface OrderByClause {
- field?: string;
- fn?: FunctionExp;
- order?: OrderByCriterion;
- nulls?: NullsOrder;
-}
+export type OrderByClause = OrderByFieldClause | OrderByFnClause;-export interface GroupByClause {
- field?: string | string[];
- fn?: FunctionExp;
- having?: HavingClause;
-}
+export type GroupByClause = GroupByFieldClause | GroupByFnClause;
-export interface HavingClause {
- left: Condition;
- right?: HavingClause;
- operator?: LogicalOperator;
-}
+export type HavingClause = HavingClauseWithoutOperator | HavingClauseWithRightCondition;Aug 23, 2020
getFlattenedFieldsignorestypeofclauses in query. (#115)
April 12, 2020
getFlattenedFieldsreturns incorrect results if relationship field is grouped and you are grouping for only one field (#113)
April 24, 2020
- Fixed nanoseconds on date (#102)
April 23, 2020
- Added support for dates formatted with nanoseconds, such as
2020-04-15T02:40:03.000+0000. (#102) - Added support for aggregate function in the
ORDER BYclause (#103) - Queries would not be properly composed if an order by had a function and also specified ASC or DESC (#104)
April 23, 2020
- Queries with date functions in a WHERE clause would throw an error when parsing. (#100)
April 3, 2020
- Passing in null or undefined to compose query no longer throws an exception, but instead returns an empty string. (#95)
- Regular fields in a select clause now allow aliases because this is allowed if the field is used as part of a group by clause. (#97)
getFlattenedFields()now considers if a relationship field was used as part of a group by, and if so it returns just the field name instead of the entire field path, as this is how Salesforce will return the records. (#98)
Mar 22, 2020
- Updated dependencies to resolve known security vulnerabilities.
Feb 25, 2020
- Date literals were not properly parsed if they were included as part of a SET within a WHERE clause, such as
WHERE IN (TODAY, LAST_N_DAYS:5).- As part of this change, the
dateLiteralVariableproperty in theConditionwill be an array if a variable date literal is included in a SET where clause. Refer to test cases93through98for examples
- As part of this change, the
Jan 13, 2020
- The
DESCoperator in theORDER BYclause was treated as a case-sensitive field. - The following fields we treated as case-sensitive:
NEXT_N_FISCAL_QUARTERS,LAST_N_FISCAL_QUARTERS,N_FISCAL_QUARTERS_AGO,NEXT_N_FISCAL_YEARS,LAST_N_FISCAL_YEARS,mi,kmonGEOLOCATIONfunctions
- Updated the
DISTANCEfunction to properly be tagged asisAggregateFn=trueif used as a field- This fixed an issue where
getFlattenedFields()would throw an exception
- This fixed an issue where
Jan 4, 2020
- Added logo to README and updated docs.
Dec 2, 2019
- When composing a query, if an empty OrderBy array was provided, the composed query would still include the
ORDER BYclause in the composed query.
Nov 17, 2019
GROUP BYdid not allow multiple fields to be listed, for example:SELECT BillingState, BillingStreet, COUNT(Id) FROM Account GROUP BY BillingState, BillingStreetwould fail to parse.
Nov 6, 2019
DISTANCEandGEOLOCATIONfunctions failed to parse when used in aWHEREclauses andORDER BYclauses.
Oct 28, 2019
- The method signature for
getFlattenedFieldshas changed to allowQuery | Subquery | FieldSubqueryto be passed in. this is not being considered a breaking change because it is fully backwards compatible. - A new helper method
isFieldSubquery(value: any)was added to allow determining if a Field is a FieldSubquery. This is used internally forgetFlattenedFields().
Oct 6, 2019
Version 2.0 brings some significant bundle size and performance improvements. This library now uses Chevrotain instead of antlr4. With this change, everything related to parsing had to be re-written from scratch. Chevrotain uses pure javascript to handle lexing, parsing, and visiting the generated ast/cst as opposed to using a grammar file and generating a javascript parser based on the grammar.
With this change, the data model was reviewed and analyzed, and there are some significant breaking changes to the data structures. Review the 🔥breaking changes🔥 below for a detailed description of each breaking change.
To compare the bundle size, the following small program was written and then compiled using the default configuration of webpack, and the resulting webpack bundle was compared to determine the full size of the library.
Minified, uncompressed:
- Version 1.x: 545kb
- Version 2.0: 207kb
var soqlParser = require('@jetstreamapp/soql-parser-js');
const query = soqlParser.parseQuery(`SELECT Id FROM Account WHERE Id = 'FOO'`);
console.log('query', query);
const soql = soqlParser.composeQuery(query);
console.log('soql', soql);Performance testing was done by iterating the unit tests 60K times, here are the results:
Version 1.x parser
Library import (startup time): 0.8671 milliseconds
Parsing: 58 X 1000 = 58000 iterations.
Duration: 5.7648 seconds
Average of 0.0994 milliseconds per query
Version 2.0 parser
Library import (startup time): 1.3793 milliseconds
Parsing: 87 X 1000 = 87000 iterations.
Duration: 3.6582 seconds
Average of 0.0420 milliseconds per query
- The CLI was removed.
- The
parseQuery()optionshave changed. The only option allowed isallowApexBindVariableswith will allow parsing queries with apex variables. rawValuewill always have a space between parametersGROUPING(Id, BillingCountry)- Some
literalTypevalues may have differing case from prior versions, regardless of the data input.TRUE,FALSE, and all functions except those listed below will always be returned in uppercase, regardless of case of input.- Exceptions:
toLabel,convertTimezone,convertCurrencywill always be in camelCase.
- Added types for
DateLiteralandDateNLiteralvalues. If you are using TypeScript, you can utilize these types.
- A new
LiteralTypevalue was added forAPEX_BIND_VARIABLE. - When composing functions in a where clause or group by clause, the
rawValuewill be preferred (if it exists) (no change here), but if rawValue is not provided, then the function will be composed using thefunctionNameandparameters. - A new
LiteralTypevalue was added forINTEGER_WITH_CURRENCY_PREFIXandDECIMAL_WITH_CURRENCY_PREFIX. e.x.USD500.01
getComposedField()is deprecated, you should now usegetField().getComposedField()will remain available for backward compatibility.getField()/getComposedField()has the following changes:fnproperty is has been deprecated (but still exists), you should now usefunctionNameinstead.- The
fromproperty has been removed for subqueries. TherelationshipNameis required to be populated to compose a subquery.
- On the FormatOptions interface
fieldMaxLineLenwas renamed tofieldMaxLineLength. - Added support for
usingScope- https://developer.salesforce.com/docs/atlas.en-us.soql_sosl.meta/soql_sosl/sforce_api_calls_soql_select_using_scope.htm?search_text=format() - The
havingclause condition (e.x.left) now uses theConditioninterface instead of having it's own structure.
export interface FormatOptions {
numIndent?: number;
- fieldMaxLineLen?: number;
+ fieldMaxLineLength?: number;
fieldSubqueryParensOnOwnLine?: boolean;
whereClauseOperatorsIndented?: boolean;
logging?: boolean;
}rawValuewill now be included onFieldifobjectPrefixis defined.aliasmay be included onField, if defined.- On
FieldFunctionExpression,fnwas renamed tofunctionName. this was done because all other usages offnwereFunctionExp, but it was a string in this case. - The
parameterstype onFieldFunctionExpressionwas modified to allow an array of varying types. - Removed
fromproperty fromFieldSubquery. havingwas removed fromQueryBaseand now lives as a property onGroupByClause.- On the
Conditionobject,literalTypemay be an array. This will be an array ifvalueis an array and there are variable types within thevalue. For example:WHERE Foo IN ('a', null, 'b')would produceliteralType: ['STRING', 'NULL', 'STRING']. - The
GroupByClausehas the following modifications:fieldis now optional, and will be populated only if the grouping is on a single field.typehas been renamed tofnand will be populated whenCUBEandROLLUPare used.- The
havingclause has been added as a top-level property to theGroupByClauseand will be populated only if ahavingclause is present.
- The
HavingConditioninterface has been removed and now uses the sameConditioninterface that theWhereClauseuses, but withoutvalueQuery.- The parser uses the same code to process both of these, so the
havingclause now supports the exact same capability as awhereclause.
- The parser uses the same code to process both of these, so the
FunctionExphas the following modificationstextwas renamed torawValueto be more consistent with other places in the data model.namewas renamed tofunctionName.parameterwas renamed toparametersand the type was changed to(string | FunctionExp)[]to support nested functions. This will ALWAYS be an array now even if there is only one parameter.fnwas removed, as nested functionParameters are always stored as an entry in theparametersarray.- Added support for
usingScope- https://developer.salesforce.com/docs/atlas.en-us.soql_sosl.meta/soql_sosl/sforce_api_calls_soql_select_using_scope.htm?search_text=format()
export type LiteralType =
| 'STRING'
| 'INTEGER'
| 'DECIMAL'
+ | 'INTEGER_WITH_CURRENCY_PREFIX'
+ | 'DECIMAL_WITH_CURRENCY_PREFIX'
| 'BOOLEAN'
| 'NULL'
| 'DATETIME'
| 'DATE'
| 'DATE_LITERAL'
| 'DATE_N_LITERAL'
+ | 'APEX_BIND_VARIABLE';
+ export type DateLiteral =
+ | 'YESTERDAY'
+ | 'TODAY'
+ | 'TOMORROW'
+ | 'LAST_WEEK'
+ | 'THIS_WEEK'
+ | 'NEXT_WEEK'
+ | 'LAST_MONTH'
+ | 'THIS_MONTH'
+ | 'NEXT_MONTH'
+ | 'LAST_90_DAYS'
+ | 'NEXT_90_DAYS'
+ | 'THIS_QUARTER'
+ | 'LAST_QUARTER'
+ | 'NEXT_QUARTER'
+ | 'THIS_YEAR'
+ | 'LAST_YEAR'
+ | 'NEXT_YEAR'
+ | 'THIS_FISCAL_QUARTER'
+ | 'LAST_FISCAL_QUARTER'
+ | 'NEXT_FISCAL_QUARTER'
+ | 'THIS_FISCAL_YEAR'
+ | 'LAST_FISCAL_YEAR'
+ | 'NEXT_FISCAL_YEAR';
+ export type DateNLiteral =
+ | 'YESTERDAY'
+ | 'NEXT_N_DAYS'
+ | 'LAST_N_DAYS'
+ | 'N_DAYS_AGO'
+ | 'NEXT_N_WEEKS'
+ | 'LAST_N_WEEKS'
+ | 'N_WEEKS_AGO'
+ | 'NEXT_N_MONTHS'
+ | 'LAST_N_MONTHS'
+ | 'N_MONTHS_AGO'
+ | 'NEXT_N_QUARTERS'
+ | 'LAST_N_QUARTERS'
+ | 'N_QUARTERS_AGO'
+ | 'NEXT_N_YEARS'
+ | 'LAST_N_YEARS'
+ | 'N_YEARS_AGO'
+ | 'NEXT_N_FISCAL_QUARTERS'
+ | 'LAST_N_FISCAL_QUARTERS'
+ | 'N_FISCAL_QUARTERS_AGO'
+ | 'NEXT_N_FISCAL_YEARS'
+ | 'LAST_N_FISCAL_YEARS'
+ | 'N_FISCAL_YEARS_AGO';
export interface Field {
type: 'Field';
field: string;
objectPrefix?: string;
+ rawValue?: string;
+ alias?: string;
}
export interface FieldFunctionExpression {
type: 'FieldFunctionExpression';
- fn: string;
+ functionName: string;
- parameters?: string[] | FieldFunctionExpression[];
+ parameters: (string | FieldFunctionExpression)[];
alias?: string;
isAggregateFn?: boolean;
rawValue?: string;
}
export interface FieldRelationship {
type: 'FieldRelationship';
field: string;
relationships: string[];
objectPrefix?: string;
rawValue?: string;
+ alias?: string;
}
export interface FieldSubquery {
type: 'FieldSubquery';
subquery: Subquery;
- from?: string;
}
export interface QueryBase {
fields: FieldType[];
sObjectAlias?: string;
+ usingScope?: string;
where?: WhereClause;
limit?: number;
offset?: number;
groupBy?: GroupByClause;
- having?: HavingClause;
orderBy?: OrderByClause | OrderByClause[];
withDataCategory?: WithDataCategoryClause;
withSecurityEnforced?: boolean;
for?: ForClause;
update?: UpdateClause;
}
export interface WhereClause {
- left: Condition;
+ left: Condition & ValueQuery;
right?: WhereClause;
operator?: LogicalOperator;
}
+ export interface ValueQuery {
+ valueQuery?: Query;
+ }
export interface Condition {
openParen?: number;
closeParen?: number;
logicalPrefix?: LogicalPrefix;
field?: string;
fn?: FunctionExp;
operator: Operator;
value?: string | string[];
- valueQuery?: Query;
- literalType?: LiteralType;
+ literalType?: LiteralType | LiteralType[];
dateLiteralVariable?: number;
}
export interface GroupByClause {
- field: string | string[];
+ field?: string | string[];
- type?: GroupByType;
+ fn?: FunctionExp;
+ having?: HavingClause;
}
export interface HavingClause {
+ left: Condition;
right?: HavingClause;
operator?: LogicalOperator;
}
- export interface HavingCondition {
- openParen?: number;
- closeParen?: number;
- field?: string;
- fn?: FunctionExp;
- operator: string;
- value: string | number;
- }
export interface FunctionExp {
- text?: string;
+ rawValue?: string;
- name?: string;
+ functionName?: string;
alias?: string;
- parameter?: string | string[];
+ parameters?: (string | FunctionExp)[];
isAggregateFn?: boolean;
- fn?: FunctionExp;
}- Queries with multiple consecutive left parens in a where clause were not correctly parsed. (#69)
- Fixed npm reported security vulnerabilities.
- Changed compose methods to public to allow external access (#65)
- Fixed lodash security vulnerability
- Updated README to reflect new changes and other minor changes
- Removed files that accidentally got included with release with update of
release-it
- Updated
Contributing.mdwith more detailed instructions on grammar updates - Added support for
WITH SECURITY_ENFORCED(#61)
- If a field in a query happened to have a function reserved word, such as
Format, then parsing the query failed. (#59)
- Ensured that nothing is logged directly to the console unless logging is enabled
!BREAKING CHANGES!
- Added literal type information to fields to provide additional information about the field type. (#51)
- WHERE clause fields have one of the following types
'STRING' | 'INTEGER' | 'DECIMAL' | 'BOOLEAN' | 'NULL' | 'DATE_LITERAL' | 'DATE_N_LITERAL';stored in the condition. - For date literal fields that have variables,
dateLiteralVariablewill be populated with the value
- WHERE clause fields have one of the following types
- Modified Field data structure to have explicit type information. (#46, #52)
- The data structure for fields has been modified to include specific information about the structure of a given field to ease making sense of a parsed query,
- To aid in creating compose fields, a new helper method is available -
getComposedField(). This takes in a simple data structure (or even a string) and will return the structure needed to compose a query.
- An additional
queryUtilsobject is available with the following functions:function getComposedField(input: string | ComposeFieldInput): FieldTypefunction getFlattenedFields(query: Query, isAggregateResult?: boolean): string[]function isSubquery(query: Query | Subquery): query is Subquery- Look at the README and refer to the unit tests for example usage.