2017-01-25 18:51:41 +01:00
|
|
|
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
2016-06-03 11:51:11 +02:00
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2017-05-24 12:24:07 +02:00
|
|
|
use std::collections::BTreeMap;
|
2016-06-03 11:51:11 +02:00
|
|
|
use std::io;
|
|
|
|
use std::io::Read;
|
|
|
|
use std::fs;
|
2017-01-06 16:05:58 +01:00
|
|
|
use std::path::{Path, PathBuf};
|
2017-10-05 12:35:01 +02:00
|
|
|
use futures_cpupool::CpuPool;
|
|
|
|
|
2016-08-18 12:19:09 +02:00
|
|
|
use apps::manifest::{MANIFEST_FILENAME, deserialize_manifest};
|
2017-10-05 12:35:01 +02:00
|
|
|
use endpoint::{Endpoint, EndpointInfo};
|
|
|
|
use page::{local, PageCache};
|
2017-07-04 16:04:09 +02:00
|
|
|
use Embeddable;
|
2016-06-03 11:51:11 +02:00
|
|
|
|
|
|
|
struct LocalDapp {
|
|
|
|
id: String,
|
|
|
|
path: PathBuf,
|
|
|
|
info: EndpointInfo,
|
|
|
|
}
|
|
|
|
|
2017-01-06 16:05:58 +01:00
|
|
|
/// Tries to find and read manifest file in given `path` to extract `EndpointInfo`
|
|
|
|
/// If manifest is not found sensible default `EndpointInfo` is returned based on given `name`.
|
2016-06-03 11:51:11 +02:00
|
|
|
fn read_manifest(name: &str, mut path: PathBuf) -> EndpointInfo {
|
2016-08-18 12:19:09 +02:00
|
|
|
path.push(MANIFEST_FILENAME);
|
2016-06-03 11:51:11 +02:00
|
|
|
|
|
|
|
fs::File::open(path.clone())
|
|
|
|
.map_err(|e| format!("{:?}", e))
|
|
|
|
.and_then(|mut f| {
|
|
|
|
// Reat file
|
|
|
|
let mut s = String::new();
|
2016-12-27 12:53:56 +01:00
|
|
|
f.read_to_string(&mut s).map_err(|e| format!("{:?}", e))?;
|
2016-06-03 11:51:11 +02:00
|
|
|
// Try to deserialize manifest
|
2016-08-18 12:19:09 +02:00
|
|
|
deserialize_manifest(s)
|
2016-06-03 11:51:11 +02:00
|
|
|
})
|
2016-08-18 12:19:09 +02:00
|
|
|
.map(Into::into)
|
2016-06-03 11:51:11 +02:00
|
|
|
.unwrap_or_else(|e| {
|
|
|
|
warn!(target: "dapps", "Cannot read manifest file at: {:?}. Error: {:?}", path, e);
|
|
|
|
|
|
|
|
EndpointInfo {
|
|
|
|
name: name.into(),
|
|
|
|
description: name.into(),
|
|
|
|
version: "0.0.0".into(),
|
|
|
|
author: "?".into(),
|
|
|
|
icon_url: "icon.png".into(),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-01-06 16:05:58 +01:00
|
|
|
/// Returns Dapp Id and Local Dapp Endpoint for given filesystem path.
|
|
|
|
/// Parses the path to extract last component (for name).
|
|
|
|
/// `None` is returned when path is invalid or non-existent.
|
2017-10-05 12:35:01 +02:00
|
|
|
pub fn local_endpoint<P: AsRef<Path>>(path: P, embeddable: Embeddable, pool: CpuPool) -> Option<(String, Box<local::Dapp>)> {
|
2017-01-06 16:05:58 +01:00
|
|
|
let path = path.as_ref().to_owned();
|
|
|
|
path.canonicalize().ok().and_then(|path| {
|
|
|
|
let name = path.file_name().and_then(|name| name.to_str());
|
|
|
|
name.map(|name| {
|
|
|
|
let dapp = local_dapp(name.into(), path.clone());
|
2017-10-05 12:35:01 +02:00
|
|
|
(dapp.id, Box::new(local::Dapp::new(
|
|
|
|
pool.clone(), dapp.path, dapp.info, PageCache::Disabled, embeddable.clone())
|
2017-01-06 16:05:58 +01:00
|
|
|
))
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn local_dapp(name: String, path: PathBuf) -> LocalDapp {
|
|
|
|
// try to get manifest file
|
|
|
|
let info = read_manifest(&name, path.clone());
|
|
|
|
LocalDapp {
|
|
|
|
id: name,
|
|
|
|
path: path,
|
|
|
|
info: info,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns endpoints for Local Dapps found for given filesystem path.
|
2017-10-05 12:35:01 +02:00
|
|
|
/// Scans the directory and collects `local::Dapp`.
|
|
|
|
pub fn local_endpoints<P: AsRef<Path>>(dapps_path: P, embeddable: Embeddable, pool: CpuPool) -> BTreeMap<String, Box<Endpoint>> {
|
2017-05-24 12:24:07 +02:00
|
|
|
let mut pages = BTreeMap::<String, Box<Endpoint>>::new();
|
2017-01-06 16:05:58 +01:00
|
|
|
for dapp in local_dapps(dapps_path.as_ref()) {
|
2016-06-03 11:51:11 +02:00
|
|
|
pages.insert(
|
|
|
|
dapp.id,
|
2017-10-05 12:35:01 +02:00
|
|
|
Box::new(local::Dapp::new(pool.clone(), dapp.path, dapp.info, PageCache::Disabled, embeddable.clone()))
|
2016-06-03 11:51:11 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
pages
|
|
|
|
}
|
2017-01-06 16:05:58 +01:00
|
|
|
|
|
|
|
|
|
|
|
fn local_dapps(dapps_path: &Path) -> Vec<LocalDapp> {
|
|
|
|
let files = fs::read_dir(dapps_path);
|
|
|
|
if let Err(e) = files {
|
|
|
|
warn!(target: "dapps", "Unable to load local dapps from: {}. Reason: {:?}", dapps_path.display(), e);
|
|
|
|
return vec![];
|
|
|
|
}
|
|
|
|
|
|
|
|
let files = files.expect("Check is done earlier");
|
|
|
|
files.map(|dir| {
|
|
|
|
let entry = dir?;
|
|
|
|
let file_type = entry.file_type()?;
|
|
|
|
|
|
|
|
// skip files
|
|
|
|
if file_type.is_file() {
|
|
|
|
return Err(io::Error::new(io::ErrorKind::NotFound, "Not a file"));
|
|
|
|
}
|
|
|
|
|
|
|
|
// take directory name and path
|
|
|
|
entry.file_name().into_string()
|
|
|
|
.map(|name| (name, entry.path()))
|
|
|
|
.map_err(|e| {
|
|
|
|
info!(target: "dapps", "Unable to load dapp: {:?}. Reason: {:?}", entry.path(), e);
|
|
|
|
io::Error::new(io::ErrorKind::NotFound, "Invalid name")
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.filter_map(|m| {
|
|
|
|
if let Err(ref e) = m {
|
|
|
|
debug!(target: "dapps", "Ignoring local dapp: {:?}", e);
|
|
|
|
}
|
|
|
|
m.ok()
|
|
|
|
})
|
|
|
|
.map(|(name, path)| local_dapp(name, path))
|
|
|
|
.collect()
|
|
|
|
}
|