mirror of
https://github.com/tiennm99/is-a-dev.git
synced 2026-05-29 14:19:11 +00:00
Refactors validations
This commit is contained in:
@@ -1,4 +1,4 @@
|
||||
const { getDomains, validateDomainData } = require('../utils/domain');
|
||||
const { getDomains } = require('../utils/domain');
|
||||
|
||||
describe('getDomains', () => {
|
||||
it('should resolve with the list of domains', async () => {
|
||||
@@ -7,67 +7,3 @@ describe('getDomains', () => {
|
||||
});
|
||||
});
|
||||
|
||||
const defaultDomain = {
|
||||
name: 'aaa',
|
||||
record: {
|
||||
A: ['121.121.121.121']
|
||||
},
|
||||
owner: {
|
||||
username: 'betsy',
|
||||
email: 'betsyfuckyoassup@foobar.com',
|
||||
},
|
||||
};
|
||||
|
||||
const getstroflen = len => Array(len).fill('a').join('');
|
||||
|
||||
describe('validateDomainData', () => {
|
||||
const invalidCases = [
|
||||
{},
|
||||
{ name: 'helo' },
|
||||
{ name: 'wwow', record: { A: ['12312'] } },
|
||||
...['', ' ', undefined, 'hlo wld', 'g32++13', 'ajsdD_123yq', 'khsda%', '122*dsd', getstroflen(101)].map(name => ({
|
||||
...defaultDomain,
|
||||
name,
|
||||
})),
|
||||
{ ...defaultDomain, record: { CNAME: 'sd', A: ['121,3213'] } },
|
||||
{ ...defaultDomain, record: { A: ['121', '12'], FOOBAR: ['sd'] } },
|
||||
{ ...defaultDomain, record: { A: [] } },
|
||||
{ ...defaultDomain, record: { A: ['11122'], URL: 'foobar' } },
|
||||
{ ...defaultDomain, owner: {}, },
|
||||
{ ...defaultDomain, owner: { username: 'hwelo', }, },
|
||||
{ ...defaultDomain, owner: { email: 'hwelo' }, },
|
||||
{ ...defaultDomain, record: { CNAME: 'http://foobar.com' } },
|
||||
{ ...defaultDomain, record: { CNAME: 'https://foobar.com' } },
|
||||
];
|
||||
|
||||
const validCases = [
|
||||
defaultDomain,
|
||||
...['hello', 'hello-world', '11111111111', '--wow--', 'wow--', '--wow'].map(name => ({
|
||||
...defaultDomain,
|
||||
name,
|
||||
})),
|
||||
{
|
||||
...defaultDomain,
|
||||
description: getstroflen(99),
|
||||
},
|
||||
{ ...defaultDomain, record: { CNAME: 'aa.sd', URL: '121,3213' } },
|
||||
];
|
||||
|
||||
it('should return false for invalid data', () => {
|
||||
invalidCases.forEach(data => {
|
||||
const { valid, errors } = validateDomainData(data);
|
||||
expect(valid).toBe(false);
|
||||
expect(errors.length).toBeGreaterThan(0);
|
||||
});
|
||||
});
|
||||
|
||||
it('should return true if the name is valid', () => {
|
||||
validCases.forEach(data => {
|
||||
const { valid, errors } = validateDomainData(data);
|
||||
if (!valid) console.log(errors);
|
||||
expect(valid).toBe(true);
|
||||
expect(errors).toEqual([]);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
const R = require('ramda');
|
||||
const fs = require('fs');
|
||||
const { getDomains, validateDomainData } = require('../utils/domain');
|
||||
const { getDomains } = require('../utils/domain');
|
||||
const { validateDomainData } = require('../utils/validations');
|
||||
const { DOMAINS_PATH } = require('../utils/constants');
|
||||
|
||||
describe('Domains', () => {
|
||||
|
||||
@@ -0,0 +1,66 @@
|
||||
const { validateDomainData } = require('../utils/validations');
|
||||
|
||||
const defaultDomain = {
|
||||
name: 'aaa',
|
||||
record: {
|
||||
A: ['121.121.121.121']
|
||||
},
|
||||
owner: {
|
||||
username: 'betsy',
|
||||
email: 'betsyfuckyoassup@foobar.com',
|
||||
},
|
||||
};
|
||||
|
||||
const getstroflen = len => Array(len).fill('a').join('');
|
||||
|
||||
describe('validateDomainData', () => {
|
||||
const invalidCases = [
|
||||
{},
|
||||
{ name: 'helo' },
|
||||
{ name: 'wwow', record: { A: ['12312'] } },
|
||||
...['', ' ', undefined, 'hlo wld', 'g32++13', 'ajsdD_123yq', 'khsda%', '122*dsd', getstroflen(101)].map(name => ({
|
||||
...defaultDomain,
|
||||
name,
|
||||
})),
|
||||
{ ...defaultDomain, record: { CNAME: 'sd', A: ['121,3213'] } },
|
||||
{ ...defaultDomain, record: { A: ['121', '12'], FOOBAR: ['sd'] } },
|
||||
{ ...defaultDomain, record: { A: [] } },
|
||||
{ ...defaultDomain, record: { A: ['11122'], URL: 'foobar' } },
|
||||
{ ...defaultDomain, owner: {}, },
|
||||
{ ...defaultDomain, owner: { username: 'hwelo', }, },
|
||||
{ ...defaultDomain, owner: { email: 'hwelo' }, },
|
||||
{ ...defaultDomain, record: { CNAME: 'http://foobar.com' } },
|
||||
{ ...defaultDomain, record: { CNAME: 'https://foobar.com' } },
|
||||
];
|
||||
|
||||
const validCases = [
|
||||
defaultDomain,
|
||||
...['hello', 'hello-world', '11111111111', '--wow--', 'wow--', '--wow'].map(name => ({
|
||||
...defaultDomain,
|
||||
name,
|
||||
})),
|
||||
{
|
||||
...defaultDomain,
|
||||
description: getstroflen(99),
|
||||
},
|
||||
{ ...defaultDomain, record: { CNAME: 'aa.sd', URL: '121,3213' } },
|
||||
];
|
||||
|
||||
it('should return false for invalid data', () => {
|
||||
invalidCases.forEach(data => {
|
||||
const { valid, errors } = validateDomainData(data);
|
||||
expect(valid).toBe(false);
|
||||
expect(errors.length).toBeGreaterThan(0);
|
||||
});
|
||||
});
|
||||
|
||||
it('should return true if the name is valid', () => {
|
||||
validCases.forEach(data => {
|
||||
const { valid, errors } = validateDomainData(data);
|
||||
if (!valid) console.log(errors);
|
||||
expect(valid).toBe(true);
|
||||
expect(errors).toEqual([]);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
+3
-68
@@ -1,13 +1,11 @@
|
||||
const fs = require('fs');
|
||||
const path = require('path');
|
||||
const R = require('ramda');
|
||||
const { VALID_RECORD_TYPES, DOMAINS_PATH } = require('./constants');
|
||||
|
||||
const log = m => x => console.log(m, x) || x;
|
||||
const { DOMAINS_PATH } = require('./constants');
|
||||
|
||||
const toDomain = str => path.join(DOMAINS_PATH, str);
|
||||
|
||||
const toDomainData = R.compose(require, toDomain);
|
||||
const toDomainData = R.compose(JSON.parse, R.toString, fs.readFileSync, toDomain);
|
||||
|
||||
const getDomains = () =>
|
||||
fs.promises.readdir(DOMAINS_PATH, {})
|
||||
@@ -16,67 +14,4 @@ const getDomains = () =>
|
||||
name: name.replace(/\.json$/, ''),
|
||||
})));
|
||||
|
||||
const between = (min, max) => num => num >= min && num <= max;
|
||||
const testRegex = regex => str => !!(str && str.match(regex));
|
||||
|
||||
const validate = pattern => data => R.compose(
|
||||
invalidPairs => invalidPairs.length ? { errors: invalidPairs, valid: false } : { errors: [], valid: true },
|
||||
R.filter(([key, { fn }]) => fn ? !fn(data[key]) : false),
|
||||
R.toPairs,
|
||||
)(pattern);
|
||||
|
||||
const or = R.anyPass;
|
||||
const and = R.allPass;
|
||||
|
||||
const validateCnameRecord = key => and([
|
||||
R.propSatisfies(R.is(String), key),
|
||||
R.compose(R.equals(1), R.length, R.reject(R.equals('URL')), R.keys),
|
||||
R.propSatisfies(R.compose(R.gte(R.__, 3), R.length), key),
|
||||
R.propSatisfies(R.complement(testRegex(/^https?:\/\//ig)), key),
|
||||
]);
|
||||
|
||||
const validateARecord = key => and([
|
||||
R.compose(R.equals(1), R.length, R.keys),
|
||||
R.propSatisfies(R.compose(R.gte(R.__, 1), R.length), key),
|
||||
]);
|
||||
|
||||
const validateDomainData = validate({
|
||||
name: {
|
||||
reason: 'The name of the file is invalid',
|
||||
fn: or([
|
||||
R.equals('@'),
|
||||
and([
|
||||
R.compose(between(2, 100), R.length),
|
||||
testRegex(/^[a-z0-9\-]+$/g),
|
||||
])
|
||||
]),
|
||||
},
|
||||
description: { reason: '', fn: R.T, },
|
||||
repo: { reason: '', fn: R.T, },
|
||||
owner: {
|
||||
reason: '`owner` needs username and email properties',
|
||||
fn: and([
|
||||
R.is(Object),
|
||||
R.complement(R.isEmpty),
|
||||
R.where({
|
||||
username: R.is(String),
|
||||
email: R.is(String),
|
||||
}),
|
||||
]),
|
||||
},
|
||||
record: {
|
||||
reason: 'Invalid record. CNAME records have to be a host name and A records has to be a list of ips',
|
||||
fn: and([
|
||||
R.is(Object),
|
||||
R.compose(R.isEmpty, R.flip(R.difference)(VALID_RECORD_TYPES), R.keys),
|
||||
R.cond([
|
||||
[R.has('CNAME'), validateCnameRecord('CNAME')],
|
||||
[R.has('A'), validateARecord('A')],
|
||||
[R.has('URL'), R.propSatisfies(R.is(String), 'URL')],
|
||||
[R.T, R.T],
|
||||
]),
|
||||
]),
|
||||
},
|
||||
});
|
||||
|
||||
module.exports = { getDomains, validateDomainData };
|
||||
module.exports = { getDomains };
|
||||
|
||||
@@ -0,0 +1,20 @@
|
||||
const R = require('ramda');
|
||||
|
||||
const log = m => x => console.log(m, x) || x;
|
||||
|
||||
const between = (min, max) => num => num >= min && num <= max;
|
||||
const testRegex = regex => str => !!(str && str.match(regex));
|
||||
|
||||
const validate = pattern => data => R.compose(
|
||||
invalidPairs => invalidPairs.length ? { errors: invalidPairs, valid: false } : { errors: [], valid: true },
|
||||
R.filter(([key, { fn }]) => fn ? !fn(data[key]) : false),
|
||||
R.toPairs,
|
||||
)(pattern);
|
||||
|
||||
const or = R.anyPass;
|
||||
const and = R.allPass;
|
||||
|
||||
const then = fn => p => p.then(fn);
|
||||
|
||||
module.exports = { or, and, validate, between, testRegex, log, then };
|
||||
|
||||
@@ -0,0 +1,56 @@
|
||||
const R = require('ramda');
|
||||
const { VALID_RECORD_TYPES } = require('./constants');
|
||||
const { or, and, validate, between, testRegex } = require('./helpers');
|
||||
|
||||
const validateCnameRecord = key => and([
|
||||
R.propSatisfies(R.is(String), key),
|
||||
R.compose(R.equals(1), R.length, R.reject(R.equals('URL')), R.keys),
|
||||
R.propSatisfies(R.compose(R.gte(R.__, 3), R.length), key),
|
||||
R.propSatisfies(R.complement(testRegex(/^https?:\/\//ig)), key),
|
||||
]);
|
||||
|
||||
const validateARecord = key => and([
|
||||
R.compose(R.equals(1), R.length, R.keys),
|
||||
R.propSatisfies(R.compose(R.gte(R.__, 1), R.length), key),
|
||||
]);
|
||||
|
||||
const validateDomainData = validate({
|
||||
name: {
|
||||
reason: 'The name of the file is invalid',
|
||||
fn: or([
|
||||
R.equals('@'),
|
||||
and([
|
||||
R.compose(between(2, 100), R.length),
|
||||
testRegex(/^[a-z0-9\-]+$/g),
|
||||
])
|
||||
]),
|
||||
},
|
||||
description: { reason: '', fn: R.T, },
|
||||
repo: { reason: '', fn: R.T, },
|
||||
owner: {
|
||||
reason: '`owner` needs username and email properties',
|
||||
fn: and([
|
||||
R.is(Object),
|
||||
R.complement(R.isEmpty),
|
||||
R.where({
|
||||
username: R.is(String),
|
||||
email: R.is(String),
|
||||
}),
|
||||
]),
|
||||
},
|
||||
record: {
|
||||
reason: 'Invalid record. CNAME records have to be a host name and A records has to be a list of ips',
|
||||
fn: and([
|
||||
R.is(Object),
|
||||
R.compose(R.isEmpty, R.flip(R.difference)(VALID_RECORD_TYPES), R.keys),
|
||||
R.cond([
|
||||
[R.has('CNAME'), validateCnameRecord('CNAME')],
|
||||
[R.has('A'), validateARecord('A')],
|
||||
[R.has('URL'), R.propSatisfies(R.is(String), 'URL')],
|
||||
[R.T, R.T],
|
||||
]),
|
||||
]),
|
||||
},
|
||||
});
|
||||
|
||||
module.exports = { validateDomainData };
|
||||
Reference in New Issue
Block a user