diff --git a/src/app/components/local-plugin.js b/src/app/components/local-plugin.js index 59c1a5bcf5..d760eb0714 100644 --- a/src/app/components/local-plugin.js +++ b/src/app/components/local-plugin.js @@ -35,7 +35,8 @@ module.exports = class LocalPlugin { ...this.profile, icon: '', methods: [], - events: [] + events: [], + hash: `local-${this.profile.name}` } if (!this.profile.name) throw new Error('Plugin should have a name') if (!this.profile.url) throw new Error('Plugin should have an URL') diff --git a/src/app/tabs/compile-tab.js b/src/app/tabs/compile-tab.js index cf55d583e5..f87d43f61a 100644 --- a/src/app/tabs/compile-tab.js +++ b/src/app/tabs/compile-tab.js @@ -65,7 +65,8 @@ class CompileTab extends ApiFactory { events: ['compilationFinished'], icon: '', description: 'compile solidity contracts', - kind: 'compile' + kind: 'compile', + permission: true } } diff --git a/src/persmission-handler.js b/src/persmission-handler.js new file mode 100644 index 0000000000..7a6142df03 --- /dev/null +++ b/src/persmission-handler.js @@ -0,0 +1,126 @@ +/* global localStorage */ +const yo = require('yo-yo') +const csjs = require('csjs-inject') +const modalDialog = require('./app/ui/modaldialog') + +const css = csjs` +.permission p { + text-align: center; +} +.images { + display: flex; + justify-content: center; + align-item: center; +} +.images img { + width: 40px; + height: 40px; +} +` + +export class PermissionHandler { + + constructor () { + const permission = localStorage.getItem('plugin-permissions') + this.permissions = permission ? JSON.parse(permission) : {} + } + + persistPermissions () { + const permissions = JSON.stringify(this.permissions) + localStorage.setItem('plugin-permissions', permissions) + } + + clear () { + localStorage.removeItem('plugin-permissions') + } + + /** + * Show a message to ask the user for a permission + * @param {PluginProfile} from The name and hash of the plugin that make the call + * @param {ModuleProfile} to The name of the plugin that receive the call + * @returns {Promise<{ allow: boolean; remember: boolean }} Answer from the user to the permission + */ + async openPermission (from, to) { + return new Promise((resolve, reject) => { + modalDialog( + `Permission needed for ${to.displayName || to.name}`, + this.form(from, to), + { + label: 'Accept', + fn: () => { + if (this.permissions[to.name][from.name]) { + this.permissions[to.name][from.name].allow = true + this.persistPermissions() + } + resolve(true) + } + }, + { + label: 'Decline', + fn: () => { + if (this.permissions[to.name][from.name]) { + this.permissions[to.name][from.name].allow = false + this.persistPermissions() + } + resolve(false) + } + } + ) + }) + } + + /** + * Check if a plugin has the permission to call another plugin and askPermission if needed + * @param {PluginProfile} from the profile of the plugin that make the call + * @param {ModuleProfile} to The profile of the module that receive the call + * @returns {Promise} + */ + async askPermission (from, to) { + if (!this.permissions[to.name]) this.permissions[to.name] = {} + if (!this.permissions[to.name][from.name]) return this.openPermission(from, to) + + const { allow, hash } = this.permissions[to.name][from.name] + if (!allow) return false + return hash === from.hash + ? true // Allow + : this.openPermission(from, to) // New version of a plugin + } + + /** + * The permission form + * @param {PluginProfile} from The name and hash of the plugin that make the call + * @param {ModuleProfile} to The name of the plugin that receive the call + */ + form (from, to) { + const fromName = from.displayName || from.name + const toName = from.displayName || from.name + const remember = this.permissions[to.name][from.name] + + const switchMode = (e) => { + e.target.checked + ? this.permissions[to.name][from.name] = {} + : delete this.permissions[to.name][from.name] + } + const rememberSwitch = remember + ? yo`` + : yo`` + const message = remember + ? `${fromName} has changed and would like to access the plugin ${toName}.` + : `${fromName} would like to access plugin ${toName}.` + + return yo` +
+
+ + -> + +
+

${message}

+
+ ${rememberSwitch} + +
+
+ ` + } +} diff --git a/src/remixAppManager.js b/src/remixAppManager.js index e2d60ce83f..65e8647384 100644 --- a/src/remixAppManager.js +++ b/src/remixAppManager.js @@ -1,11 +1,13 @@ import { AppManagerApi, Plugin } from 'remix-plugin' import { EventEmitter } from 'events' import PluginManagerProxy from './app/components/plugin-manager-proxy' +import { PermissionHandler } from './persmission-handler' export class RemixAppManager extends AppManagerApi { constructor (store) { super(null) + this.permissionHandler = new PermissionHandler() this.store = store this.hiddenServices = {} this.event = new EventEmitter()