Add permission handler

pull/1/head
Grandschtroumpf 6 years ago committed by François
parent 726689f07f
commit 485785231c
  1. 3
      src/app/components/local-plugin.js
  2. 3
      src/app/tabs/compile-tab.js
  3. 126
      src/persmission-handler.js
  4. 2
      src/remixAppManager.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')

@ -65,7 +65,8 @@ class CompileTab extends ApiFactory {
events: ['compilationFinished'],
icon: '',
description: 'compile solidity contracts',
kind: 'compile'
kind: 'compile',
permission: true
}
}

@ -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<boolean>}
*/
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`<input type="checkbox" onchange="${switchMode}" checkbox class="custom-control-input" id="remember">`
: yo`<input type="checkbox" class="custom-control-input" id="remember">`
const message = remember
? `${fromName} has changed and would like to access the plugin ${toName}.`
: `${fromName} would like to access plugin ${toName}.`
return yo`
<section class="${css.permission}">
<article class="${css.images}">
<img src="${from.icon}" />
<span> -> </span>
<img src="${to.icon}" />
</article>
<p>${message}</p>
<div class="custom-control custom-checkbox">
${rememberSwitch}
<label class="custom-control-label" for="remember">Remember this choice</label>
</div>
</section>
`
}
}

@ -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()

Loading…
Cancel
Save