From 58cda9cc7fb79ca9df6746de7f9662bc08dc156a Mon Sep 17 00:00:00 2001 From: Stefan Malzner Date: Fri, 13 Oct 2017 12:29:40 +0200 Subject: initial commit --- src/helpers/password-helpers.js | 36 ++++++++++++++++++++++++ src/helpers/recipe-helpers.js | 39 ++++++++++++++++++++++++++ src/helpers/routing-helpers.js | 4 +++ src/helpers/validation-helpers.js | 48 ++++++++++++++++++++++++++++++++ src/helpers/webview-ime-focus-helpers.js | 38 +++++++++++++++++++++++++ 5 files changed, 165 insertions(+) create mode 100644 src/helpers/password-helpers.js create mode 100644 src/helpers/recipe-helpers.js create mode 100644 src/helpers/routing-helpers.js create mode 100644 src/helpers/validation-helpers.js create mode 100644 src/helpers/webview-ime-focus-helpers.js (limited to 'src/helpers') diff --git a/src/helpers/password-helpers.js b/src/helpers/password-helpers.js new file mode 100644 index 000000000..7aacaa4d0 --- /dev/null +++ b/src/helpers/password-helpers.js @@ -0,0 +1,36 @@ +import { SHA256 } from 'jshashes'; + +export function hash(password) { + return new SHA256().b64(password); +} + +export function scorePassword(password) { + let score = 0; + if (!password) { + return score; + } + + // award every unique letter until 5 repetitions + const letters = {}; + for (let i = 0; i < password.length; i += 1) { + letters[password[i]] = (letters[password[i]] || 0) + 1; + score += 5.0 / letters[password[i]]; + } + + // bonus points for mixing it up + const variations = { + digits: /\d/.test(password), + lower: /[a-z]/.test(password), + upper: /[A-Z]/.test(password), + nonWords: /\W/.test(password), + }; + + let variationCount = 0; + Object.keys(variations).forEach((key) => { + variationCount += (variations[key] === true) ? 1 : 0; + }); + + score += (variationCount - 1) * 10; + + return parseInt(score, 10); +} diff --git a/src/helpers/recipe-helpers.js b/src/helpers/recipe-helpers.js new file mode 100644 index 000000000..257e322fb --- /dev/null +++ b/src/helpers/recipe-helpers.js @@ -0,0 +1,39 @@ +import path from 'path'; +import { remote } from 'electron'; + +// import ServiceModel from '../models/Service'; + +const app = remote.app; + +export function getRecipeDirectory(id = '') { + return path.join(app.getPath('userData'), 'recipes', id); +} + +export function getDevRecipeDirectory(id = '') { + return path.join(app.getPath('userData'), 'recipes', 'dev', id); +} + +export function loadRecipeConfig(recipeId) { + try { + const configPath = `${recipeId}/package.json`; + // Delete module from cache + delete require.cache[require.resolve(configPath)]; + + // eslint-disable-next-line + let config = require(configPath); + + const moduleConfigPath = require.resolve(configPath); + const paths = path.parse(moduleConfigPath); + config.path = paths.dir; + + return config; + } catch (e) { + console.error(e); + return null; + } +} + +module.paths.unshift( + getDevRecipeDirectory(), + getRecipeDirectory(), +); diff --git a/src/helpers/routing-helpers.js b/src/helpers/routing-helpers.js new file mode 100644 index 000000000..14922ebf3 --- /dev/null +++ b/src/helpers/routing-helpers.js @@ -0,0 +1,4 @@ +import RouteParser from 'route-parser'; + +// eslint-disable-next-line +export const matchRoute = (pattern, path) => new RouteParser(pattern).match(path); diff --git a/src/helpers/validation-helpers.js b/src/helpers/validation-helpers.js new file mode 100644 index 000000000..eeb12cab7 --- /dev/null +++ b/src/helpers/validation-helpers.js @@ -0,0 +1,48 @@ +export function required({ field }) { + const isValid = (field.value.trim() !== ''); + return [isValid, `${field.label} is required`]; +} + +export function email({ field }) { + const value = field.value.trim(); + let isValid = false; + + if (value !== '') { + isValid = Boolean(value.match(/[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,63}/i)); + } else { + isValid = true; + } + + return [isValid, `${field.label} is not a valid email address`]; +} + +export function url({ field }) { + const value = field.value.trim(); + let isValid = false; + + if (value !== '') { + // eslint-disable-next-line + isValid = Boolean(value.match(/(^|[\s.:;?\-\]<(])(https?:\/\/[-\w;/?:@&=+$|_.!~*|'()[\]%#,☺]+[\w/#](\(\))?)(?=$|[\s',|().:;?\-[\]>)])/i)); + } else { + isValid = true; + } + + return [isValid, `${field.label} is not a valid url`]; +} + +export function minLength(length) { + return ({ field }) => { + let isValid = true; + if (field.touched) { + isValid = field.value.length >= length; + } + return [isValid, `${field.label} should be at least ${length} characters long.`]; + }; +} + +export function oneRequired(targets) { + return ({ field, form }) => { + const invalidFields = targets.filter(target => form.$(target).value === ''); + return [targets.length !== invalidFields.length, `${field.label} is required`]; + }; +} diff --git a/src/helpers/webview-ime-focus-helpers.js b/src/helpers/webview-ime-focus-helpers.js new file mode 100644 index 000000000..2593a5f26 --- /dev/null +++ b/src/helpers/webview-ime-focus-helpers.js @@ -0,0 +1,38 @@ +module.exports.releaseDocumentFocus = () => { + const element = document.createElement('span'); + document.body.appendChild(element); + + const range = document.createRange(); + range.setStart(element, 0); + + const selection = window.getSelection(); + selection.removeAllRanges(); + selection.addRange(range); + selection.removeAllRanges(); + + document.body.removeChild(element); +}; + +module.exports.claimDocumentFocus = () => { + const { activeElement } = document; + const selection = window.getSelection(); + + let selectionStart; + let selectionEnd; + let range; + + if (activeElement) ({ selectionStart, selectionEnd } = activeElement); + if (selection.rangeCount) range = selection.getRangeAt(0); + + const restoreOriginalSelection = () => { + if (selectionStart >= 0 && selectionEnd >= 0) { + activeElement.selectionStart = selectionStart; + activeElement.selectionEnd = selectionEnd; + } else if (range) { + selection.addRange(range); + } + }; + + exports.releaseDocumentFocus(); + window.requestAnimationFrame(restoreOriginalSelection); +}; -- cgit v1.2.3-70-g09d2