mirror of
https://github.com/sasjs/lint.git
synced 2025-12-11 01:44:36 +00:00
Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b13302a315 | ||
|
|
0790a447f3 | ||
|
|
11182aaaa7 | ||
|
|
7144d0cfe3 | ||
|
|
0caf31b7ff | ||
|
|
020a1e08d0 | ||
|
|
a762dadf37 | ||
|
|
c9fa366130 | ||
|
|
5701064c07 | ||
|
|
cbfa1f40d1 | ||
|
|
d391a4e8fc | ||
|
|
f793eb3a76 | ||
|
|
af2d2c12c1 | ||
|
|
8bfb547427 | ||
|
|
d7721f8e5e | ||
|
|
482ecec150 | ||
|
|
b4ec32b72c | ||
|
|
dcfeb7a641 | ||
|
|
e5780cd69a | ||
|
|
021f36663a |
6
package-lock.json
generated
6
package-lock.json
generated
@@ -4540,9 +4540,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"ts-jest": {
|
"ts-jest": {
|
||||||
"version": "26.5.5",
|
"version": "26.5.6",
|
||||||
"resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-26.5.5.tgz",
|
"resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-26.5.6.tgz",
|
||||||
"integrity": "sha512-7tP4m+silwt1NHqzNRAPjW1BswnAhopTdc2K3HEkRZjF0ZG2F/e/ypVH0xiZIMfItFtD3CX0XFbwPzp9fIEUVg==",
|
"integrity": "sha512-rua+rCP8DxpA8b4DQD/6X2HQS8Zy/xzViVYfEs2OQu68tkCuKLV0Md8pmX55+W24uRIyAsf/BajRfxOs+R2MKA==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"bs-logger": "0.x",
|
"bs-logger": "0.x",
|
||||||
|
|||||||
@@ -41,7 +41,7 @@
|
|||||||
"@types/node": "^15.0.2",
|
"@types/node": "^15.0.2",
|
||||||
"jest": "^26.6.3",
|
"jest": "^26.6.3",
|
||||||
"rimraf": "^3.0.2",
|
"rimraf": "^3.0.2",
|
||||||
"ts-jest": "^26.5.5",
|
"ts-jest": "^26.5.6",
|
||||||
"typescript": "^4.2.4"
|
"typescript": "^4.2.4"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
|||||||
@@ -16,7 +16,8 @@
|
|||||||
"noSpacesInFileNames": true,
|
"noSpacesInFileNames": true,
|
||||||
"noTabIndentation": true,
|
"noTabIndentation": true,
|
||||||
"noTrailingSpaces": true,
|
"noTrailingSpaces": true,
|
||||||
"lineEndings": "lf"
|
"lineEndings": "lf",
|
||||||
|
"strictMacroDefinition": true
|
||||||
},
|
},
|
||||||
"examples": [
|
"examples": [
|
||||||
{
|
{
|
||||||
@@ -31,7 +32,8 @@
|
|||||||
"hasMacroNameInMend": true,
|
"hasMacroNameInMend": true,
|
||||||
"noNestedMacros": true,
|
"noNestedMacros": true,
|
||||||
"hasMacroParentheses": true,
|
"hasMacroParentheses": true,
|
||||||
"lineEndings": "crlf"
|
"lineEndings": "crlf",
|
||||||
|
"strictMacroDefinition": true
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"properties": {
|
"properties": {
|
||||||
@@ -130,6 +132,14 @@
|
|||||||
"description": "Enforces the configured terminating character for each line. Shows a warning when incorrect line endings are present.",
|
"description": "Enforces the configured terminating character for each line. Shows a warning when incorrect line endings are present.",
|
||||||
"default": "lf",
|
"default": "lf",
|
||||||
"examples": ["lf", "crlf"]
|
"examples": ["lf", "crlf"]
|
||||||
|
},
|
||||||
|
"strictMacroDefinition": {
|
||||||
|
"$id": "#/properties/strictMacroDefinition",
|
||||||
|
"type": "boolean",
|
||||||
|
"title": "strictMacroDefinition",
|
||||||
|
"description": "Enforces Macro Definition syntax. Shows a warning when incorrect syntax is used.",
|
||||||
|
"default": true,
|
||||||
|
"examples": [true, false]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,14 +12,14 @@ describe('formatText', () => {
|
|||||||
new LintConfig(getLintConfigModule.DefaultLintConfiguration)
|
new LintConfig(getLintConfigModule.DefaultLintConfiguration)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
const text = `%macro test
|
const text = `%macro test;
|
||||||
%put 'hello';\r\n%mend; `
|
%put 'hello';\r\n%mend; `
|
||||||
|
|
||||||
const expectedOutput = `/**
|
const expectedOutput = `/**
|
||||||
@file
|
@file
|
||||||
@brief <Your brief here>
|
@brief <Your brief here>
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
**/\n%macro test
|
**/\n%macro test;
|
||||||
%put 'hello';\n%mend test;`
|
%put 'hello';\n%mend test;`
|
||||||
|
|
||||||
const output = await formatText(text)
|
const output = await formatText(text)
|
||||||
@@ -38,9 +38,9 @@ describe('formatText', () => {
|
|||||||
})
|
})
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
const text = `%macro test\n %put 'hello';\r\n%mend; `
|
const text = `%macro test;\n %put 'hello';\r\n%mend; `
|
||||||
|
|
||||||
const expectedOutput = `/**\r\n @file\r\n @brief <Your brief here>\r\n <h4> SAS Macros </h4>\r\n**/\r\n%macro test\r\n %put 'hello';\r\n%mend test;`
|
const expectedOutput = `/**\r\n @file\r\n @brief <Your brief here>\r\n <h4> SAS Macros </h4>\r\n**/\r\n%macro test;\r\n %put 'hello';\r\n%mend test;`
|
||||||
|
|
||||||
const output = await formatText(text)
|
const output = await formatText(text)
|
||||||
|
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ describe('hasMacroNameInMend - test', () => {
|
|||||||
it('should return an array with a diagnostic for each macro missing an %mend statement', () => {
|
it('should return an array with a diagnostic for each macro missing an %mend statement', () => {
|
||||||
const content = `%macro somemacro;
|
const content = `%macro somemacro;
|
||||||
%put &sysmacroname;
|
%put &sysmacroname;
|
||||||
%macro othermacro`
|
%macro othermacro;`
|
||||||
|
|
||||||
expect(hasMacroNameInMend.test(content)).toEqual([
|
expect(hasMacroNameInMend.test(content)).toEqual([
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ const test = (value: string, config?: LintConfig) => {
|
|||||||
const macros = parseMacros(value, config)
|
const macros = parseMacros(value, config)
|
||||||
const diagnostics: Diagnostic[] = []
|
const diagnostics: Diagnostic[] = []
|
||||||
macros.forEach((macro) => {
|
macros.forEach((macro) => {
|
||||||
if (macro.startLineNumber === null && macro.endLineNumber !== null) {
|
if (macro.startLineNumbers.length === 0 && macro.endLineNumber !== null) {
|
||||||
const endLine = lines[macro.endLineNumber - 1]
|
const endLine = lines[macro.endLineNumber - 1]
|
||||||
diagnostics.push({
|
diagnostics.push({
|
||||||
message: `%mend statement is redundant`,
|
message: `%mend statement is redundant`,
|
||||||
@@ -27,10 +27,13 @@ const test = (value: string, config?: LintConfig) => {
|
|||||||
getColumnNumber(endLine, '%mend') + macro.termination.length,
|
getColumnNumber(endLine, '%mend') + macro.termination.length,
|
||||||
severity: Severity.Warning
|
severity: Severity.Warning
|
||||||
})
|
})
|
||||||
} else if (macro.endLineNumber === null && macro.startLineNumber !== null) {
|
} else if (
|
||||||
|
macro.endLineNumber === null &&
|
||||||
|
macro.startLineNumbers.length !== 0
|
||||||
|
) {
|
||||||
diagnostics.push({
|
diagnostics.push({
|
||||||
message: `Missing %mend statement for macro - ${macro.name}`,
|
message: `Missing %mend statement for macro - ${macro.name}`,
|
||||||
lineNumber: macro.startLineNumber,
|
lineNumber: macro.startLineNumbers![0],
|
||||||
startColumnNumber: 1,
|
startColumnNumber: 1,
|
||||||
endColumnNumber: 1,
|
endColumnNumber: 1,
|
||||||
severity: Severity.Warning
|
severity: Severity.Warning
|
||||||
@@ -73,7 +76,7 @@ const fix = (value: string, config?: LintConfig): string => {
|
|||||||
const macros = parseMacros(value, config)
|
const macros = parseMacros(value, config)
|
||||||
|
|
||||||
macros.forEach((macro) => {
|
macros.forEach((macro) => {
|
||||||
if (macro.startLineNumber === null && macro.endLineNumber !== null) {
|
if (macro.startLineNumbers.length === 0 && macro.endLineNumber !== null) {
|
||||||
// %mend statement is redundant
|
// %mend statement is redundant
|
||||||
const endLine = lines[macro.endLineNumber - 1]
|
const endLine = lines[macro.endLineNumber - 1]
|
||||||
const startColumnNumber = getColumnNumber(endLine, '%mend')
|
const startColumnNumber = getColumnNumber(endLine, '%mend')
|
||||||
@@ -83,7 +86,10 @@ const fix = (value: string, config?: LintConfig): string => {
|
|||||||
const beforeStatement = endLine.slice(0, startColumnNumber - 1)
|
const beforeStatement = endLine.slice(0, startColumnNumber - 1)
|
||||||
const afterStatement = endLine.slice(endColumnNumber)
|
const afterStatement = endLine.slice(endColumnNumber)
|
||||||
lines[macro.endLineNumber - 1] = beforeStatement + afterStatement
|
lines[macro.endLineNumber - 1] = beforeStatement + afterStatement
|
||||||
} else if (macro.endLineNumber === null && macro.startLineNumber !== null) {
|
} else if (
|
||||||
|
macro.endLineNumber === null &&
|
||||||
|
macro.startLineNumbers.length !== 0
|
||||||
|
) {
|
||||||
// missing %mend statement
|
// missing %mend statement
|
||||||
} else if (macro.mismatchedMendMacroName) {
|
} else if (macro.mismatchedMendMacroName) {
|
||||||
// mismatched macro name
|
// mismatched macro name
|
||||||
|
|||||||
@@ -139,18 +139,4 @@ describe('hasMacroParentheses', () => {
|
|||||||
])
|
])
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should return an array with a single diagnostic when a macro definition contains a space', () => {
|
|
||||||
const content = `%macro test ()`
|
|
||||||
|
|
||||||
expect(hasMacroParentheses.test(content)).toEqual([
|
|
||||||
{
|
|
||||||
message: 'Macro definition contains space(s)',
|
|
||||||
lineNumber: 1,
|
|
||||||
startColumnNumber: 8,
|
|
||||||
endColumnNumber: 14,
|
|
||||||
severity: Severity.Warning
|
|
||||||
}
|
|
||||||
])
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -16,36 +16,36 @@ const test = (value: string, config?: LintConfig) => {
|
|||||||
if (!macro.name) {
|
if (!macro.name) {
|
||||||
diagnostics.push({
|
diagnostics.push({
|
||||||
message: 'Macro definition missing name',
|
message: 'Macro definition missing name',
|
||||||
lineNumber: macro.startLineNumber!,
|
lineNumber: macro.startLineNumbers![0],
|
||||||
startColumnNumber: getColumnNumber(macro.declarationLine, '%macro'),
|
startColumnNumber: getColumnNumber(
|
||||||
|
macro.declarationLines![0],
|
||||||
|
'%macro'
|
||||||
|
),
|
||||||
endColumnNumber:
|
endColumnNumber:
|
||||||
getColumnNumber(macro.declarationLine, '%macro') +
|
getColumnNumber(macro.declarationLines![0], '%macro') +
|
||||||
macro.declaration.length,
|
macro.declaration.length,
|
||||||
severity: Severity.Warning
|
severity: Severity.Warning
|
||||||
})
|
})
|
||||||
} else if (!macro.declarationLine.includes('(')) {
|
} else if (!macro.declarationLines.find((dl) => dl.includes('('))) {
|
||||||
|
const macroNameLineIndex = macro.declarationLines.findIndex((dl) =>
|
||||||
|
dl.includes(macro.name)
|
||||||
|
)
|
||||||
diagnostics.push({
|
diagnostics.push({
|
||||||
message,
|
message,
|
||||||
lineNumber: macro.startLineNumber!,
|
lineNumber: macro.startLineNumbers![macroNameLineIndex],
|
||||||
startColumnNumber: getColumnNumber(macro.declarationLine, macro.name),
|
startColumnNumber: getColumnNumber(
|
||||||
|
macro.declarationLines[macroNameLineIndex],
|
||||||
|
macro.name
|
||||||
|
),
|
||||||
endColumnNumber:
|
endColumnNumber:
|
||||||
getColumnNumber(macro.declarationLine, macro.name) +
|
getColumnNumber(
|
||||||
|
macro.declarationLines[macroNameLineIndex],
|
||||||
|
macro.name
|
||||||
|
) +
|
||||||
macro.name.length -
|
macro.name.length -
|
||||||
1,
|
1,
|
||||||
severity: Severity.Warning
|
severity: Severity.Warning
|
||||||
})
|
})
|
||||||
} else if (macro.name !== macro.name.trim()) {
|
|
||||||
diagnostics.push({
|
|
||||||
message: 'Macro definition contains space(s)',
|
|
||||||
lineNumber: macro.startLineNumber!,
|
|
||||||
startColumnNumber: getColumnNumber(macro.declarationLine, macro.name),
|
|
||||||
endColumnNumber:
|
|
||||||
getColumnNumber(macro.declarationLine, macro.name) +
|
|
||||||
macro.name.length -
|
|
||||||
1 +
|
|
||||||
`()`.length,
|
|
||||||
severity: Severity.Warning
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -3,3 +3,4 @@ export { hasMacroNameInMend } from './hasMacroNameInMend'
|
|||||||
export { hasMacroParentheses } from './hasMacroParentheses'
|
export { hasMacroParentheses } from './hasMacroParentheses'
|
||||||
export { lineEndings } from './lineEndings'
|
export { lineEndings } from './lineEndings'
|
||||||
export { noNestedMacros } from './noNestedMacros'
|
export { noNestedMacros } from './noNestedMacros'
|
||||||
|
export { strictMacroDefinition } from './strictMacroDefinition'
|
||||||
|
|||||||
@@ -22,17 +22,17 @@ const test = (value: string, config?: LintConfig) => {
|
|||||||
message: message
|
message: message
|
||||||
.replace('{macro}', macro.name)
|
.replace('{macro}', macro.name)
|
||||||
.replace('{parent}', macro.parentMacro),
|
.replace('{parent}', macro.parentMacro),
|
||||||
lineNumber: macro.startLineNumber as number,
|
lineNumber: macro.startLineNumbers![0] as number,
|
||||||
startColumnNumber: getColumnNumber(
|
startColumnNumber: getColumnNumber(
|
||||||
lines[(macro.startLineNumber as number) - 1],
|
lines[(macro.startLineNumbers![0] as number) - 1],
|
||||||
'%macro'
|
'%macro'
|
||||||
),
|
),
|
||||||
endColumnNumber:
|
endColumnNumber:
|
||||||
getColumnNumber(
|
getColumnNumber(
|
||||||
lines[(macro.startLineNumber as number) - 1],
|
lines[(macro.startLineNumbers![0] as number) - 1],
|
||||||
'%macro'
|
'%macro'
|
||||||
) +
|
) +
|
||||||
lines[(macro.startLineNumber as number) - 1].trim().length -
|
lines[(macro.startLineNumbers![0] as number) - 1].trim().length -
|
||||||
1,
|
1,
|
||||||
severity: Severity.Warning
|
severity: Severity.Warning
|
||||||
})
|
})
|
||||||
|
|||||||
216
src/rules/file/strictMacroDefinition.spec.ts
Normal file
216
src/rules/file/strictMacroDefinition.spec.ts
Normal file
@@ -0,0 +1,216 @@
|
|||||||
|
import { LintConfig, Severity } from '../../types'
|
||||||
|
import { strictMacroDefinition } from './strictMacroDefinition'
|
||||||
|
|
||||||
|
describe('strictMacroDefinition', () => {
|
||||||
|
it('should return an empty array when the content has correct macro definition syntax', () => {
|
||||||
|
const content = '%macro somemacro;'
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([])
|
||||||
|
|
||||||
|
const content2 = '%macro somemacro();'
|
||||||
|
expect(strictMacroDefinition.test(content2)).toEqual([])
|
||||||
|
|
||||||
|
const content3 = '%macro somemacro(var1);'
|
||||||
|
expect(strictMacroDefinition.test(content3)).toEqual([])
|
||||||
|
|
||||||
|
const content4 = '%macro somemacro/minoperator;'
|
||||||
|
expect(strictMacroDefinition.test(content4)).toEqual([])
|
||||||
|
|
||||||
|
const content5 = '%macro somemacro /minoperator;'
|
||||||
|
expect(strictMacroDefinition.test(content5)).toEqual([])
|
||||||
|
|
||||||
|
const content6 = '%macro somemacro(var1, var2)/minoperator;'
|
||||||
|
expect(strictMacroDefinition.test(content6)).toEqual([])
|
||||||
|
|
||||||
|
const content7 =
|
||||||
|
' /* Some Comment */ %macro somemacro(var1, var2) /minoperator ; /* Some Comment */'
|
||||||
|
expect(strictMacroDefinition.test(content7)).toEqual([])
|
||||||
|
|
||||||
|
const content8 =
|
||||||
|
'%macro macroName( arr, arr/* / store source */3 ) /* / store source */;/* / store source */'
|
||||||
|
expect(strictMacroDefinition.test(content8)).toEqual([])
|
||||||
|
|
||||||
|
const content9 = '%macro macroName(var1, var2=with space, var3=);'
|
||||||
|
expect(strictMacroDefinition.test(content9)).toEqual([])
|
||||||
|
|
||||||
|
const content10 = '%macro macroName()/ /* some comment */ store source;'
|
||||||
|
expect(strictMacroDefinition.test(content10)).toEqual([])
|
||||||
|
|
||||||
|
const content11 = '`%macro macroName() /* / store source */;'
|
||||||
|
expect(strictMacroDefinition.test(content11)).toEqual([])
|
||||||
|
|
||||||
|
const content12 =
|
||||||
|
'%macro macroName()/ /* some comment */ store des="some description";'
|
||||||
|
expect(strictMacroDefinition.test(content12)).toEqual([])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single diagnostic when Macro definition has space in param', () => {
|
||||||
|
const content = '%macro somemacro(va r1);'
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Param 'va r1' cannot have space`,
|
||||||
|
lineNumber: 1,
|
||||||
|
startColumnNumber: 18,
|
||||||
|
endColumnNumber: 22,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a two diagnostics when Macro definition has space in params', () => {
|
||||||
|
const content = '%macro somemacro(var1, var 2, v ar3, var4);'
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Param 'var 2' cannot have space`,
|
||||||
|
lineNumber: 1,
|
||||||
|
startColumnNumber: 24,
|
||||||
|
endColumnNumber: 28,
|
||||||
|
severity: Severity.Warning
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: `Param 'v ar3' cannot have space`,
|
||||||
|
lineNumber: 1,
|
||||||
|
startColumnNumber: 31,
|
||||||
|
endColumnNumber: 35,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a two diagnostics when Macro definition has space in params - special case', () => {
|
||||||
|
const content =
|
||||||
|
'%macro macroName( arr, ar r/* / store source */ 3 ) /* / store source */;/* / store source */'
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Param 'ar r 3' cannot have space`,
|
||||||
|
lineNumber: 1,
|
||||||
|
startColumnNumber: 24,
|
||||||
|
endColumnNumber: 49,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single diagnostic when Macro definition has invalid option', () => {
|
||||||
|
const content = '%macro somemacro(var1, var2)/minXoperator;'
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Option 'minXoperator' is not valid`,
|
||||||
|
lineNumber: 1,
|
||||||
|
startColumnNumber: 30,
|
||||||
|
endColumnNumber: 41,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a two diagnostics when Macro definition has invalid options', () => {
|
||||||
|
const content =
|
||||||
|
'%macro somemacro(var1, var2)/ store invalidoption secure ;'
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Option 'invalidoption' is not valid`,
|
||||||
|
lineNumber: 1,
|
||||||
|
startColumnNumber: 39,
|
||||||
|
endColumnNumber: 51,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('multi-content macro declarations', () => {
|
||||||
|
it('should return an empty array when the content has correct macro definition syntax', () => {
|
||||||
|
const content = `%macro mp_ds2cards(base_ds=, tgt_ds=\n ,cards_file="%sysfunc(pathname(work))/cardgen.sas"\n ,maxobs=max\n ,random_sample=NO\n ,showlog=YES\n ,outencoding=\n ,append=NO\n)/*/STORE SOURCE*/;`
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([])
|
||||||
|
|
||||||
|
const content2 = `%macro mm_createapplication(\n tree=/User Folders/sasdemo\n ,name=myApp\n ,ClassIdentifier=mcore\n ,desc=Created by mm_createapplication\n ,params= param1=1
param2=blah\n ,version=\n ,frefin=mm_in\n ,frefout=mm_out\n ,mDebug=1\n );`
|
||||||
|
expect(strictMacroDefinition.test(content2)).toEqual([])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single diagnostic when Macro definition has space in param', () => {
|
||||||
|
const content = `%macro
|
||||||
|
somemacro(va r1);`
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Param 'va r1' cannot have space`,
|
||||||
|
lineNumber: 2,
|
||||||
|
startColumnNumber: 18,
|
||||||
|
endColumnNumber: 22,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a two diagnostics when Macro definition has space in params', () => {
|
||||||
|
const content = `%macro somemacro(
|
||||||
|
var1,
|
||||||
|
var 2,
|
||||||
|
v ar3,
|
||||||
|
var4);`
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Param 'var 2' cannot have space`,
|
||||||
|
lineNumber: 3,
|
||||||
|
startColumnNumber: 7,
|
||||||
|
endColumnNumber: 11,
|
||||||
|
severity: Severity.Warning
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: `Param 'v ar3' cannot have space`,
|
||||||
|
lineNumber: 4,
|
||||||
|
startColumnNumber: 7,
|
||||||
|
endColumnNumber: 11,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a two diagnostics when Macro definition has space in params - special case', () => {
|
||||||
|
const content = `%macro macroName(
|
||||||
|
arr,
|
||||||
|
ar r/* / store source */ 3
|
||||||
|
) /* / store source */;/* / store source */`
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Param 'ar r 3' cannot have space`,
|
||||||
|
lineNumber: 3,
|
||||||
|
startColumnNumber: 7,
|
||||||
|
endColumnNumber: 32,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single diagnostic when Macro definition has invalid option', () => {
|
||||||
|
const content = `%macro somemacro(var1, var2)
|
||||||
|
/minXoperator;`
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Option 'minXoperator' is not valid`,
|
||||||
|
lineNumber: 2,
|
||||||
|
startColumnNumber: 8,
|
||||||
|
endColumnNumber: 19,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a two diagnostics when Macro definition has invalid options', () => {
|
||||||
|
const content = `%macro
|
||||||
|
somemacro(
|
||||||
|
var1, var2
|
||||||
|
)
|
||||||
|
/ store
|
||||||
|
invalidoption
|
||||||
|
secure ;`
|
||||||
|
expect(strictMacroDefinition.test(content)).toEqual([
|
||||||
|
{
|
||||||
|
message: `Option 'invalidoption' is not valid`,
|
||||||
|
lineNumber: 6,
|
||||||
|
startColumnNumber: 16,
|
||||||
|
endColumnNumber: 28,
|
||||||
|
severity: Severity.Warning
|
||||||
|
}
|
||||||
|
])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
169
src/rules/file/strictMacroDefinition.ts
Normal file
169
src/rules/file/strictMacroDefinition.ts
Normal file
@@ -0,0 +1,169 @@
|
|||||||
|
import { Diagnostic, LintConfig, Macro, Severity } from '../../types'
|
||||||
|
import { FileLintRule } from '../../types/LintRule'
|
||||||
|
import { LintRuleType } from '../../types/LintRuleType'
|
||||||
|
import { parseMacros } from '../../utils/parseMacros'
|
||||||
|
|
||||||
|
const name = 'strictMacroDefinition'
|
||||||
|
const description = 'Enforce strictly rules of macro definition syntax.'
|
||||||
|
const message = 'Incorrent Macro Definition Syntax'
|
||||||
|
|
||||||
|
const validOptions = [
|
||||||
|
'CMD',
|
||||||
|
'DES',
|
||||||
|
'MINDELIMITER',
|
||||||
|
'MINOPERATOR',
|
||||||
|
'NOMINOPERATOR',
|
||||||
|
'PARMBUFF',
|
||||||
|
'SECURE',
|
||||||
|
'NOSECURE',
|
||||||
|
'STMT',
|
||||||
|
'SOURCE',
|
||||||
|
'SRC',
|
||||||
|
'STORE'
|
||||||
|
]
|
||||||
|
|
||||||
|
const processParams = (
|
||||||
|
content: string,
|
||||||
|
macro: Macro,
|
||||||
|
diagnostics: Diagnostic[]
|
||||||
|
): string => {
|
||||||
|
const declaration = macro.declaration
|
||||||
|
|
||||||
|
const regExpParams = new RegExp(/(?<=\().*(?=\))/)
|
||||||
|
const regExpParamsResult = regExpParams.exec(declaration)
|
||||||
|
|
||||||
|
let _declaration = declaration
|
||||||
|
if (regExpParamsResult) {
|
||||||
|
const paramsPresent = regExpParamsResult[0]
|
||||||
|
|
||||||
|
const params = paramsPresent.trim().split(',')
|
||||||
|
params.forEach((param) => {
|
||||||
|
const trimedParam = param.split('=')[0].trim()
|
||||||
|
|
||||||
|
let paramLineNumber: number = 1,
|
||||||
|
paramStartIndex: number = 1,
|
||||||
|
paramEndIndex: number = content.length
|
||||||
|
|
||||||
|
if (
|
||||||
|
macro.declarationLines.findIndex(
|
||||||
|
(dl) => dl.indexOf(trimedParam) !== -1
|
||||||
|
) === -1
|
||||||
|
) {
|
||||||
|
const comment = '/\\*(.*?)\\*/'
|
||||||
|
for (let i = 1; i < trimedParam.length; i++) {
|
||||||
|
const paramWithComment =
|
||||||
|
trimedParam.slice(0, i) + comment + trimedParam.slice(i)
|
||||||
|
const regEx = new RegExp(paramWithComment)
|
||||||
|
|
||||||
|
const declarationLineIndex = macro.declarationLines.findIndex(
|
||||||
|
(dl) => !!regEx.exec(dl)
|
||||||
|
)
|
||||||
|
|
||||||
|
if (declarationLineIndex !== -1) {
|
||||||
|
const declarationLine = macro.declarationLines[declarationLineIndex]
|
||||||
|
const partFound = regEx.exec(declarationLine)![0]
|
||||||
|
|
||||||
|
paramLineNumber = macro.startLineNumbers[declarationLineIndex]
|
||||||
|
paramStartIndex = declarationLine.indexOf(partFound)
|
||||||
|
paramEndIndex =
|
||||||
|
declarationLine.indexOf(partFound) + partFound.length
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const declarationLineIndex = macro.declarationLines.findIndex(
|
||||||
|
(dl) => dl.indexOf(trimedParam) !== -1
|
||||||
|
)
|
||||||
|
const declarationLine = macro.declarationLines[declarationLineIndex]
|
||||||
|
paramLineNumber = macro.startLineNumbers[declarationLineIndex]
|
||||||
|
|
||||||
|
paramStartIndex = declarationLine.indexOf(trimedParam)
|
||||||
|
paramEndIndex =
|
||||||
|
declarationLine.indexOf(trimedParam) + trimedParam.length
|
||||||
|
}
|
||||||
|
|
||||||
|
if (trimedParam.includes(' ')) {
|
||||||
|
diagnostics.push({
|
||||||
|
message: `Param '${trimedParam}' cannot have space`,
|
||||||
|
lineNumber: paramLineNumber,
|
||||||
|
startColumnNumber: paramStartIndex + 1,
|
||||||
|
endColumnNumber: paramEndIndex,
|
||||||
|
severity: Severity.Warning
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
_declaration = declaration.split(`(${paramsPresent})`)[1]
|
||||||
|
}
|
||||||
|
return _declaration
|
||||||
|
}
|
||||||
|
|
||||||
|
const processOptions = (
|
||||||
|
_declaration: string,
|
||||||
|
macro: Macro,
|
||||||
|
diagnostics: Diagnostic[]
|
||||||
|
): void => {
|
||||||
|
let optionsPresent = _declaration.split('/')?.[1]?.trim()
|
||||||
|
|
||||||
|
if (optionsPresent) {
|
||||||
|
const regex = new RegExp(/="(.*?)"/, 'g')
|
||||||
|
|
||||||
|
let result = regex.exec(optionsPresent)
|
||||||
|
|
||||||
|
// removing Option's `="..."` part, e.g. des="..."
|
||||||
|
while (result) {
|
||||||
|
optionsPresent =
|
||||||
|
optionsPresent.slice(0, result.index) +
|
||||||
|
optionsPresent.slice(result.index + result[0].length)
|
||||||
|
|
||||||
|
result = regex.exec(optionsPresent)
|
||||||
|
}
|
||||||
|
|
||||||
|
optionsPresent
|
||||||
|
.split(' ')
|
||||||
|
?.filter((o) => !!o)
|
||||||
|
.forEach((option) => {
|
||||||
|
const trimmedOption = option.trim()
|
||||||
|
if (!validOptions.includes(trimmedOption.toUpperCase())) {
|
||||||
|
const declarationLineIndex = macro.declarationLines.findIndex(
|
||||||
|
(dl) => dl.indexOf(trimmedOption) !== -1
|
||||||
|
)
|
||||||
|
const declarationLine = macro.declarationLines[declarationLineIndex]
|
||||||
|
|
||||||
|
diagnostics.push({
|
||||||
|
message: `Option '${trimmedOption}' is not valid`,
|
||||||
|
lineNumber: macro.startLineNumbers[declarationLineIndex],
|
||||||
|
startColumnNumber: declarationLine.indexOf(trimmedOption) + 1,
|
||||||
|
endColumnNumber:
|
||||||
|
declarationLine.indexOf(trimmedOption) + trimmedOption.length,
|
||||||
|
severity: Severity.Warning
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const test = (value: string, config?: LintConfig) => {
|
||||||
|
const diagnostics: Diagnostic[] = []
|
||||||
|
|
||||||
|
const macros = parseMacros(value, config)
|
||||||
|
|
||||||
|
macros.forEach((macro) => {
|
||||||
|
const _declaration = processParams(value, macro, diagnostics)
|
||||||
|
|
||||||
|
processOptions(_declaration, macro, diagnostics)
|
||||||
|
})
|
||||||
|
|
||||||
|
return diagnostics
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lint rule that checks if a line has followed syntax for macro definition
|
||||||
|
*/
|
||||||
|
export const strictMacroDefinition: FileLintRule = {
|
||||||
|
type: LintRuleType.File,
|
||||||
|
name,
|
||||||
|
description,
|
||||||
|
message,
|
||||||
|
test
|
||||||
|
}
|
||||||
@@ -3,7 +3,8 @@ import {
|
|||||||
hasMacroNameInMend,
|
hasMacroNameInMend,
|
||||||
noNestedMacros,
|
noNestedMacros,
|
||||||
hasMacroParentheses,
|
hasMacroParentheses,
|
||||||
lineEndings
|
lineEndings,
|
||||||
|
strictMacroDefinition
|
||||||
} from '../rules/file'
|
} from '../rules/file'
|
||||||
import {
|
import {
|
||||||
indentationMultiple,
|
indentationMultiple,
|
||||||
@@ -90,5 +91,9 @@ export class LintConfig {
|
|||||||
if (json?.hasMacroParentheses) {
|
if (json?.hasMacroParentheses) {
|
||||||
this.fileLintRules.push(hasMacroParentheses)
|
this.fileLintRules.push(hasMacroParentheses)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (json?.strictMacroDefinition) {
|
||||||
|
this.fileLintRules.push(strictMacroDefinition)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
12
src/types/Macro.ts
Normal file
12
src/types/Macro.ts
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
export interface Macro {
|
||||||
|
name: string
|
||||||
|
startLineNumbers: number[]
|
||||||
|
endLineNumber: number | null
|
||||||
|
declarationLines: string[]
|
||||||
|
terminationLine: string
|
||||||
|
declaration: string
|
||||||
|
termination: string
|
||||||
|
parentMacro: string
|
||||||
|
hasMacroNameInMend: boolean
|
||||||
|
mismatchedMendMacroName: string
|
||||||
|
}
|
||||||
@@ -4,3 +4,4 @@ export * from './LintConfig'
|
|||||||
export * from './LintRule'
|
export * from './LintRule'
|
||||||
export * from './LintRuleType'
|
export * from './LintRuleType'
|
||||||
export * from './Severity'
|
export * from './Severity'
|
||||||
|
export * from './Macro'
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import * as fileModule from '@sasjs/utils/file'
|
|||||||
import { LintConfig } from '../types/LintConfig'
|
import { LintConfig } from '../types/LintConfig'
|
||||||
import { getLintConfig } from './getLintConfig'
|
import { getLintConfig } from './getLintConfig'
|
||||||
|
|
||||||
const expectedFileLintRulesCount = 4
|
const expectedFileLintRulesCount = 5
|
||||||
const expectedLineLintRulesCount = 5
|
const expectedLineLintRulesCount = 5
|
||||||
const expectedPathLintRulesCount = 2
|
const expectedPathLintRulesCount = 2
|
||||||
|
|
||||||
|
|||||||
@@ -17,7 +17,8 @@ export const DefaultLintConfiguration = {
|
|||||||
indentationMultiple: 2,
|
indentationMultiple: 2,
|
||||||
hasMacroNameInMend: true,
|
hasMacroNameInMend: true,
|
||||||
noNestedMacros: true,
|
noNestedMacros: true,
|
||||||
hasMacroParentheses: true
|
hasMacroParentheses: true,
|
||||||
|
strictMacroDefinition: true
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -3,103 +3,277 @@ import { parseMacros } from './parseMacros'
|
|||||||
|
|
||||||
describe('parseMacros', () => {
|
describe('parseMacros', () => {
|
||||||
it('should return an array with a single macro', () => {
|
it('should return an array with a single macro', () => {
|
||||||
const text = `%macro test;
|
const text = ` %macro test;\n %put 'hello';\n%mend`
|
||||||
%put 'hello';
|
|
||||||
%mend`
|
|
||||||
|
|
||||||
const macros = parseMacros(text, new LintConfig())
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
expect(macros.length).toEqual(1)
|
expect(macros.length).toEqual(1)
|
||||||
expect(macros).toContainEqual({
|
expect(macros).toContainEqual({
|
||||||
name: 'test',
|
name: 'test',
|
||||||
declarationLine: '%macro test;',
|
declarationLines: [' %macro test;'],
|
||||||
terminationLine: '%mend',
|
terminationLine: '%mend',
|
||||||
declaration: '%macro test',
|
declaration: '%macro test',
|
||||||
termination: '%mend',
|
termination: '%mend',
|
||||||
startLineNumber: 1,
|
startLineNumbers: [1],
|
||||||
endLineNumber: 3,
|
endLineNumber: 3,
|
||||||
parentMacro: '',
|
parentMacro: '',
|
||||||
hasMacroNameInMend: false,
|
hasMacroNameInMend: false,
|
||||||
hasParentheses: false,
|
mismatchedMendMacroName: ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single macro having parameters', () => {
|
||||||
|
const text = `%macro test(var,sum);\n %put 'hello';\n%mend`
|
||||||
|
|
||||||
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
|
expect(macros.length).toEqual(1)
|
||||||
|
expect(macros).toContainEqual({
|
||||||
|
name: 'test',
|
||||||
|
declarationLines: ['%macro test(var,sum);'],
|
||||||
|
terminationLine: '%mend',
|
||||||
|
declaration: '%macro test(var,sum)',
|
||||||
|
termination: '%mend',
|
||||||
|
startLineNumbers: [1],
|
||||||
|
endLineNumber: 3,
|
||||||
|
parentMacro: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
|
mismatchedMendMacroName: ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single macro having PARMBUFF option', () => {
|
||||||
|
const text = `%macro test/parmbuff;\n %put 'hello';\n%mend`
|
||||||
|
|
||||||
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
|
expect(macros.length).toEqual(1)
|
||||||
|
expect(macros).toContainEqual({
|
||||||
|
name: 'test',
|
||||||
|
declarationLines: ['%macro test/parmbuff;'],
|
||||||
|
terminationLine: '%mend',
|
||||||
|
declaration: '%macro test/parmbuff',
|
||||||
|
termination: '%mend',
|
||||||
|
startLineNumbers: [1],
|
||||||
|
endLineNumber: 3,
|
||||||
|
parentMacro: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
|
mismatchedMendMacroName: ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single macro having paramerter & SOURCE option', () => {
|
||||||
|
const text = `/* commentary */ %macro foobar(arg) /store source\n des="This macro does not do much";\n %put 'hello';\n%mend`
|
||||||
|
|
||||||
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
|
expect(macros.length).toEqual(1)
|
||||||
|
expect(macros).toContainEqual({
|
||||||
|
name: 'foobar',
|
||||||
|
declarationLines: [
|
||||||
|
'/* commentary */ %macro foobar(arg) /store source',
|
||||||
|
' des="This macro does not do much";'
|
||||||
|
],
|
||||||
|
terminationLine: '%mend',
|
||||||
|
declaration:
|
||||||
|
'%macro foobar(arg) /store source des="This macro does not do much"',
|
||||||
|
termination: '%mend',
|
||||||
|
startLineNumbers: [1, 2],
|
||||||
|
endLineNumber: 4,
|
||||||
|
parentMacro: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
mismatchedMendMacroName: ''
|
mismatchedMendMacroName: ''
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should return an array with multiple macros', () => {
|
it('should return an array with multiple macros', () => {
|
||||||
const text = `%macro foo;
|
const text = `%macro foo;\n %put 'foo';\n%mend;\n%macro bar();\n %put 'bar';\n%mend bar;`
|
||||||
%put 'foo';
|
|
||||||
%mend;
|
|
||||||
%macro bar();
|
|
||||||
%put 'bar';
|
|
||||||
%mend bar;`
|
|
||||||
|
|
||||||
const macros = parseMacros(text, new LintConfig())
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
expect(macros.length).toEqual(2)
|
expect(macros.length).toEqual(2)
|
||||||
expect(macros).toContainEqual({
|
expect(macros).toContainEqual({
|
||||||
name: 'foo',
|
name: 'foo',
|
||||||
declarationLine: '%macro foo;',
|
declarationLines: ['%macro foo;'],
|
||||||
terminationLine: '%mend;',
|
terminationLine: '%mend;',
|
||||||
declaration: '%macro foo',
|
declaration: '%macro foo',
|
||||||
termination: '%mend',
|
termination: '%mend',
|
||||||
startLineNumber: 1,
|
startLineNumbers: [1],
|
||||||
endLineNumber: 3,
|
endLineNumber: 3,
|
||||||
parentMacro: '',
|
parentMacro: '',
|
||||||
hasMacroNameInMend: false,
|
hasMacroNameInMend: false,
|
||||||
hasParentheses: false,
|
|
||||||
mismatchedMendMacroName: ''
|
mismatchedMendMacroName: ''
|
||||||
})
|
})
|
||||||
expect(macros).toContainEqual({
|
expect(macros).toContainEqual({
|
||||||
name: 'bar',
|
name: 'bar',
|
||||||
declarationLine: '%macro bar();',
|
declarationLines: ['%macro bar();'],
|
||||||
terminationLine: '%mend bar;',
|
terminationLine: '%mend bar;',
|
||||||
declaration: '%macro bar()',
|
declaration: '%macro bar()',
|
||||||
termination: '%mend bar',
|
termination: '%mend bar',
|
||||||
startLineNumber: 4,
|
startLineNumbers: [4],
|
||||||
endLineNumber: 6,
|
endLineNumber: 6,
|
||||||
parentMacro: '',
|
parentMacro: '',
|
||||||
hasMacroNameInMend: true,
|
hasMacroNameInMend: true,
|
||||||
hasParentheses: true,
|
|
||||||
mismatchedMendMacroName: ''
|
mismatchedMendMacroName: ''
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should detect nested macro definitions', () => {
|
it('should detect nested macro definitions', () => {
|
||||||
const text = `%macro test()
|
const text = `%macro test();\n %put 'hello';\n %macro test2;\n %put 'world;\n %mend\n%mend test`
|
||||||
%put 'hello';
|
|
||||||
%macro test2
|
|
||||||
%put 'world;
|
|
||||||
%mend
|
|
||||||
%mend test`
|
|
||||||
|
|
||||||
const macros = parseMacros(text, new LintConfig())
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
expect(macros.length).toEqual(2)
|
expect(macros.length).toEqual(2)
|
||||||
expect(macros).toContainEqual({
|
expect(macros).toContainEqual({
|
||||||
name: 'test',
|
name: 'test',
|
||||||
declarationLine: '%macro test()',
|
declarationLines: ['%macro test();'],
|
||||||
terminationLine: '%mend test',
|
terminationLine: '%mend test',
|
||||||
declaration: '%macro test()',
|
declaration: '%macro test()',
|
||||||
termination: '%mend test',
|
termination: '%mend test',
|
||||||
startLineNumber: 1,
|
startLineNumbers: [1],
|
||||||
endLineNumber: 6,
|
endLineNumber: 6,
|
||||||
parentMacro: '',
|
parentMacro: '',
|
||||||
hasMacroNameInMend: true,
|
hasMacroNameInMend: true,
|
||||||
hasParentheses: true,
|
|
||||||
mismatchedMendMacroName: ''
|
mismatchedMendMacroName: ''
|
||||||
})
|
})
|
||||||
expect(macros).toContainEqual({
|
expect(macros).toContainEqual({
|
||||||
name: 'test2',
|
name: 'test2',
|
||||||
declarationLine: ' %macro test2',
|
declarationLines: [' %macro test2;'],
|
||||||
terminationLine: ' %mend',
|
terminationLine: ' %mend',
|
||||||
declaration: '%macro test2',
|
declaration: '%macro test2',
|
||||||
termination: '%mend',
|
termination: '%mend',
|
||||||
startLineNumber: 3,
|
startLineNumbers: [3],
|
||||||
endLineNumber: 5,
|
endLineNumber: 5,
|
||||||
parentMacro: 'test',
|
parentMacro: 'test',
|
||||||
hasMacroNameInMend: false,
|
hasMacroNameInMend: false,
|
||||||
hasParentheses: false,
|
|
||||||
mismatchedMendMacroName: ''
|
mismatchedMendMacroName: ''
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe(`multi-line macro declarations`, () => {
|
||||||
|
it('should return an array with a single macro', () => {
|
||||||
|
const text = `%macro \n test;\n %put 'hello';\n%mend`
|
||||||
|
|
||||||
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
|
expect(macros.length).toEqual(1)
|
||||||
|
expect(macros).toContainEqual({
|
||||||
|
name: 'test',
|
||||||
|
declarationLines: ['%macro ', ' test;'],
|
||||||
|
terminationLine: '%mend',
|
||||||
|
declaration: '%macro test',
|
||||||
|
termination: '%mend',
|
||||||
|
startLineNumbers: [1, 2],
|
||||||
|
endLineNumber: 4,
|
||||||
|
parentMacro: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
|
mismatchedMendMacroName: ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single macro having parameters', () => {
|
||||||
|
const text = `%macro \n test(\n var,\n sum);%put 'hello';\n%mend`
|
||||||
|
|
||||||
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
|
expect(macros.length).toEqual(1)
|
||||||
|
expect(macros).toContainEqual({
|
||||||
|
name: 'test',
|
||||||
|
declarationLines: [
|
||||||
|
'%macro ',
|
||||||
|
` test(`,
|
||||||
|
` var,`,
|
||||||
|
` sum);%put 'hello';`
|
||||||
|
],
|
||||||
|
terminationLine: '%mend',
|
||||||
|
declaration: '%macro test( var, sum)',
|
||||||
|
termination: '%mend',
|
||||||
|
startLineNumbers: [1, 2, 3, 4],
|
||||||
|
endLineNumber: 5,
|
||||||
|
parentMacro: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
|
mismatchedMendMacroName: ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single macro having PARMBUFF option', () => {
|
||||||
|
const text = `%macro test\n /parmbuff;\n %put 'hello';\n%mend`
|
||||||
|
|
||||||
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
|
expect(macros.length).toEqual(1)
|
||||||
|
expect(macros).toContainEqual({
|
||||||
|
name: 'test',
|
||||||
|
declarationLines: ['%macro test', ' /parmbuff;'],
|
||||||
|
terminationLine: '%mend',
|
||||||
|
declaration: '%macro test /parmbuff',
|
||||||
|
termination: '%mend',
|
||||||
|
startLineNumbers: [1, 2],
|
||||||
|
endLineNumber: 4,
|
||||||
|
parentMacro: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
|
mismatchedMendMacroName: ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single macro having paramerter & SOURCE option', () => {
|
||||||
|
const text = `/* commentary */ %macro foobar/* commentary */(arg) \n /* commentary */\n /store\n /* commentary */source\n des="This macro does not do much";\n %put 'hello';\n%mend`
|
||||||
|
|
||||||
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
|
expect(macros.length).toEqual(1)
|
||||||
|
expect(macros).toContainEqual({
|
||||||
|
name: 'foobar',
|
||||||
|
declarationLines: [
|
||||||
|
'/* commentary */ %macro foobar/* commentary */(arg) ',
|
||||||
|
' /* commentary */',
|
||||||
|
' /store',
|
||||||
|
' /* commentary */source',
|
||||||
|
' des="This macro does not do much";'
|
||||||
|
],
|
||||||
|
terminationLine: '%mend',
|
||||||
|
declaration:
|
||||||
|
'%macro foobar(arg) /store source des="This macro does not do much"',
|
||||||
|
termination: '%mend',
|
||||||
|
startLineNumbers: [1, 2, 3, 4, 5],
|
||||||
|
endLineNumber: 7,
|
||||||
|
parentMacro: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
|
mismatchedMendMacroName: ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an array with a single macro having semi-colon in params', () => {
|
||||||
|
const text = `\n%macro mm_createapplication(\n tree=/User Folders/sasdemo\n ,name=myApp\n ,ClassIdentifier=mcore\n ,desc=Created by mm_createapplication\n ,params= param1=1
param2=blah\n ,version=\n ,frefin=mm_in\n ,frefout=mm_out\n ,mDebug=1\n );`
|
||||||
|
|
||||||
|
const macros = parseMacros(text, new LintConfig())
|
||||||
|
|
||||||
|
expect(macros.length).toEqual(1)
|
||||||
|
expect(macros).toContainEqual({
|
||||||
|
name: 'mm_createapplication',
|
||||||
|
declarationLines: [
|
||||||
|
`%macro mm_createapplication(`,
|
||||||
|
` tree=/User Folders/sasdemo`,
|
||||||
|
` ,name=myApp`,
|
||||||
|
` ,ClassIdentifier=mcore`,
|
||||||
|
` ,desc=Created by mm_createapplication`,
|
||||||
|
` ,params= param1=1
param2=blah`,
|
||||||
|
` ,version=`,
|
||||||
|
` ,frefin=mm_in`,
|
||||||
|
` ,frefout=mm_out`,
|
||||||
|
` ,mDebug=1`,
|
||||||
|
` );`
|
||||||
|
],
|
||||||
|
terminationLine: '',
|
||||||
|
declaration:
|
||||||
|
'%macro mm_createapplication( tree=/User Folders/sasdemo ,name=myApp ,ClassIdentifier=mcore ,desc=Created by mm_createapplication ,params= param1=1
param2=blah ,version= ,frefin=mm_in ,frefout=mm_out ,mDebug=1 )',
|
||||||
|
termination: '',
|
||||||
|
startLineNumbers: [2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12],
|
||||||
|
endLineNumber: null,
|
||||||
|
parentMacro: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
|
mismatchedMendMacroName: ''
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,21 +1,7 @@
|
|||||||
import { LintConfig } from '../types/LintConfig'
|
import { LintConfig, Macro } from '../types'
|
||||||
import { LineEndings } from '../types/LineEndings'
|
import { LineEndings } from '../types/LineEndings'
|
||||||
import { trimComments } from './trimComments'
|
import { trimComments } from './trimComments'
|
||||||
|
|
||||||
interface Macro {
|
|
||||||
name: string
|
|
||||||
startLineNumber: number | null
|
|
||||||
endLineNumber: number | null
|
|
||||||
declarationLine: string
|
|
||||||
terminationLine: string
|
|
||||||
declaration: string
|
|
||||||
termination: string
|
|
||||||
parentMacro: string
|
|
||||||
hasMacroNameInMend: boolean
|
|
||||||
hasParentheses: boolean
|
|
||||||
mismatchedMendMacroName: string
|
|
||||||
}
|
|
||||||
|
|
||||||
export const parseMacros = (text: string, config?: LintConfig): Macro[] => {
|
export const parseMacros = (text: string, config?: LintConfig): Macro[] => {
|
||||||
const lineEnding = config?.lineEndings === LineEndings.CRLF ? '\r\n' : '\n'
|
const lineEnding = config?.lineEndings === LineEndings.CRLF ? '\r\n' : '\n'
|
||||||
const lines: string[] = text ? text.split(lineEnding) : []
|
const lines: string[] = text ? text.split(lineEnding) : []
|
||||||
@@ -23,38 +9,107 @@ export const parseMacros = (text: string, config?: LintConfig): Macro[] => {
|
|||||||
|
|
||||||
let isCommentStarted = false
|
let isCommentStarted = false
|
||||||
let macroStack: Macro[] = []
|
let macroStack: Macro[] = []
|
||||||
lines.forEach((line, index) => {
|
let isReadingMacroDefinition = false
|
||||||
|
let isStatementContinues = true
|
||||||
|
let tempMacroDeclaration = ''
|
||||||
|
let tempMacroDeclarationLines: string[] = []
|
||||||
|
let tempStartLineNumbers: number[] = []
|
||||||
|
lines.forEach((line, lineIndex) => {
|
||||||
const { statement: trimmedLine, commentStarted } = trimComments(
|
const { statement: trimmedLine, commentStarted } = trimComments(
|
||||||
line,
|
line,
|
||||||
isCommentStarted
|
isCommentStarted
|
||||||
)
|
)
|
||||||
isCommentStarted = commentStarted
|
isCommentStarted = commentStarted
|
||||||
const statements: string[] = trimmedLine ? trimmedLine.split(';') : []
|
|
||||||
|
|
||||||
statements.forEach((statement) => {
|
isStatementContinues = !trimmedLine.endsWith(';')
|
||||||
|
|
||||||
|
const statements: string[] = trimmedLine.split(';')
|
||||||
|
|
||||||
|
if (isReadingMacroDefinition) {
|
||||||
|
// checking if code is split into statements based on `;` is a part of HTML Encoded Character
|
||||||
|
// if it happened, merges two statements into one
|
||||||
|
statements.forEach((statement, statementIndex) => {
|
||||||
|
if (/&[^\s]{1,5}$/.test(statement)) {
|
||||||
|
const next = statements[statementIndex]
|
||||||
|
const updatedStatement = `${statement};${
|
||||||
|
statements[statementIndex + 1]
|
||||||
|
}`
|
||||||
|
statements.splice(statementIndex, 1, updatedStatement)
|
||||||
|
statements.splice(statementIndex + 1, 1)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
statements.forEach((statement, statementIndex) => {
|
||||||
const { statement: trimmedStatement, commentStarted } = trimComments(
|
const { statement: trimmedStatement, commentStarted } = trimComments(
|
||||||
statement,
|
statement,
|
||||||
isCommentStarted
|
isCommentStarted
|
||||||
)
|
)
|
||||||
isCommentStarted = commentStarted
|
isCommentStarted = commentStarted
|
||||||
|
|
||||||
|
if (isReadingMacroDefinition) {
|
||||||
|
tempMacroDeclaration =
|
||||||
|
tempMacroDeclaration +
|
||||||
|
(trimmedStatement ? ' ' + trimmedStatement : '')
|
||||||
|
tempMacroDeclarationLines.push(line)
|
||||||
|
tempStartLineNumbers.push(lineIndex + 1)
|
||||||
|
|
||||||
|
if (!Object.is(statements.length - 1, statementIndex)) {
|
||||||
|
isReadingMacroDefinition = false
|
||||||
|
|
||||||
|
const name = tempMacroDeclaration
|
||||||
|
.slice(7, tempMacroDeclaration.length)
|
||||||
|
.trim()
|
||||||
|
.split('/')[0]
|
||||||
|
.split('(')[0]
|
||||||
|
.trim()
|
||||||
|
macroStack.push({
|
||||||
|
name,
|
||||||
|
startLineNumbers: tempStartLineNumbers,
|
||||||
|
endLineNumber: null,
|
||||||
|
parentMacro: macroStack.length
|
||||||
|
? macroStack[macroStack.length - 1].name
|
||||||
|
: '',
|
||||||
|
hasMacroNameInMend: false,
|
||||||
|
mismatchedMendMacroName: '',
|
||||||
|
declarationLines: tempMacroDeclarationLines,
|
||||||
|
terminationLine: '',
|
||||||
|
declaration: tempMacroDeclaration,
|
||||||
|
termination: ''
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (trimmedStatement.startsWith('%macro')) {
|
if (trimmedStatement.startsWith('%macro')) {
|
||||||
const startLineNumber = index + 1
|
const startLineNumber = lineIndex + 1
|
||||||
|
|
||||||
|
if (
|
||||||
|
isStatementContinues &&
|
||||||
|
Object.is(statements.length - 1, statementIndex)
|
||||||
|
) {
|
||||||
|
tempMacroDeclaration = trimmedStatement
|
||||||
|
tempMacroDeclarationLines = [line]
|
||||||
|
tempStartLineNumbers = [startLineNumber]
|
||||||
|
isReadingMacroDefinition = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
const name = trimmedStatement
|
const name = trimmedStatement
|
||||||
.slice(7, trimmedStatement.length)
|
.slice(7, trimmedStatement.length)
|
||||||
.trim()
|
.trim()
|
||||||
|
.split('/')[0]
|
||||||
.split('(')[0]
|
.split('(')[0]
|
||||||
|
.trim()
|
||||||
macroStack.push({
|
macroStack.push({
|
||||||
name,
|
name,
|
||||||
startLineNumber,
|
startLineNumbers: [startLineNumber],
|
||||||
endLineNumber: null,
|
endLineNumber: null,
|
||||||
parentMacro: macroStack.length
|
parentMacro: macroStack.length
|
||||||
? macroStack[macroStack.length - 1].name
|
? macroStack[macroStack.length - 1].name
|
||||||
: '',
|
: '',
|
||||||
hasParentheses: trimmedStatement.endsWith('()'),
|
|
||||||
hasMacroNameInMend: false,
|
hasMacroNameInMend: false,
|
||||||
mismatchedMendMacroName: '',
|
mismatchedMendMacroName: '',
|
||||||
declarationLine: line,
|
declarationLines: [line],
|
||||||
terminationLine: '',
|
terminationLine: '',
|
||||||
declaration: trimmedStatement,
|
declaration: trimmedStatement,
|
||||||
termination: ''
|
termination: ''
|
||||||
@@ -64,7 +119,7 @@ export const parseMacros = (text: string, config?: LintConfig): Macro[] => {
|
|||||||
const macro = macroStack.pop() as Macro
|
const macro = macroStack.pop() as Macro
|
||||||
const mendMacroName =
|
const mendMacroName =
|
||||||
trimmedStatement.split(' ').filter((s: string) => !!s)[1] || ''
|
trimmedStatement.split(' ').filter((s: string) => !!s)[1] || ''
|
||||||
macro.endLineNumber = index + 1
|
macro.endLineNumber = lineIndex + 1
|
||||||
macro.hasMacroNameInMend = mendMacroName === macro.name
|
macro.hasMacroNameInMend = mendMacroName === macro.name
|
||||||
macro.mismatchedMendMacroName = macro.hasMacroNameInMend
|
macro.mismatchedMendMacroName = macro.hasMacroNameInMend
|
||||||
? ''
|
? ''
|
||||||
@@ -75,13 +130,12 @@ export const parseMacros = (text: string, config?: LintConfig): Macro[] => {
|
|||||||
} else {
|
} else {
|
||||||
macros.push({
|
macros.push({
|
||||||
name: '',
|
name: '',
|
||||||
startLineNumber: null,
|
startLineNumbers: [],
|
||||||
endLineNumber: index + 1,
|
endLineNumber: lineIndex + 1,
|
||||||
parentMacro: '',
|
parentMacro: '',
|
||||||
hasParentheses: false,
|
|
||||||
hasMacroNameInMend: false,
|
hasMacroNameInMend: false,
|
||||||
mismatchedMendMacroName: '',
|
mismatchedMendMacroName: '',
|
||||||
declarationLine: '',
|
declarationLines: [],
|
||||||
terminationLine: line,
|
terminationLine: line,
|
||||||
declaration: '',
|
declaration: '',
|
||||||
termination: trimmedStatement
|
termination: trimmedStatement
|
||||||
|
|||||||
@@ -7,6 +7,27 @@ describe('trimComments', () => {
|
|||||||
/* some comment */ some code;
|
/* some comment */ some code;
|
||||||
`)
|
`)
|
||||||
).toEqual({ statement: 'some code;', commentStarted: false })
|
).toEqual({ statement: 'some code;', commentStarted: false })
|
||||||
|
|
||||||
|
expect(
|
||||||
|
trimComments(`
|
||||||
|
/*/ some comment */ some code;
|
||||||
|
`)
|
||||||
|
).toEqual({ statement: 'some code;', commentStarted: false })
|
||||||
|
|
||||||
|
expect(
|
||||||
|
trimComments(`
|
||||||
|
some code;/*/ some comment */ some code;
|
||||||
|
`)
|
||||||
|
).toEqual({ statement: 'some code; some code;', commentStarted: false })
|
||||||
|
|
||||||
|
expect(
|
||||||
|
trimComments(`/* some comment */
|
||||||
|
/* some comment */ CODE_Keyword1 /* some comment */ CODE_Keyword2/* some comment */;/* some comment */
|
||||||
|
/* some comment */`)
|
||||||
|
).toEqual({
|
||||||
|
statement: 'CODE_Keyword1 CODE_Keyword2;',
|
||||||
|
commentStarted: false
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should return statment, having multi-line comment', () => {
|
it('should return statment, having multi-line comment', () => {
|
||||||
|
|||||||
@@ -1,19 +1,38 @@
|
|||||||
export const trimComments = (
|
export const trimComments = (
|
||||||
statement: string,
|
statement: string,
|
||||||
commentStarted: boolean = false
|
commentStarted: boolean = false,
|
||||||
|
trimEnd: boolean = false
|
||||||
): { statement: string; commentStarted: boolean } => {
|
): { statement: string; commentStarted: boolean } => {
|
||||||
let trimmed = (statement || '').trim()
|
let trimmed = trimEnd ? (statement || '').trimEnd() : (statement || '').trim()
|
||||||
|
|
||||||
if (commentStarted || trimmed.startsWith('/*')) {
|
if (commentStarted || trimmed.startsWith('/*')) {
|
||||||
const parts = trimmed.split('*/')
|
const parts = trimmed.startsWith('/*')
|
||||||
if (parts.length > 1) {
|
? trimmed.slice(2).split('*/')
|
||||||
|
: trimmed.split('*/')
|
||||||
|
if (parts.length === 2) {
|
||||||
return {
|
return {
|
||||||
statement: (parts.pop() as string).trim(),
|
statement: (parts.pop() as string).trim(),
|
||||||
commentStarted: false
|
commentStarted: false
|
||||||
}
|
}
|
||||||
|
} else if (parts.length > 2) {
|
||||||
|
parts.shift()
|
||||||
|
return trimComments(parts.join('*/'), false)
|
||||||
} else {
|
} else {
|
||||||
return { statement: '', commentStarted: true }
|
return { statement: '', commentStarted: true }
|
||||||
}
|
}
|
||||||
|
} else if (trimmed.includes('/*')) {
|
||||||
|
const statementBeforeCommentStarts = trimmed.slice(0, trimmed.indexOf('/*'))
|
||||||
|
trimmed = trimmed.slice(trimmed.indexOf('/*') + 2)
|
||||||
|
const remainingStatement = trimmed.slice(trimmed.indexOf('*/') + 2)
|
||||||
|
|
||||||
|
const result = trimComments(remainingStatement, false, true)
|
||||||
|
const completeStatement = statementBeforeCommentStarts + result.statement
|
||||||
|
return {
|
||||||
|
statement: trimEnd
|
||||||
|
? completeStatement.trimEnd()
|
||||||
|
: completeStatement.trim(),
|
||||||
|
commentStarted: result.commentStarted
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return { statement: trimmed, commentStarted: false }
|
return { statement: trimmed, commentStarted: false }
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user