* feat: show open boilerplate prs on dashboard fix: rest of boilerplate server changes fix: more fix: other * fix: update lib functions * fix: retrofitted one-off scripts * feat: added rateLimit for requests * fix: reduce time * fix: put limiter inside each route * fix: make client show when rated limited * fix: removed unused probot from app * fix: renamed folders * fix: consolidate config.js and constants.js * chore: update octokit to latest version * fix: remove invalid file * fix: refactored update-db.js * feat: add fcc logo * fix: logo url * fix: remove Home link * fix: change link colors * fix: added rate limiter to landing page * fix: ran npm install in client to create package-lock.json * fix: correct typo in doc Co-authored-by: Nicholas Carrigan (he/him) <nhcarrigan@gmail.com> * fix: Replace favicon, Gitter => Discord Signed-off-by: nhcarrigan <nhcarrigan@gmail.com> * fix: add extra linting guidance to package.json * Ignore contributor app Signed-off-by: nhcarrigan <nhcarrigan@gmail.com> * fix: revert linting rules for client * fix: add skip_preflight_check=true for tests Co-authored-by: Kristofer Koishigawa <scissorsneedfoodtoo@gmail.com> Co-authored-by: Oliver Eyton-Williams <ojeytonwilliams@gmail.com> Co-authored-by: Kris Koishigawa <scissorsneedfoodtoo@gmail.com> Co-authored-by: Nicholas Carrigan (he/him) <nhcarrigan@gmail.com>
		
			
				
	
	
		
			158 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			158 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /*
 | |
| This is a one-off script to run on all open PRs to add
 | |
| a comment and "status: needs update" label to any PR with guide articles which
 | |
| have frontmatter issues.
 | |
| */
 | |
| 
 | |
| const fetch = require('node-fetch');
 | |
| const { getPRs, getUserInput, getFiles } = require('../lib/get-prs');
 | |
| const { addLabels, addComment } = require('../lib/pr-tasks');
 | |
| const { rateLimiter, ProcessingLog } = require('../lib/utils');
 | |
| const {
 | |
|   frontmatterCheck
 | |
| } = require('../lib/validation/guide-folder-checks/frontmatter-check');
 | |
| const {
 | |
|   createErrorMsg
 | |
| } = require('../lib/validation/guide-folder-checks/create-error-msg');
 | |
| 
 | |
| const {
 | |
|   github: { freeCodeCampRepo, defaultBase },
 | |
|   oneoff: { productionRun }
 | |
| } = require('../config');
 | |
| 
 | |
| const allowedLangDirNames = [
 | |
|   'arabic',
 | |
|   'chinese',
 | |
|   'english',
 | |
|   'portuguese',
 | |
|   'russian',
 | |
|   'spanish'
 | |
| ];
 | |
| 
 | |
| const log = new ProcessingLog('all-frontmatter-checks');
 | |
| 
 | |
| const labeler = async (
 | |
|   number,
 | |
|   prFiles,
 | |
|   currentLabels,
 | |
|   guideFolderErrorsComment
 | |
| ) => {
 | |
|   // holds potential labels to add based on file path
 | |
|   const labelsToAdd = {};
 | |
|   if (guideFolderErrorsComment) {
 | |
|     labelsToAdd['status: needs update'] = 1;
 | |
|   }
 | |
|   const existingLabels = currentLabels.map(({ name }) => name);
 | |
| 
 | |
|   /* only adds needed labels which are NOT currently on the PR. */
 | |
|   const newLabels = Object.keys(labelsToAdd).filter(label => {
 | |
|     return !existingLabels.includes(label);
 | |
|   });
 | |
|   if (newLabels.length) {
 | |
|     if (productionRun) {
 | |
|       addLabels(number, newLabels);
 | |
|       await rateLimiter();
 | |
|     }
 | |
|   }
 | |
|   return newLabels;
 | |
| };
 | |
| 
 | |
| const checkPath = (fullPath, fileContent) => {
 | |
|   let errorMsgs = [];
 | |
|   const remaining = fullPath.split('/');
 | |
|   const isTranslation =
 | |
|     allowedLangDirNames.includes(remaining[1]) && remaining[1] !== 'english';
 | |
|   const frontMatterErrMsgs = frontmatterCheck(
 | |
|     fullPath,
 | |
|     isTranslation,
 | |
|     fileContent
 | |
|   );
 | |
|   return errorMsgs.concat(frontMatterErrMsgs);
 | |
| };
 | |
| 
 | |
| const guideFolderChecks = async (number, prFiles, user) => {
 | |
|   let prErrors = [];
 | |
|   for (let { filename: fullPath, raw_url: fileUrl } of prFiles) {
 | |
|     let newErrors;
 | |
|     if (/^guide\//.test(fullPath)) {
 | |
|       const response = await fetch(fileUrl);
 | |
|       const fileContent = await response.text();
 | |
|       newErrors = checkPath(fullPath, fileContent);
 | |
|     }
 | |
|     if (newErrors) {
 | |
|       prErrors = prErrors.concat(newErrors);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (prErrors.length) {
 | |
|     const comment = createErrorMsg(prErrors, user);
 | |
|     if (productionRun) {
 | |
|       await addComment(number, comment);
 | |
|       await rateLimiter();
 | |
|     }
 | |
|     return comment;
 | |
|   } else {
 | |
|     return null;
 | |
|   }
 | |
| };
 | |
| 
 | |
| (async () => {
 | |
|   const { totalPRs, firstPR, lastPR } = await getUserInput(
 | |
|     freeCodeCampRepo,
 | |
|     defaultBase
 | |
|   );
 | |
|   const prPropsToGet = ['number', 'labels', 'user'];
 | |
|   const { openPRs } = await getPRs(
 | |
|     freeCodeCampRepo,
 | |
|     defaultBase,
 | |
|     totalPRs,
 | |
|     firstPR,
 | |
|     lastPR,
 | |
|     prPropsToGet
 | |
|   );
 | |
| 
 | |
|   log.start();
 | |
|   console.log('Starting frontmatter checks process...');
 | |
|   let count = 0;
 | |
|   for (let i = 0; i < openPRs.length; i++) {
 | |
|     if (openPRs.length) {
 | |
|       let {
 | |
|         number,
 | |
|         labels: currentLabels,
 | |
|         user: { login: username }
 | |
|       } = openPRs[count];
 | |
| 
 | |
|       const prFiles = await getFiles(freeCodeCampRepo, number);
 | |
|       if (count > 4000) {
 | |
|         await rateLimiter(2350);
 | |
|       }
 | |
|       const guideFolderErrorsComment = await guideFolderChecks(
 | |
|         number,
 | |
|         prFiles,
 | |
|         username
 | |
|       );
 | |
|       const commentLogVal = guideFolderErrorsComment
 | |
|         ? guideFolderErrorsComment
 | |
|         : 'none';
 | |
| 
 | |
|       const labelsAdded = await labeler(
 | |
|         number,
 | |
|         prFiles,
 | |
|         currentLabels,
 | |
|         guideFolderErrorsComment
 | |
|       );
 | |
|       const labelLogVal = labelsAdded.length ? labelsAdded : 'none added';
 | |
| 
 | |
|       log.add(number, { number, comment: commentLogVal, labels: labelLogVal });
 | |
|     }
 | |
|   }
 | |
| })()
 | |
|   .then(() => {
 | |
|     log.finish();
 | |
|     console.log('Successfully completed frontmatter checks');
 | |
|   })
 | |
|   .catch(err => {
 | |
|     log.finish();
 | |
|     console.log(err);
 | |
|   });
 |