From de972e0710623443618bacf51c8e6dea9c918dc2 Mon Sep 17 00:00:00 2001 From: vipinpaul Date: Wed, 11 Mar 2020 16:14:19 +0530 Subject: [PATCH 1/4] Adding auto-backup feature --- src/components/Settings.js | 64 ++++++++++++-- src/translations/en.js | 4 + src/util/auto_export.js | 177 +++++++++++++++++++++++++++++++++++++ 3 files changed, 236 insertions(+), 9 deletions(-) create mode 100644 src/util/auto_export.js diff --git a/src/components/Settings.js b/src/components/Settings.js index d9ccf71..3382c65 100644 --- a/src/components/Settings.js +++ b/src/components/Settings.js @@ -19,6 +19,7 @@ import ExpansionPanelSummary from "@material-ui/core/ExpansionPanelSummary"; import ExpansionPanelDetails from "@material-ui/core/ExpansionPanelDetails"; import Typography from "@material-ui/core/Typography"; import ExpandMoreIcon from "@material-ui/icons/ExpandMore"; +import * as auto_export from "../util/auto_export"; const numberFormat = require("../util/getNumberFormat") const { dialog, getCurrentWindow } = require('electron').remote; @@ -47,7 +48,8 @@ class SettingsModal extends React.Component { langCodeValue: "", langCode: "", langVersion: "", - folderPath: "" + folderPath: "", + backupFrequency: "" }, refSetting: { bibleName: "", @@ -116,10 +118,14 @@ class SettingsModal extends React.Component { settingData.langCode = doc.targetLang; settingData.langCodeValue = doc.targetLang; settingData.langVersion = doc.targetVersion; - settingData.folderPath = doc.targetPath; - }, (err) => { + settingData.folderPath = doc.targetPath; + settingData.backupFrequency = (doc.backupFrequency) === undefined ? "daily" : doc.backupFrequency; + if (settingData.langCodeValue && settingData.langVersion && settingData.backupFrequency !== "none") { + auto_export.autoBackUp(); + } + }, (err) => { // console.log(err); - }); + }); } loadReference = () => { @@ -241,14 +247,15 @@ class SettingsModal extends React.Component { saveSetting = () => { if (this.target_setting() === false) return; - const currentTrans = AutographaStore.currentTrans; - const {langCode, langVersion, folderPath} = this.state.settingData; + const currentTrans = AutographaStore.currentTrans; + const {langCode, langVersion, folderPath, backupFrequency} = this.state.settingData; const settingData = { _id: 'targetBible', targetLang: langCode.toLowerCase(), targetVersion: langVersion, targetPath: folderPath, - langScript: AutographaStore.scriptDirection.toUpperCase() + langScript: AutographaStore.scriptDirection.toUpperCase(), + backupFrequency: backupFrequency } db.get('targetBible').then((doc) => { settingData._rev = doc._rev; @@ -261,7 +268,11 @@ class SettingsModal extends React.Component { }, (err) => { swal(currentTrans["dynamic-msg-trans-data"], currentTrans["dynamic-msg-went-wrong"], "success"); }); - }); + }); + + if (langCode && langVersion && backupFrequency !== "none") { + auto_export.autoBackUp(); + } } openFileDialogSettingData = (event) => { @@ -752,6 +763,9 @@ class SettingsModal extends React.Component { onChangeScriptDir = (value) => { AutographaStore.scriptDirection = value; + } + onChangeBackupSetting = (value) => { + this.setState({ settingData: { ...this.state.settingData, backupFrequency: value} }); } onChangeRefScriptDir = (value) => { AutographaStore.refScriptDirection = value; @@ -945,7 +959,7 @@ class SettingsModal extends React.Component { let closeSetting = () => AutographaStore.showModalSettings = false const { show } = this.props; - const { langCodeValue, langVersion, folderPath } = this.state.settingData; + const { langCodeValue, langVersion, folderPath, backupFrequency } = this.state.settingData; const { bibleName, refVersion, refLangCode, refFolderPath } = this.state.refSetting; const listCode = this.state._listArray; @@ -1083,6 +1097,38 @@ class SettingsModal extends React.Component { /> +
+ + this.onChangeBackupSetting(value)} + > + } + style={{width: "70%"}} + /> + } + style={{width: "70%", marginLeft: "25px"}} + /> + } + style={{width: "70%", marginLeft: "25px"}} + /> + +
+ { (message)=> { + try { + // Initial Backup call + callBackUp(); + const delay = 60000 * 10; // 10 minutes + setInterval(async () => { + callBackUp(); + }, delay); + } catch(e) { + console.log(e); + } +} + +async function callBackUp() { + let targetLangDoc = await db.get('targetBible'); + const outputPath = targetLangDoc.targetPath; + const directory = path.join(Array.isArray(outputPath) ? outputPath[0] : outputPath,"auto-backup"); + const nameElements = [ + targetLangDoc && targetLangDoc.targetLang, + targetLangDoc && targetLangDoc.targetVersion + ].filter(Boolean); + const foldername = nameElements.join("_"); + const dirPath = path.join(directory, foldername); + if (!fs.existsSync(dirPath)){ + fs.mkdirSync(dirPath, { recursive: true }); + } + await fs.readdir(dirPath, (err, folders) => { + if (folders.length >= 1) { + let folderName = (folders[(folders.length)-1]).split('_'); + var d = new Date(); + let timeStamp = [ d && getTimeStamp(d) ].filter(Boolean); + let currentTime = timeStamp[0].split('_'); + let days = calculateHours(currentTime,folderName); + if (days >= 1 && (targetLangDoc.backupFrequency).toLowerCase() === "daily") { + buildFilePath(new Date(), dirPath); + folderHandling (folders, dirPath); + } else if (days >= 7 && (targetLangDoc.backupFrequency).toLowerCase() === "weekly") { + buildFilePath(new Date(), dirPath); + folderHandling (folders, dirPath); + } else { + return null; + } + + } else { + buildFilePath(new Date(), dirPath); + } + }); +} + +async function allBooks(dir) { + let doc = await db.get('targetBible'); + constants.bookCodeList.forEach(async(value, index) => { + let book={}; + book.bookNumber = (index+1).toString(); + book.bookName = AutographaStore.editBookNamesMode ? AutographaStore.translatedBookNames[index] : constants.booksList[index]; + book.bookCode = value; + book.outputPath = doc.targetPath; + await toUsfm(book, dir); + }) +} + +export const toUsfm = async (book, dir) => { + const usfmDoc = await toUsfmDoc(book, false); + const filename = (book.bookCode) + '.usfm'; + const filePath = path.join(dir, filename); + writeUsfm(usfmDoc, filePath); +}; + +async function toUsfmDoc(book, returnNullForEmptyBook=false) { + try { + const usfmContent = []; + var isEmpty = true; + usfmContent.push('\\id ' + book.bookCode); + usfmContent.push('\\mt ' + book.bookName); + const doc = await db.get(book.bookNumber); + for (const chapter of doc.chapters) { + usfmContent.push('\n\\c ' + chapter.chapter); + usfmContent.push('\\p'); + let i = 0; + let verseNumber; + let verses; + for (const verse of chapter.verses) { + i = i + 1; + if (i < (chapter.verses).length && chapter.verses[i].joint_verse) { + // Finding out the join verses and get their verse number(s) + verseNumber = chapter.verses[i].joint_verse + "-" + chapter.verses[i].verse_number; + verses = chapter.verses[(chapter.verses[i].joint_verse)-1].verse; + continue; + } else { + if (verseNumber) { + // Push join verse number (1-3) and content. + let newVerse = addMtag(verses); + usfmContent.push('\\v ' + verseNumber + ' ' + newVerse); + verseNumber = undefined; + verses = undefined; + } else { + // Push verse number and content. + let newVerse = addMtag(verse.verse); + usfmContent.push('\\v ' + verse.verse_number + ' ' + newVerse); + } + isEmpty = isEmpty && !verse.verse; + } + } + } + return (returnNullForEmptyBook && isEmpty) + ? null + : usfmContent.join('\n'); + } catch(err) { + console.log(err); + } +} + +function addMtag(verses) { + let newVerse = verses; + if(verses.indexOf('\n') !== -1 ){ + newVerse = verses.trim().replace(new RegExp(/[\n\r]/, 'gu'), '\n\\m ') + verses = newVerse + } + return newVerse; +} + +async function buildFilePath(date, dirPath) { + const dateElement = [ date && getTimeStamp(date) ].filter(Boolean); + const dir = path.join(dirPath, dateElement[0]); + if (!fs.existsSync(dir)){ + fs.mkdirSync(dir, { recursive: true }); + } + allBooks(dir); +} + +function folderHandling (folders, dirPath) { + if (folders.length >= 5) { + var files = fs.readdirSync(path.join(dirPath, folders[0])); + files.forEach(file => { + fs.unlinkSync(path.join(dirPath, folders[0], file)); + }); + fs.rmdir(path.join(dirPath, folders[0]), function(err, result) { + if(err) console.log('error', err); + }); + } +} + +function writeUsfm(usfmDoc, filePath) { + if (usfmDoc && filePath) { + fs.writeFileSync(filePath, usfmDoc, 'utf8', function(err, result) { + if(err) console.log('error', err); + }); + } +} + +function getTimeStamp(date) { + var months = ["January","February","March","April","May","June","July","August","September","October","November","December"]; + var year = date.getFullYear(), + month = ((date.getMonth() + 1) < 10 ? '0' : '') + (date.getMonth() + 1), + day = (date.getDate() < 10 ? '0' : '') + date.getDate(), + hour = (date.getHours() < 10 ? '0' : '') + date.getHours(), + minute = (date.getMinutes() < 10 ? '0' : '') + date.getMinutes(), + second = (date.getSeconds() < 10 ? '0' : '') + date.getSeconds(); + + return (months[parseInt(month)-1] + " " + day + "-" + year.toString() + "_" + hour + ":" + minute + ":" + second).toString(); +} + +function calculateHours(currDate, preDate) + { + var currTime = new Date(currDate[0] + " " + currDate[1]); + var preTime = new Date(preDate[0] + " " + preDate[1]); + var difference = Math.abs(currTime.getTime() - preTime.getTime()) / 1000; + // var hour = Math.floor(difference / 3600) % 24; + var days = Math.floor(difference / 86400); + return days; + } From 9c679c9327506aee788e01929f1d59d1740cf3c3 Mon Sep 17 00:00:00 2001 From: vipinpaul Date: Fri, 13 Mar 2020 17:48:33 +0530 Subject: [PATCH 2/4] Backup updated to days (number of days) --- src/translations/en.js | 2 +- src/util/auto_export.js | 20 +++++++++++--------- 2 files changed, 12 insertions(+), 10 deletions(-) diff --git a/src/translations/en.js b/src/translations/en.js index e0bf547..7bd5ee9 100644 --- a/src/translations/en.js +++ b/src/translations/en.js @@ -119,7 +119,7 @@ export default { "label-incomplete-verses" : "Possible Incomplete Verses", "label-multiple-spaces" : "Verses with Multiple Spaces", "label-statistic-heading": "Statistics for the Book of", - "label-auto-backup": "Auto Backup", + "label-auto-backup": "Auto-backup", "label-none": "None", "label-daily": "Daily", "label-weekly": "Weekly", diff --git a/src/util/auto_export.js b/src/util/auto_export.js index f607f14..2e8423c 100644 --- a/src/util/auto_export.js +++ b/src/util/auto_export.js @@ -4,20 +4,21 @@ const fs = require("fs"); const constants = require("../util/constants"); const db = require(`${__dirname}/data-provider`).targetDb(); -export const autoBackUp = async () => { +export const initializeBackUp = async () => { try { // Initial Backup call - callBackUp(); - const delay = 60000 * 10; // 10 minutes + backUp(); + const delay = 60000 * 2; // 2 minutes setInterval(async () => { - callBackUp(); + backUp(); }, delay); } catch(e) { console.log(e); } } -async function callBackUp() { +async function backUp() { + console.log("backUp"); let targetLangDoc = await db.get('targetBible'); const outputPath = targetLangDoc.targetPath; const directory = path.join(Array.isArray(outputPath) ? outputPath[0] : outputPath,"auto-backup"); @@ -36,7 +37,7 @@ async function callBackUp() { var d = new Date(); let timeStamp = [ d && getTimeStamp(d) ].filter(Boolean); let currentTime = timeStamp[0].split('_'); - let days = calculateHours(currentTime,folderName); + let days = calculateDays(currentTime,folderName); if (days >= 1 && (targetLangDoc.backupFrequency).toLowerCase() === "daily") { buildFilePath(new Date(), dirPath); folderHandling (folders, dirPath); @@ -53,7 +54,7 @@ async function callBackUp() { }); } -async function allBooks(dir) { +async function exportAllBooks(dir) { let doc = await db.get('targetBible'); constants.bookCodeList.forEach(async(value, index) => { let book={}; @@ -131,7 +132,7 @@ async function buildFilePath(date, dirPath) { if (!fs.existsSync(dir)){ fs.mkdirSync(dir, { recursive: true }); } - allBooks(dir); + exportAllBooks(dir); } function folderHandling (folders, dirPath) { @@ -166,12 +167,13 @@ function getTimeStamp(date) { return (months[parseInt(month)-1] + " " + day + "-" + year.toString() + "_" + hour + ":" + minute + ":" + second).toString(); } -function calculateHours(currDate, preDate) +function calculateDays(currDate, preDate) { var currTime = new Date(currDate[0] + " " + currDate[1]); var preTime = new Date(preDate[0] + " " + preDate[1]); var difference = Math.abs(currTime.getTime() - preTime.getTime()) / 1000; // var hour = Math.floor(difference / 3600) % 24; + // var minutes = Math.floor(difference / 60) % 60; var days = Math.floor(difference / 86400); return days; } From fdfc90720f96703d5a9179339eb02c656ffc620d Mon Sep 17 00:00:00 2001 From: vipinpaul Date: Mon, 16 Mar 2020 11:25:59 +0530 Subject: [PATCH 3/4] Added translations for Auto Backup Labels --- src/components/Settings.js | 4 ++-- src/translations/ar.js | 4 ++++ src/translations/es.js | 4 ++++ src/translations/hi.js | 4 ++++ src/translations/pt.js | 4 ++++ src/util/auto_export.js | 3 +-- 6 files changed, 19 insertions(+), 4 deletions(-) diff --git a/src/components/Settings.js b/src/components/Settings.js index 3382c65..098cd4b 100644 --- a/src/components/Settings.js +++ b/src/components/Settings.js @@ -121,7 +121,7 @@ class SettingsModal extends React.Component { settingData.folderPath = doc.targetPath; settingData.backupFrequency = (doc.backupFrequency) === undefined ? "daily" : doc.backupFrequency; if (settingData.langCodeValue && settingData.langVersion && settingData.backupFrequency !== "none") { - auto_export.autoBackUp(); + auto_export.initializeBackUp(); } }, (err) => { // console.log(err); @@ -271,7 +271,7 @@ class SettingsModal extends React.Component { }); if (langCode && langVersion && backupFrequency !== "none") { - auto_export.autoBackUp(); + auto_export.initializeBackUp(); } } diff --git a/src/translations/ar.js b/src/translations/ar.js index 3674695..2325e23 100644 --- a/src/translations/ar.js +++ b/src/translations/ar.js @@ -43,6 +43,10 @@ export default { "label-pwd" : "كلمه السر", "label-credentials": "شهاداته", "label-no-project" : "لا توجد مشاريع لإظهارها.", + "label-auto-backup": "لصناعة السيارات في النسخ الاحتياطي", + "label-none": "لا يوجد", + "label-daily": "اليومي", + "label-weekly": "أسبوعي", "placeholder-search-text" : "البحث عن النص", "placeholder-replace-text" : "إستبدال", "placeholder-path-to-destination" : "مسار إلى مجلد الوجهة", diff --git a/src/translations/es.js b/src/translations/es.js index 5173418..d798d0a 100644 --- a/src/translations/es.js +++ b/src/translations/es.js @@ -38,6 +38,10 @@ export default { "label-unjoin-this-verse": "Desunir este verso", "label-join-to-preceding-verse": "Únete al verso anterior", "label-joint-with-the-preceding-verse(s)" : "----- Conjunta con los versos anteriores -----", + "label-auto-backup": "copia de seguridad automática", + "label-none": "Ninguno", + "label-daily": "Diario", + "label-weekly": "Semanal", "placeholder-search-text" : "Buscar texto", "placeholder-replace-text" : "Reemplazo", "placeholder-path-to-destination" : "Ruta a la carpeta de destino", diff --git a/src/translations/hi.js b/src/translations/hi.js index b76824a..76712fc 100644 --- a/src/translations/hi.js +++ b/src/translations/hi.js @@ -38,6 +38,10 @@ export default { "label-unjoin-this-verse": "आयत को अलग करें", "label-join-to-preceding-verse": "पिछले आयत से जोड़े", "label-joint-with-the-preceding-verse(s)" : "----- पिछले आयत(तों) के साथ जोड़े -----", + "label-auto-backup": "ऑटो बैकअप", + "label-none": "नहीं", + "label-daily": "दैनिक", + "label-weekly": "साप्ताहिक", "placeholder-search-text" : "विषय ढूंढें", "placeholder-replace-text" : "प्रतिस्थापन", "placeholder-path-to-destination" : "लक्षित फोल्डर की ओर मार्ग", diff --git a/src/translations/pt.js b/src/translations/pt.js index ce42dde..35698d3 100644 --- a/src/translations/pt.js +++ b/src/translations/pt.js @@ -37,6 +37,10 @@ export default { "label-unjoin-this-verse": "Desunir este verso", "label-join-to-preceding-verse": "Junte-se ao verso anterior", "label-joint-with-the-preceding-verse(s)" : "----- Juntar-se ao (s) versículo (s) -----", + "label-auto-backup": "backup automático", + "label-none": "Nenhum", + "label-daily": "Diariamente", + "label-weekly": "Semanal", "placeholder-search-text" : "Buscar Texto", "placeholder-replace-text" : "Substituição", "placeholder-path-to-destination" : "Localização da pasta de destino", diff --git a/src/util/auto_export.js b/src/util/auto_export.js index 2e8423c..321c4b8 100644 --- a/src/util/auto_export.js +++ b/src/util/auto_export.js @@ -8,7 +8,7 @@ export const initializeBackUp = async () => { try { // Initial Backup call backUp(); - const delay = 60000 * 2; // 2 minutes + const delay = 60000 * 60; // 60 minutes/1 hour setInterval(async () => { backUp(); }, delay); @@ -18,7 +18,6 @@ export const initializeBackUp = async () => { } async function backUp() { - console.log("backUp"); let targetLangDoc = await db.get('targetBible'); const outputPath = targetLangDoc.targetPath; const directory = path.join(Array.isArray(outputPath) ? outputPath[0] : outputPath,"auto-backup"); From e0984b2ec69bb31065319913a132c61d26a8f093 Mon Sep 17 00:00:00 2001 From: vipinpaul Date: Mon, 16 Mar 2020 11:45:26 +0530 Subject: [PATCH 4/4] Changed padding for the radio buttons div --- src/components/Settings.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/Settings.js b/src/components/Settings.js index 098cd4b..2ce4e06 100644 --- a/src/components/Settings.js +++ b/src/components/Settings.js @@ -1071,7 +1071,7 @@ class SettingsModal extends React.Component { } -
+