refactoring locales management

This commit is contained in:
les
2020-02-20 18:37:10 +01:00
parent 0416ddb1a3
commit 5769d1a3ed
18 changed files with 68 additions and 55 deletions

View File

@@ -11,8 +11,9 @@
span.float-left {{timezone.value}} span.float-left {{timezone.value}}
small.float-right.text-danger {{timezone.offset}} small.float-right.text-danger {{timezone.offset}}
el-form-item(:label="$t('common.title')") div.mt-4 {{$t('admin.instance_locale')}}
el-input(v-model='title' @blur='save("title", title)') el-select(v-model='instance_locale')
el-option(v-for='locale in Object.keys(locales)' :key='locale' :label='locales[locale]' :value='locale')
el-form-item(:label="$t('common.description')") el-form-item(:label="$t('common.description')")
el-input(v-model='description' @blur='save("description", description)') el-input(v-model='description' @blur='save("description", description)')
@@ -45,6 +46,7 @@
import { mapActions, mapState } from 'vuex' import { mapActions, mapState } from 'vuex'
import moment from 'moment-timezone' import moment from 'moment-timezone'
import _ from 'lodash' import _ from 'lodash'
import locales from '../../locales/esm'
export default { export default {
name: 'Settings', name: 'Settings',
@@ -52,12 +54,16 @@ export default {
return { return {
queryTz: '', queryTz: '',
title: $store.state.settings.title, title: $store.state.settings.title,
description: $store.state.settings.description description: $store.state.settings.description,
locales
} }
}, },
computed: { computed: {
...mapState(['settings']), ...mapState(['settings']),
...mapState(['settings']), instance_locale: {
get () { return this.settings.instance_locale },
set (value) { this.setSetting({ key: 'instance_locale', value }) }
},
instance_timezone: { instance_timezone: {
get () { return this.settings.instance_timezone }, get () { return this.settings.instance_timezone },
set (value) { this.setSetting({ key: 'instance_timezone', value }) } set (value) { this.setSetting({ key: 'instance_timezone', value }) }
@@ -87,7 +93,6 @@ export default {
.unshift(current_timezone) .unshift(current_timezone)
.map(tz => ({ value: tz, offset: moment().tz(tz).format('z Z') })) .map(tz => ({ value: tz, offset: moment().tz(tz).format('z Z') }))
.value() .value()
console.error(ret)
return ret return ret
} }
}, },

6
locales/esm.js Normal file
View File

@@ -0,0 +1,6 @@
export default {
it: 'Italiano',
en: 'English',
es: 'Español',
ca: 'Català'
}

View File

@@ -1,8 +1,6 @@
import it from './it.json' module.exports = {
import en from './en.json' it: 'Italiano',
import es from './es.json' en: 'English',
import ca from './ca.json' es: 'Español',
ca: 'Català'
export default {
it, en, es, ca
} }

View File

@@ -181,7 +181,8 @@
"user_block_confirm": "Sicura di voler bloccare l'utente?", "user_block_confirm": "Sicura di voler bloccare l'utente?",
"delete_announcement_confirm": "Sicura di voler eliminare l'annuncio?", "delete_announcement_confirm": "Sicura di voler eliminare l'annuncio?",
"announcement_remove_ok": "Annuncio rimosso", "announcement_remove_ok": "Annuncio rimosso",
"announcement_description": "In questa sezione puoi inserire annunci che rimarranno in homepage" "announcement_description": "In questa sezione puoi inserire annunci che rimarranno in homepage",
"instance_locale": "Lingua di default",
}, },
"auth": { "auth": {
"not_confirmed": "Non abbiamo ancora confermato questa mail...", "not_confirmed": "Non abbiamo ancora confermato questa mail...",

View File

@@ -34,12 +34,12 @@ module.exports = {
** Plugins to load before mounting the App ** Plugins to load before mounting the App
*/ */
plugins: [ plugins: [
'@/plugins/i18n.js',
'@/plugins/element-ui', // UI library -> https://element.eleme.io/#/en-US/ '@/plugins/element-ui', // UI library -> https://element.eleme.io/#/en-US/
'@/plugins/filters', // text filters, datetime, etc. '@/plugins/filters', // text filters, datetime, etc.
'@/plugins/vue-awesome', // icon '@/plugins/vue-awesome', // icon
'@/plugins/axios', // axios baseurl configuration '@/plugins/axios', // axios baseurl configuration
{ src: '@/plugins/v-calendar', ssr: false }, // calendar, fix ssr { src: '@/plugins/v-calendar', ssr: false } // calendar, fix ssr
'@/plugins/i18n.js'
], ],
render: { render: {

View File

@@ -7,17 +7,9 @@ import {
Container, Footer, Timeline, TimelineItem, Menu, MenuItem, Carousel, CarouselItem Container, Footer, Timeline, TimelineItem, Menu, MenuItem, Carousel, CarouselItem
} from 'element-ui' } from 'element-ui'
import locale from 'element-ui/lib/locale' import locale from 'element-ui/lib/locale'
// import '../assets/style.scss'
const locales = {
it: require('element-ui/lib/locale/lang/it'),
en: require('element-ui/lib/locale/lang/en'),
es: require('element-ui/lib/locale/lang/es'),
ca: require('element-ui/lib/locale/lang/ca')
}
export default ({ app, store }) => { export default ({ app, store }) => {
locale.use(locales[store.state.locale]) locale.use(require(`element-ui/lib/locale/lang/${store.state.locale}`))
Vue.use(Button) Vue.use(Button)
Vue.use(Carousel) Vue.use(Carousel)
Vue.use(CarouselItem) Vue.use(CarouselItem)

View File

@@ -1,7 +1,6 @@
import Vue from 'vue' import Vue from 'vue'
import VueI18n from 'vue-i18n' import VueI18n from 'vue-i18n'
import merge from 'lodash/merge' import merge from 'lodash/merge'
import messages from '../locales'
Vue.use(VueI18n) Vue.use(VueI18n)
@@ -11,6 +10,14 @@ export default ({ app, store, req }) => {
if (req.settings.user_locale) { store.commit('setUserLocale', req.settings.user_locale) } if (req.settings.user_locale) { store.commit('setUserLocale', req.settings.user_locale) }
} }
const messages = {}
messages[store.state.locale] = require(`../locales/${store.state.locale}.json`)
// always include en fallback locale
if (store.state.locale !== 'en') {
messages.en = require('../locales/en.json')
}
if (store.state.user_locale) { if (store.state.user_locale) {
merge(messages[store.state.locale], store.state.user_locale) merge(messages[store.state.locale], store.state.user_locale)
} }

View File

@@ -272,12 +272,9 @@ const eventController = {
// return created event to the client // return created event to the client
res.json(event) res.json(event)
// send notification (mastodon/email) // send notifications (mastodon / email)
// only if user is authenticated const notifier = require('../../notifier')
if (req.user && !event.recurrent) { notifier.notifyEvent('Create', event.id)
const notifier = require('../../notifier')
notifier.notifyEvent('Create', event.id)
}
} catch (e) { } catch (e) {
res.sendStatus(400) res.sendStatus(400)
debug(e) debug(e)

View File

@@ -11,6 +11,7 @@ const generateKeyPair = util.promisify(crypto.generateKeyPair)
const defaultSettings = { const defaultSettings = {
instance_timezone: 'Europe/Rome', instance_timezone: 'Europe/Rome',
instance_locale: 'en',
instance_name: config.title.toLowerCase().replace(/ /g, ''), instance_name: config.title.toLowerCase().replace(/ /g, ''),
allow_registration: true, allow_registration: true,
allow_anon_event: true, allow_anon_event: true,

View File

@@ -100,7 +100,7 @@ const userController = {
const user = await User.create(req.body) const user = await User.create(req.body)
debug(`Sending registration email to ${user.email}`) debug(`Sending registration email to ${user.email}`)
mail.send(user.email, 'register', { user, config }, req.settings.locale) mail.send(user.email, 'register', { user, config }, req.settings.locale)
mail.send(config.admin_email, 'admin_register', { user, config }, req.settings.locale) mail.send(config.admin_email, 'admin_register', { user, config })
res.sendStatus(200) res.sendStatus(200)
} catch (e) { } catch (e) {
res.status(404).json(e) res.status(404).json(e)

View File

@@ -2,12 +2,13 @@ const Email = require('email-templates')
const path = require('path') const path = require('path')
const moment = require('moment-timezone') const moment = require('moment-timezone')
const config = require('config') const config = require('config')
const settings = require('./controller/settings') const settingsController = require('./controller/settings')
const debug = require('debug')('email') const debug = require('debug')('email')
const { Task, TaskManager } = require('../taskManager') const { Task, TaskManager } = require('../taskManager')
const locales = require('../../locales')
const mail = { const mail = {
send (addresses, template, locals, locale) { send (addresses, template, locals, locale = settingsController.settings.instance_locale) {
const task = new Task({ const task = new Task({
name: 'MAIL', name: 'MAIL',
removable: true, removable: true,
@@ -17,8 +18,8 @@ const mail = {
TaskManager.add(task) TaskManager.add(task)
}, },
_send (addresses, template, locales, locale) { _send (addresses, template, locals, locale) {
debug(`Send ${template} email to ${addresses}`) debug(`Send ${template} email to ${addresses} with locale ${locale}`)
const email = new Email({ const email = new Email({
views: { root: path.join(__dirname, '..', 'emails') }, views: { root: path.join(__dirname, '..', 'emails') },
htmlToText: true, htmlToText: true,
@@ -38,12 +39,13 @@ const mail = {
objectNotation: true, objectNotation: true,
syncFiles: false, syncFiles: false,
updateFiles: false, updateFiles: false,
defaultLocale: settings.locale, defaultLocale: settingsController.settings.instance_locale || 'en',
locale: settings.locale, locale,
locales: ['it', 'es', 'en', 'ca'] locales: Object.keys(locales)
}, },
transport: config.smtp transport: config.smtp
}) })
const msg = { const msg = {
template, template,
message: { message: {
@@ -51,10 +53,10 @@ const mail = {
bcc: config.admin_email bcc: config.admin_email
}, },
locals: { locals: {
...locales, ...locals,
locale, locale,
config: { title: config.title, baseurl: config.baseurl, description: config.description, admin_email: config.admin_email }, config: { title: config.title, baseurl: config.baseurl, description: config.description, admin_email: config.admin_email },
datetime: datetime => moment.unix(datetime).format('ddd, D MMMM HH:mm') datetime: datetime => moment.unix(datetime).locale(locale).format('ddd, D MMMM HH:mm')
} }
} }
return email.send(msg) return email.send(msg)

View File

@@ -1,5 +1,6 @@
const config = require('config') const config = require('config')
const moment = require('moment-timezone') const moment = require('moment-timezone')
const settingsController = require('../controller/settings')
// const debug = require('debug')('event:modals') // const debug = require('debug')('event:modals')
module.exports = (sequelize, DataTypes) => { module.exports = (sequelize, DataTypes) => {
@@ -39,7 +40,7 @@ module.exports = (sequelize, DataTypes) => {
Event.belongsTo(models.event, { as: 'parent' }) Event.belongsTo(models.event, { as: 'parent' })
} }
Event.prototype.toNoteAP = function (username, follower = []) { Event.prototype.toNoteAP = function (username, locale, follower = []) {
const tags = this.tags && this.tags.map(t => t.tag.replace(/[ #]/g, '_')) const tags = this.tags && this.tags.map(t => t.tag.replace(/[ #]/g, '_'))
const tag_links = tags.map(t => { const tag_links = tags.map(t => {
return `<a href='/tags/${t}' class='mention hashtag status-link' rel='tag'><span>#${t}</span></a>` return `<a href='/tags/${t}' class='mention hashtag status-link' rel='tag'><span>#${t}</span></a>`
@@ -47,8 +48,8 @@ module.exports = (sequelize, DataTypes) => {
const content = `<a href='${config.baseurl}/event/${this.id}'>${this.title}</a><br/> const content = `<a href='${config.baseurl}/event/${this.id}'>${this.title}</a><br/>
📍 ${this.place.name}<br/> 📍 ${this.place.name}<br/>
📅 ${moment.unix(this.start_datetime).format('dddd, D MMMM (HH:mm)')}<br/><br/> 📅 ${moment.unix(this.start_datetime).locale(locale).format('dddd, D MMMM (HH:mm)')}<br/><br/>
${this.description.length > 300 ? this.description.substr(0, 300) + '...' : this.description}<br/> ${this.description.length > 500 ? this.description.substr(0, 500) + '...' : this.description}<br/>
${tag_links} <br/>` ${tag_links} <br/>`
const attachment = [] const attachment = []

View File

@@ -80,7 +80,9 @@ const Helpers = {
to: recipients[sharedInbox], to: recipients[sharedInbox],
cc: ['https://www.w3.org/ns/activitystreams#Public', `${config.baseurl}/federation/u/${settingsController.settings.instance_name}/followers`], cc: ['https://www.w3.org/ns/activitystreams#Public', `${config.baseurl}/federation/u/${settingsController.settings.instance_name}/followers`],
actor: `${config.baseurl}/federation/u/${settingsController.settings.instance_name}`, actor: `${config.baseurl}/federation/u/${settingsController.settings.instance_name}`,
object: event.toNoteAP(settingsController.settings.instance_name, recipients[sharedInbox]) object: event.toNoteAP(settingsController.settings.instance_name,
settingsController.settings.instance_locale,
recipients[sharedInbox])
} }
body['@context'] = [ body['@context'] = [
'https://www.w3.org/ns/activitystreams', 'https://www.w3.org/ns/activitystreams',

View File

@@ -33,7 +33,7 @@ router.get('/m/:event_id', async (req, res) => {
const event = await Event.findByPk(req.params.event_id, { include: [User, Tag, Place] }) const event = await Event.findByPk(req.params.event_id, { include: [User, Tag, Place] })
if (!event) { return res.status(404).send('Not found') } if (!event) { return res.status(404).send('Not found') }
return res.json(event.toNoteAP(event.user.username)) return res.json(event.toNoteAP(event.user.username, req.settings.locale))
}) })
// get any message coming from federation // get any message coming from federation

View File

@@ -115,7 +115,7 @@ module.exports = {
type: 'OrderedCollectionPage', type: 'OrderedCollectionPage',
totalItems: events.length, totalItems: events.length,
partOf: `${config.baseurl}/federation/u/${name}/outbox`, partOf: `${config.baseurl}/federation/u/${name}/outbox`,
orderedItems: events.map(e => ({ ...e.toNoteAP(name), actor: `${config.baseurl}/federation/u/${name}` })) orderedItems: events.map(e => ({ ...e.toNoteAP(name, req.settings.locale), actor: `${config.baseurl}/federation/u/${name}` }))
// user.events.map(e => ({ // user.events.map(e => ({
// id: `${config.baseurl}/federation/m/${e.id}#create`, // id: `${config.baseurl}/federation/m/${e.id}#create`,
// type: 'Create', // type: 'Create',

View File

@@ -60,15 +60,15 @@ module.exports = {
// announcement: 'TODO: HTML First presentation post' // announcement: 'TODO: HTML First presentation post'
// }) // })
// send confirmed event to mastodon // send confirmed events to mastodon
await db.notification.create({ action: 'Create', type: 'ap', filters: { is_visible: true } }) await db.notification.create({ action: 'Create', type: 'ap', filters: { is_visible: true } })
await db.notification.create({ action: 'Update', type: 'ap', filters: { is_visible: true } }) await db.notification.create({ action: 'Update', type: 'ap', filters: { is_visible: true } })
await db.notification.create({ action: 'Delete', type: 'ap', filters: { is_visible: true } }) await db.notification.create({ action: 'Delete', type: 'ap', filters: { is_visible: true } })
// send anon event to administrator // send anon events to admin
await db.notification.create({ action: 'Create', type: 'admin_email', filters: { is_visible: false } }) await db.notification.create({ action: 'Create', type: 'admin_email', filters: { is_visible: false } })
// TODO // TODO email's notifications
// await db.notification.create({ action: 'Create', type: 'email', filters: { is_visible: true } }) // await db.notification.create({ action: 'Create', type: 'email', filters: { is_visible: true } })
// close db connection // close db connection

View File

@@ -9,6 +9,7 @@ const { JSDOM } = require('jsdom')
const { window } = new JSDOM('<!DOCTYPE html>') const { window } = new JSDOM('<!DOCTYPE html>')
const domPurify = DOMPurify(window) const domPurify = DOMPurify(window)
const URL = require('url') const URL = require('url')
const locales = require('../locales')
domPurify.addHook('beforeSanitizeElements', node => { domPurify.addHook('beforeSanitizeElements', node => {
if (node.hasAttribute && node.hasAttribute('href')) { if (node.hasAttribute && node.hasAttribute('href')) {
@@ -52,8 +53,7 @@ module.exports = {
// set locale and user locale // set locale and user locale
const acceptedLanguages = req.headers['accept-language'] const acceptedLanguages = req.headers['accept-language']
const supportedLanguages = ['en', 'it', 'es', 'ca'] acceptLanguage.languages(Object.keys(locales))
acceptLanguage.languages(supportedLanguages)
req.settings.locale = acceptLanguage.get(acceptedLanguages) req.settings.locale = acceptLanguage.get(acceptedLanguages)
req.settings.user_locale = settingsController.user_locale[req.settings.locale] req.settings.user_locale = settingsController.user_locale[req.settings.locale]
moment.locale(req.settings.locale) moment.locale(req.settings.locale)

View File

@@ -15,10 +15,11 @@ const notifier = {
debug('Send %s notification %s', notification.type, notification.action) debug('Send %s notification %s', notification.type, notification.action)
let p let p
switch (notification.type) { switch (notification.type) {
case 'mail': // case 'mail': TODO: locale?
return mail.send(notification.email, 'event', { event, config, notification }) // return mail.send(notification.email, 'event', { event, notification })
case 'admin_email': case 'admin_email':
p = mail.send([config.smtp.auth.user, config.admin_email], 'event', { event, to_confirm: !event.is_visible, config, notification }) p = mail.send(config.admin_email, 'event',
{ event, to_confirm: !event.is_visible, notification })
promises.push(p) promises.push(p)
break break
case 'ap': case 'ap':