Skip to content

Commit

Permalink
Merge pull request #189 from rei/pr/stylelint
Browse files Browse the repository at this point in the history
Pr/stylelint
  • Loading branch information
mhewson authored Nov 12, 2024
2 parents 8f7f625 + 5ad7e40 commit e5c109c
Show file tree
Hide file tree
Showing 104 changed files with 958 additions and 360 deletions.
5 changes: 4 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -52,4 +52,7 @@ src/**/package-lock\.json
unmigrated/

## Dev output for hosting
pages
pages

# Ignore reports directory
/reports/
53 changes: 41 additions & 12 deletions .stylelintrc
Original file line number Diff line number Diff line change
@@ -1,19 +1,48 @@
{
"extends": "stylelint-config-recommended-scss",
"extends": [
"stylelint-config-recommended-scss",
"stylelint-config-standard-scss"
],
"ignoreFiles": [
"**/*.js",
"**/*.snap",
"**/*.vars.pcss"
"**/*.js",
"**/*.snap",
"**/*.vars.pcss"
],
"plugins": [
"stylelint-scss"
"stylelint-scss"
],
"rules": {
"at-rule-disallowed-list": ["debug"],
"no-empty-source": null,
"no-descending-specificity": null,
"selector-pseudo-class-no-unknown": [true, {
"ignorePseudoClasses": ["global"]
}]
"at-rule-disallowed-list": ["debug"],
"no-empty-source": null,
"no-descending-specificity": null,
"selector-pseudo-class-no-unknown": [
true,
{
"ignorePseudoClasses": ["global"]
}
],
"scss/at-extend-no-missing-placeholder": true,
"scss/comment-no-loud": null,
"scss/function-color-relative": true,
"scss/at-if-no-null": true,
"scss/dollar-variable-no-missing-interpolation": true,
"scss/selector-no-redundant-nesting-selector": true,
"scss/at-import-partial-extension": null,
"scss/at-mixin-pattern": null,
"scss/at-function-pattern": null,
"scss/dollar-variable-pattern": null,
"scss/percent-placeholder-pattern": null,
"function-disallowed-list": null,
"color-function-notation": null,
"alpha-value-notation": null,
"color-hex-length": null,
"custom-property-pattern": null,
"selector-class-pattern": [
"^[a-z0-9]+(?:-[a-z0-9]+)*(?:__(?:[a-z0-9]+(?:-[a-z0-9]+)*))?(?:--(?:[a-z0-9]+(?:-[a-z0-9]+)*))?$",
{
"message": "Expected class selector to follow BEM naming convention (selector-class-pattern)"
}
]
}
}
}

148 changes: 148 additions & 0 deletions lint-report-generator.mjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,148 @@
/* eslint-env node */

// lint-report-generator.mjs
import stylelint from 'stylelint';
import fs from 'fs';
import { join, relative } from 'path';
import { fileURLToPath } from 'url';
import { dirname } from 'path';

const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);

/**
* Formats the current date as YYYY-MM-DD
*/
const getFormattedDate = () => {
const date = new Date();
return date.toISOString().split('T')[0];
};

/**
* Creates directory if it doesn't exist
*/
const ensureDirectoryExists = (dirPath) => {
if (!fs.existsSync(dirPath)) {
fs.mkdirSync(dirPath, { recursive: true });
}
};

/**
* Processes stylelint results into a structured format
*/
const processStylelintResults = (styleResults) => {
return styleResults.results.reduce((acc, file) => {
if (file.warnings.length > 0) {
const relativePath = relative(process.cwd(), file.source);
acc.push({
file: relativePath,
warnings: file.warnings.map(w => ({
line: w.line,
column: w.column,
rule: w.rule,
text: w.text,
severity: w.severity
}))
});
}
return acc;
}, []);
};

/**
* Generates markdown content from warnings
*/
const generateMarkdownContent = (styleWarnings, reportDate) => {
let mdReport = `# Stylelint Warning Report (${reportDate})\n\n`;
mdReport += `## Summary\n\n`;

const totalWarnings = styleWarnings.reduce((sum, file) =>
sum + file.warnings.length, 0);

mdReport += `- Total files with warnings: ${styleWarnings.length}\n`;
mdReport += `- Total warnings: ${totalWarnings}\n\n`;

if (styleWarnings.length === 0) {
mdReport += 'No style warnings found.\n\n';
return mdReport;
}

mdReport += `## Detailed Warnings\n\n`;
styleWarnings.forEach(file => {
mdReport += `### ${file.file}\n\n`;
mdReport += '| Line:Col | Rule | Message |\n';
mdReport += '|----------|------|---------||\n';
file.warnings.forEach(warning => {
const escapedText = warning.text.replace(/\|/g, '\\|');
mdReport += `
| ${warning.line}:${warning.column}
| \`${warning.rule}\`
| ${escapedText} |\n`;
});
mdReport += '\n';
});

return mdReport;
};

/**
* Main function to generate lint reports
*/
const generateLintReport = async () => {
try {
const reportDate = getFormattedDate();
const reportDir = join(process.cwd(), 'reports');

ensureDirectoryExists(reportDir);

const styleResults = await stylelint.lint({
files: 'src/**/*.scss',
formatter: 'json'
});

const styleWarnings = processStylelintResults(styleResults);
const mdReport = generateMarkdownContent(styleWarnings, reportDate);

const reportPath = join(reportDir, `stylelint-report-${reportDate}.md`);
const jsonReportPath = join(reportDir, `stylelint-report-${reportDate}.json`);

fs.writeFileSync(reportPath, mdReport);
fs.writeFileSync(jsonReportPath, JSON.stringify({
date: reportDate,
styleWarnings
}, null, 2));

const summary = {
totalFiles: styleWarnings.length,
totalWarnings: styleWarnings.reduce((sum, file) =>
sum + file.warnings.length, 0),
reportPath,
jsonReportPath
};

console.log(`
Reports generated successfully:
- Markdown: ${summary.reportPath}
- JSON: ${summary.jsonReportPath}
Summary:
- Total files with warnings: ${summary.totalFiles}
- Total warnings: ${summary.totalWarnings}
`);

return summary;

} catch (error) {
console.error('Error generating lint report:', error);
throw error;
}
};

// Execute if called directly
if (import.meta.url === `file://${__filename}`) {
generateLintReport().catch(error => {
console.error('An error occurred:', error);
process.exit(1);
});
}

export default generateLintReport;
Loading

0 comments on commit e5c109c

Please sign in to comment.