diff --git a/src/proxy/chain.js b/src/proxy/chain.js index 2873f1ac3..0c62599c5 100644 --- a/src/proxy/chain.js +++ b/src/proxy/chain.js @@ -15,6 +15,7 @@ const pushActionChain = [ proc.push.checkForAiMlUsage, proc.push.checkExifJpeg, proc.push.clearBareClone, + proc.push.checkCryptoImplementation, proc.push.scanDiff, proc.push.blockForAuth, ]; diff --git a/src/proxy/processors/push-action/checkCryptoImplementation.js b/src/proxy/processors/push-action/checkCryptoImplementation.js new file mode 100644 index 000000000..5be14bf30 --- /dev/null +++ b/src/proxy/processors/push-action/checkCryptoImplementation.js @@ -0,0 +1,140 @@ +const Step = require('../../actions').Step; + +// Common encryption-related patterns and keywords +const CRYPTO_PATTERNS = { + // Known non-standard encryption algorithms + nonStandardAlgorithms: [ + 'xor\\s*\\(', + 'rot13', + 'caesar\\s*cipher', + 'custom\\s*encrypt', + 'simple\\s*encrypt', + 'homebrew\\s*crypto', + 'custom\\s*hash' + ], + + // Suspicious operations that might indicate custom crypto Implementation + suspiciousOperations: [ + 'bit\\s*shift', + 'bit\\s*rotate', + '\\^=', + '\\^', + '>>>', + '<<<', + 'shuffle\\s*bytes' + ], + + // Common encryption-related variable names + suspiciousVariables: [ + 'cipher', + 'encrypt', + 'decrypt', + 'scramble', + 'salt(?!\\w)', + 'iv(?!\\w)', + 'nonce' + ] +}; + +function analyzeCodeForCrypto(diffContent) { + const issues = []; + // Check for above mentioned cryto Patterns + if(!diffContent) return issues; + + CRYPTO_PATTERNS.nonStandardAlgorithms.forEach(pattern => { + const regex = new RegExp(pattern, 'gi'); + const matches = diffContent.match(regex); + if (matches) { + issues.push({ + type: 'non_standard_algorithm', + pattern: pattern, + matches: matches, + severity: 'high', + message: `Detected possible non-standard encryption algorithm: ${matches.join(', ')}` + }); + } + }); + + CRYPTO_PATTERNS.suspiciousOperations.forEach(pattern => { + const regex = new RegExp(pattern, 'gi'); + const matches = diffContent.match(regex); + if (matches) { + issues.push({ + type: 'suspicious_operation', + pattern: pattern, + matches: matches, + severity: 'medium', + message: `Detected suspicious cryptographic operation: ${matches.join(', ')}` + }); + } + }); + + CRYPTO_PATTERNS.suspiciousVariables.forEach(pattern => { + const regex = new RegExp(pattern, 'gi'); + const matches = diffContent.match(regex); + if (matches) { + issues.push({ + type: 'suspicious_variable', + pattern: pattern, + matches: matches, + severity: 'low', + message: `Detected potential encryption-related variable: ${matches.join(', ')}` + }); + } + }); + + return issues; +} + +const exec = async (req, action) => { + const step = new Step('checkCryptoImplementation'); + + try { + let hasIssues = false; + const allIssues = []; + + for (const commit of action.commitData) { + const diff = commit.diff || ''; + const issues = analyzeCodeForCrypto(diff); + + if (issues.length > 0) { + hasIssues = true; + allIssues.push({ + commit: commit.hash, + issues: issues + }); + } + } + + if (hasIssues) { + step.error = true; + + const errorMessage = allIssues.map(commitIssues => { + return `Commit ${commitIssues.commit}:\n` + + commitIssues.issues.map(issue => + `- ${issue.severity.toUpperCase()}: ${issue.message}` + ).join('\n'); + }).join('\n\n'); + + step.setError( + '\n\nYour push has been blocked.\n' + + 'Potential non-standard cryptographic implementations detected:\n\n' + + `${errorMessage}\n\n` + + 'Please use standard cryptographic libraries instead of custom implementations.\n' + + 'Recommended: Use established libraries like crypto, node-forge, or Web Crypto API.\n' + ); + } + + action.addStep(step); + return action; + } catch (error) { + step.error = true; + step.setError(`Error analyzing crypto implementation: ${error.message}`); + action.addStep(step); + return action; + } +}; + +// exec.displayName = 'checkCryptoImplementation.exec'; +exports.exec = exec; +exports.analyzeCodeForCrypto = analyzeCodeForCrypto; \ No newline at end of file diff --git a/src/proxy/processors/push-action/index.js b/src/proxy/processors/push-action/index.js index b4e0e6527..528577ea7 100644 --- a/src/proxy/processors/push-action/index.js +++ b/src/proxy/processors/push-action/index.js @@ -12,6 +12,7 @@ console.log(__dirname); exports.checkAuthorEmails = require('./checkAuthorEmails').exec; exports.checkUserPushPermission = require('./checkUserPushPermission').exec; exports.clearBareClone = require('./clearBareClone').exec; +exports.checkCryptoImplementation = require('./checkCryptoImplementation').exec; exports.checkSensitiveData = require('./checkSensitiveData').exec; exports.checkForAiMlusage = require('./checkForAiMlUsage').exec; exports.checkExifJpeg = require('./checkExifJpeg').exec; diff --git a/test/chain.test.js b/test/chain.test.js index 60d3c4a82..81d0190ba 100644 --- a/test/chain.test.js +++ b/test/chain.test.js @@ -20,6 +20,7 @@ const mockPushProcessors = { audit: sinon.stub(), checkRepoInAuthorisedList: sinon.stub(), checkCommitMessages: sinon.stub(), + checkCryptoImplementation: sinon.stub(), checkAuthorEmails: sinon.stub(), checkUserPushPermission: sinon.stub(), checkIfWaitingAuth: sinon.stub(), @@ -36,6 +37,7 @@ mockPushProcessors.parsePush.displayName = 'parsePush'; mockPushProcessors.audit.displayName = 'audit'; mockPushProcessors.checkRepoInAuthorisedList.displayName = 'checkRepoInAuthorisedList'; mockPushProcessors.checkCommitMessages.displayName = 'checkCommitMessages'; +mockPushProcessors.checkCryptoImplementation.displayName = 'checkCryptoImplementation'; mockPushProcessors.checkAuthorEmails.displayName = 'checkAuthorEmails'; mockPushProcessors.checkUserPushPermission.displayName = 'checkUserPushPermission'; mockPushProcessors.checkIfWaitingAuth.displayName = 'checkIfWaitingAuth'; @@ -110,6 +112,7 @@ describe('proxy chain', function () { mockPushProcessors.checkEXIFJpeg.resolves(continuingAction); mockPushProcessors.checkAuthorEmails.resolves(continuingAction); mockPushProcessors.checkUserPushPermission.resolves(continuingAction); + mockPushProcessors.checkCryptoImplementation.resolves(continuingAction); mockPushProcessors.checkSensitiveData.resolves(continuingAction); // this stops the chain from further execution @@ -125,6 +128,7 @@ describe('proxy chain', function () { expect(mockPushProcessors.checkIfWaitingAuth.called).to.be.true; expect(mockPushProcessors.pullRemote.called).to.be.false; expect(mockPushProcessors.audit.called).to.be.true; + expect(mockPushProcessors.checkCryptoImplementation.called).to.be.true; expect(result.type).to.equal('push'); expect(result.allowPush).to.be.false; @@ -136,10 +140,12 @@ describe('proxy chain', function () { const continuingAction = { type: 'push', continue: () => true, allowPush: false }; mockPreProcessors.parseAction.resolves({ type: 'push' }); mockPushProcessors.parsePush.resolves(continuingAction); + mockPushProcessors.checkCryptoImplementation.resolves(continuingAction); mockPushProcessors.checkRepoInAuthorisedList.resolves(continuingAction); mockPushProcessors.checkCommitMessages.resolves(continuingAction); mockPushProcessors.checkAuthorEmails.resolves(continuingAction); mockPushProcessors.checkUserPushPermission.resolves(continuingAction); + // this stops the chain from further execution mockPushProcessors.checkIfWaitingAuth.resolves({ type: 'push', continue: () => true, allowPush: true }); @@ -154,6 +160,7 @@ describe('proxy chain', function () { expect(mockPushProcessors.checkIfWaitingAuth.called).to.be.true; expect(mockPushProcessors.pullRemote.called).to.be.false; expect(mockPushProcessors.audit.called).to.be.true; + expect(mockPushProcessors.checkCryptoImplementation.called).to.be.true; expect(result.type).to.equal('push'); expect(result.allowPush).to.be.true; @@ -177,6 +184,7 @@ describe('proxy chain', function () { mockPushProcessors.clearBareClone.resolves(continuingAction); mockPushProcessors.scanDiff.resolves(continuingAction); mockPushProcessors.blockForAuth.resolves(continuingAction); + mockPushProcessors.checkCryptoImplementation.resolves(continuingAction); const result = await chain.executeChain(req); @@ -195,6 +203,7 @@ describe('proxy chain', function () { expect(mockPushProcessors.scanDiff.called).to.be.true; expect(mockPushProcessors.blockForAuth.called).to.be.true; expect(mockPushProcessors.audit.called).to.be.true; + expect(mockPushProcessors.checkCryptoImplementation.called).to.be.true; expect(mockPushProcessors.checkSensitiveData.called).to.be.true; expect(result.type).to.equal('push'); diff --git a/test/checkCryptoImplementation.test.js b/test/checkCryptoImplementation.test.js new file mode 100644 index 000000000..1d181c572 --- /dev/null +++ b/test/checkCryptoImplementation.test.js @@ -0,0 +1,223 @@ +const { expect } = require('chai'); +const { analyzeCodeForCrypto, exec } = require('../src/proxy/processors/push-action/checkCryptoImplementation.js'); + +describe('Crypto Implementation Check Plugin', () => { + describe('analyzeCodeForCrypto', () => { + it('should detect non-standard encryption algorithms', () => { + const testCode = ` + function customEncrypt(data) { + return data.split('').map(char => + String.fromCharCode(char.charCodeAt(0) ^ 0x7F) + ).join(''); + } + `; + + const issues = analyzeCodeForCrypto(testCode); + expect(issues).to.have.lengthOf.at.least(1); + expect(issues.some(i => i.type === 'non_standard_algorithm')).to.be.true; + }); + + it('should detect suspicious bit operations', () => { + const testCode = ` + function scrambleData(data) { + let result = ''; + for(let i = 0; i < data.length; i++) { + result += String.fromCharCode(data.charCodeAt(i) >>> 2); + } + return result; + } + `; + + const issues = analyzeCodeForCrypto(testCode); + expect(issues).to.have.lengthOf.at.least(1); + expect(issues.some(i => i.type === 'suspicious_operation')).to.be.true; + }); + + it('should detect suspicious variable names', () => { + const testCode = ` + const cipher = {}; + let salt = generateRandomBytes(16); + const iv = new Uint8Array(12); + `; + + const issues = analyzeCodeForCrypto(testCode); + expect(issues).to.have.lengthOf.at.least(3); + expect(issues.some(i => i.type === 'suspicious_variable')).to.be.true; + }); + + it('should not flag standard crypto library usage', () => { + const testCode = ` + const crypto = require('crypto'); + const cipher = crypto.createCipheriv('aes-256-gcm', key, iv); + `; + + const issues = analyzeCodeForCrypto(testCode); + expect(issues.filter(i => i.severity === 'high')).to.have.lengthOf(0); + }); + + it('should handle empty input', () => { + const issues = analyzeCodeForCrypto(''); + expect(issues).to.be.an('array').that.is.empty; + }); + + it('should handle null or undefined input', () => { + expect(analyzeCodeForCrypto(null)).to.be.an('array').that.is.empty; + expect(analyzeCodeForCrypto(undefined)).to.be.an('array').that.is.empty; + }); + + }); + + describe('exec', () => { + + it('should handle empty diff content', async () => { + const req = {}; + const action = { + commitData: [{ + hash: '123abc', + diff: '' + }], + addStep: function(step) { this.step = step; } + }; + + const result = await exec(req, action); + expect(result.step.error).to.be.false; + }); + + it('should handle undefined diff content', async () => { + const req = {}; + const action = { + commitData: [{ + hash: '123abc' + // diff is undefined + }], + addStep: function(step) { this.step = step; } + }; + + const result = await exec(req, action); + expect(result.step.error).to.be.false; + }); + + it('should handle empty commitData array', async () => { + const req = {}; + const action = { + commitData: [], + addStep: function(step) { this.step = step; } + }; + + const result = await exec(req, action); + expect(result.step.error).to.be.false; + }); + it('should block commits with non-standard crypto implementations', async () => { + const req = {}; + const action = { + commitData: [{ + hash: '123abc', + diff: ` + function customEncrypt(data) { + return data.split('').map(char => + String.fromCharCode(char.charCodeAt(0) ^ 0x7F) + ).join(''); + } + ` + }], + addStep: function(step) { this.step = step; } + }; + + const result = await exec(req, action); + expect(result.step.error).to.be.true; + }); + + it('should allow commits without crypto issues', async () => { + const req = {}; + const action = { + commitData: [{ + hash: '123abc', + diff: ` + function normalFunction() { + return 'Hello World'; + } + ` + }], + addStep: function(step) { this.step = step; } + }; + + const result = await exec(req, action); + expect(result.step.error).to.be.false; + }); + + it('should handle multiple commits', async () => { + const req = {}; + const action = { + commitData: [ + { + hash: '123abc', + diff: `function safe() { return true; }` + }, + { + hash: '456def', + diff: ` + function rot13(str) { + return str.replace(/[a-zA-Z]/g, c => + String.fromCharCode((c <= 'Z' ? 90 : 122) >= (c = c.charCodeAt(0) + 13) ? c : c - 26) + ); + } + ` + } + ], + addStep: function(step) { this.step = step; } + }; + + const result = await exec(req, action); + expect(result.step).to.have.property('error', true); +}); + + + it('should handle errors gracefully', async () => { + const req = {}; + const action = { + commitData: null, + addStep: function(step) { this.step = step; } + }; + + const result = await exec(req, action); + expect(result.step.error).to.be.true; + }); + }); + + describe('Pattern Detection', () => { + it('should detect various forms of XOR encryption', () => { + const testCases = [ + `function encrypt(a, b) { return a ^= b; }`, + `const result = data ^ key;`, + `function xor(plaintext, key) { return plaintext ^ key; }`, + `return char ^ 0xFF;` + ]; + + testCases.forEach(testCode => { + const issues = analyzeCodeForCrypto(testCode); + const hasXORIssue = issues.some(issue => + issue.type === 'suspicious_operation' || + issue.message.toLowerCase().includes('xor') + ); + expect(hasXORIssue, `Failed to detect XOR in: ${testCode}`).to.be.true; + }); + }); + + it('should detect custom hash implementations', () => { + const testCode = ` + function customHash(input) { + let hash = 0; + for(let i = 0; i < input.length; i++) { + hash = ((hash << 5) - hash) + input.charCodeAt(i); + hash = hash & hash; + } + return hash; + } + `; + + const issues = analyzeCodeForCrypto(testCode); + expect(issues).to.have.lengthOf.at.least(1); + expect(issues.some(i => i.severity === 'high')).to.be.true; + }); + }); +}); \ No newline at end of file