From 25a79ff125f8211b8a17ef515dc7776c7594e780 Mon Sep 17 00:00:00 2001
From: ElliottKasoar <45317199+ElliottKasoar@users.noreply.github.com>
Date: Thu, 13 Jun 2024 16:07:11 +0200
Subject: [PATCH] Apply black formatting (#109)
---
abcd/__init__.py | 35 ++-
abcd/backends/atoms_http.py | 35 ++-
abcd/backends/atoms_pymongo.py | 362 +++++++++++++----------
abcd/frontends/commandline/commands.py | 221 ++++++++------
abcd/frontends/commandline/config.py | 26 +-
abcd/frontends/commandline/decorators.py | 14 +-
abcd/frontends/commandline/parser.py | 281 ++++++++++++------
abcd/model.py | 149 +++++-----
abcd/parsers/extras.py | 71 ++---
abcd/parsers/queries.py | 56 ++--
abcd/parsers/queries_new.py | 54 ++--
abcd/server/__init__.py | 4 +-
abcd/server/app/__init__.py | 20 +-
abcd/server/app/nav.py | 133 +++++----
abcd/server/app/views/api.py | 13 +-
abcd/server/app/views/database.py | 46 +--
abcd/server/app/views/index.py | 12 +-
tests/test_database.py | 16 +-
tests/test_parsers.py | 14 +-
tutorials/gb_upload.py | 17 +-
tutorials/scripts/Preprocess.py | 51 ++--
tutorials/scripts/Reader.py | 58 ++--
tutorials/scripts/Visualise.py | 74 ++---
tutorials/scripts/Visualise_quip.py | 74 ++---
tutorials/test_db.py | 10 +-
tutorials/test_upload.py | 4 +-
26 files changed, 1045 insertions(+), 805 deletions(-)
diff --git a/abcd/__init__.py b/abcd/__init__.py
index 52279171..b8008379 100644
--- a/abcd/__init__.py
+++ b/abcd/__init__.py
@@ -14,7 +14,7 @@ class ABCD(object):
@classmethod
def from_config(cls, config):
# Factory method
- url = config['url']
+ url = config["url"]
return ABCD.from_url(url)
@classmethod
@@ -23,35 +23,38 @@ def from_url(cls, url, **kwargs):
r = parse.urlparse(url)
logger.info(r)
- if r.scheme == 'mongodb':
+ if r.scheme == "mongodb":
conn_settings = {
- 'host': r.hostname,
- 'port': r.port,
- 'username': r.username,
- 'password': r.password,
- 'authSource': 'admin',
+ "host": r.hostname,
+ "port": r.port,
+ "username": r.username,
+ "password": r.password,
+ "authSource": "admin",
}
- db = r.path.split('/')[1] if r.path else None
- db = db if db else 'abcd'
+ db = r.path.split("/")[1] if r.path else None
+ db = db if db else "abcd"
from abcd.backends.atoms_pymongo import MongoDatabase
+
return MongoDatabase(db_name=db, **conn_settings, **kwargs)
- elif r.scheme == 'http' or r.scheme == 'https':
- raise NotImplementedError('http not yet supported! soon...')
- elif r.scheme == 'ssh':
- raise NotImplementedError('ssh not yet supported! soon...')
+ elif r.scheme == "http" or r.scheme == "https":
+ raise NotImplementedError("http not yet supported! soon...")
+ elif r.scheme == "ssh":
+ raise NotImplementedError("ssh not yet supported! soon...")
else:
- raise NotImplementedError('Unable to recognise the type of connection. (url: {})'.format(url))
+ raise NotImplementedError(
+ "Unable to recognise the type of connection. (url: {})".format(url)
+ )
-if __name__ == '__main__':
+if __name__ == "__main__":
logging.basicConfig(level=logging.INFO)
# url = 'mongodb://mongoadmin:secret@localhost:27017'
- url = 'mongodb://mongoadmin:secret@localhost:27017/abcd_new'
+ url = "mongodb://mongoadmin:secret@localhost:27017/abcd_new"
abcd = ABCD.from_url(url)
abcd.print_info()
diff --git a/abcd/backends/atoms_http.py b/abcd/backends/atoms_http.py
index 1ff6e95f..ee62e61a 100644
--- a/abcd/backends/atoms_http.py
+++ b/abcd/backends/atoms_http.py
@@ -13,16 +13,15 @@
class Atoms(ase.Atoms):
-
@classmethod
def from_dict(cls, data):
- return cls(numbers=data['numbers'], positions=data['positions'])
+ return cls(numbers=data["numbers"], positions=data["positions"])
class HttpDatabase(Database):
"""client/local interface"""
- def __init__(self, url='http://localhost'):
+ def __init__(self, url="http://localhost"):
super().__init__()
self.url = url
@@ -30,7 +29,9 @@ def __init__(self, url='http://localhost'):
def push(self, atoms: ase.Atoms):
# todo: list of Atoms, metadata(user, project, tags)
- message = requests.put(self.url + '/calculation', json=DictEncoder().encode(atoms))
+ message = requests.put(
+ self.url + "/calculation", json=DictEncoder().encode(atoms)
+ )
# message = json.dumps(atoms)
# message_hash = hashlib.md5(message.encode('utf-8')).hexdigest()
logger.info(message)
@@ -44,29 +45,33 @@ def query(self, query_string):
pass
def search(self, query_string: str) -> List[str]:
- results = requests.get(self.url + '/calculation').json()
+ results = requests.get(self.url + "/calculation").json()
return results
def get_atoms(self, id: str) -> Atoms:
- data = requests.get(self.url + '/calculation/{}'.format(id)).json()
+ data = requests.get(self.url + "/calculation/{}".format(id)).json()
atoms = Atoms.from_dict(data)
return atoms
def __repr__(self):
- return 'ABCD(type={}, url={}, ...)'.format(self.__class__.__name__, self.url)
+ return "ABCD(type={}, url={}, ...)".format(self.__class__.__name__, self.url)
def _repr_html_(self):
"""jupyter notebook representation"""
- return 'ABCD database'
+ return "ABCD database"
def print_info(self):
"""shows basic information about the connected database"""
- out = linesep.join([
- '{:=^50}'.format(' ABCD Database '),
- '{:>10}: {}'.format('type', 'remote (http/https)'),
- linesep.join('{:>10}: {}'.format(k, v) for k, v in self.db.info().items())
- ])
+ out = linesep.join(
+ [
+ "{:=^50}".format(" ABCD Database "),
+ "{:>10}: {}".format("type", "remote (http/https)"),
+ linesep.join(
+ "{:>10}: {}".format(k, v) for k, v in self.db.info().items()
+ ),
+ ]
+ )
print(out)
@@ -80,6 +85,6 @@ def __exit__(self, exc_type, exc_val, exc_tb):
pass
-if __name__ == '__main__':
- abcd = HttpDatabase(url='http://localhost:8080/api')
+if __name__ == "__main__":
+ abcd = HttpDatabase(url="http://localhost:8080/api")
abcd.print_info()
diff --git a/abcd/backends/atoms_pymongo.py b/abcd/backends/atoms_pymongo.py
index 217e1c52..993c6eb0 100644
--- a/abcd/backends/atoms_pymongo.py
+++ b/abcd/backends/atoms_pymongo.py
@@ -26,12 +26,12 @@
logger = logging.getLogger(__name__)
map_types = {
- bool: 'bool',
- float: 'float',
- int: 'int',
- str: 'str',
- datetime: 'date',
- dict: 'dict'
+ bool: "bool",
+ float: "float",
+ int: "int",
+ str: "str",
+ datetime: "date",
+ dict: "dict",
}
@@ -49,15 +49,14 @@ def from_atoms(cls, collection, atoms: Atoms, extra_info=None, store_calc=True):
@property
def _id(self):
- return self.get('_id', None)
+ return self.get("_id", None)
def save(self):
if not self._id:
self._collection.insert_one(self)
else:
- new_values = { "$set": self }
- self._collection.update_one(
- {"_id": ObjectId(self._id)}, new_values)
+ new_values = {"$set": self}
+ self._collection.update_one({"_id": ObjectId(self._id)}, new_values)
def remove(self):
if self._id:
@@ -66,28 +65,27 @@ def remove(self):
class MongoQuery(AbstractQuerySet):
-
def __init__(self):
pass
def visit(self, syntax_tree):
op, *args = syntax_tree
try:
- fun = self.__getattribute__('visit_' + op.lower())
+ fun = self.__getattribute__("visit_" + op.lower())
return fun(*args)
except KeyError:
pass
def visit_name(self, field):
- return {field: {'$exists': True}}
+ return {field: {"$exists": True}}
def visit_not(self, value):
_, field = value
- return {field: {'$exists': False}}
+ return {field: {"$exists": False}}
def visit_and(self, *args):
print(args)
- return {'$and': [self.visit(arg) for arg in args]}
+ return {"$and": [self.visit(arg) for arg in args]}
# TODO recursively combining all the and statements
# out = {}
# for arg in args:
@@ -97,28 +95,28 @@ def visit_and(self, *args):
# return out
def visit_or(self, *args):
- return {'$or': [self.visit(arg) for arg in args]}
+ return {"$or": [self.visit(arg) for arg in args]}
def visit_eq(self, field, value):
return {field[1]: value[1]}
def visit_re(self, field, value):
- return {field[1]: {'$regex': value[1]}}
+ return {field[1]: {"$regex": value[1]}}
def visit_gt(self, field, value):
- return {field[1]: {'$gt': value[1]}}
+ return {field[1]: {"$gt": value[1]}}
def visit_gte(self, field, value):
- return {field[1]: {'$gte': value[1]}}
+ return {field[1]: {"$gte": value[1]}}
def visit_lt(self, field, value):
- return {field[1]: {'$lt': value[1]}}
+ return {field[1]: {"$lt": value[1]}}
def visit_lte(self, field, value):
- return {field[1]: {'$lte': value[1]}}
+ return {field[1]: {"$lte": value[1]}}
def visit_in(self, field, *values):
- return {field[1]: {'$in': [value[1] for value in values]}}
+ return {field[1]: {"$in": [value[1] for value in values]}}
def __enter__(self):
return self
@@ -127,12 +125,13 @@ def __exit__(self, exc_type, exc_val, exc_tb):
pass
def __call__(self, ast):
- logger.info('parsed ast: {}'.format(ast))
+ logger.info("parsed ast: {}".format(ast))
if isinstance(ast, dict):
return ast
elif isinstance(ast, str):
from abcd.parsers.queries import parser
+
p = parser(ast)
return self.visit(p)
@@ -156,20 +155,43 @@ def wrapper(*args, query=None, **kwargs):
class MongoDatabase(AbstractABCD):
"""Wrapper to make database operations easy"""
- def __init__(self, host='localhost', port=27017,
- db_name='abcd', collection_name='atoms',
- username=None, password=None, authSource='admin', **kwargs):
+ def __init__(
+ self,
+ host="localhost",
+ port=27017,
+ db_name="abcd",
+ collection_name="atoms",
+ username=None,
+ password=None,
+ authSource="admin",
+ **kwargs
+ ):
super().__init__()
- logger.info((host, port, db_name, collection_name, username, password, authSource, kwargs))
+ logger.info(
+ (
+ host,
+ port,
+ db_name,
+ collection_name,
+ username,
+ password,
+ authSource,
+ kwargs,
+ )
+ )
self.client = MongoClient(
- host=host, port=port, username=username, password=password,
- authSource=authSource)
+ host=host,
+ port=port,
+ username=username,
+ password=password,
+ authSource=authSource,
+ )
try:
info = self.client.server_info() # Forces a call.
- logger.info('DB info: {}'.format(info))
+ logger.info("DB info: {}".format(info))
except pymongo.errors.OperationFailure:
raise abcd.errors.AuthenticationError()
@@ -184,12 +206,12 @@ def info(self):
host, port = self.client.address
return {
- 'host': host,
- 'port': port,
- 'db': self.db.name,
- 'collection': self.collection.name,
- 'number of confs': self.collection.count_documents({}),
- 'type': 'mongodb'
+ "host": host,
+ "port": port,
+ "db": self.db.name,
+ "collection": self.collection.name,
+ "number of confs": self.collection.count_documents({}),
+ "type": "mongodb",
}
def delete(self, query=None):
@@ -205,14 +227,18 @@ def push(self, atoms: Union[Atoms, Iterable], extra_info=None, store_calc=True):
extra_info = extras.parser.parse(extra_info)
if isinstance(atoms, Atoms):
- data = AtomsModel.from_atoms(self.collection, atoms, extra_info=extra_info, store_calc=store_calc)
+ data = AtomsModel.from_atoms(
+ self.collection, atoms, extra_info=extra_info, store_calc=store_calc
+ )
data.save()
# self.collection.insert_one(data)
elif isinstance(atoms, types.GeneratorType) or isinstance(atoms, list):
for item in atoms:
- data = AtomsModel.from_atoms(self.collection, item, extra_info=extra_info, store_calc=store_calc)
+ data = AtomsModel.from_atoms(
+ self.collection, item, extra_info=extra_info, store_calc=store_calc
+ )
data.save()
def upload(self, file: Path, extra_infos=None, store_calc=True):
@@ -225,7 +251,7 @@ def upload(self, file: Path, extra_infos=None, store_calc=True):
for info in extra_infos:
extra_info.update(extras.parser.parse(info))
- extra_info['filename'] = str(file)
+ extra_info["filename"] = str(file)
data = iread(str(file))
self.push(data, extra_info, store_calc=store_calc)
@@ -243,7 +269,7 @@ def get_atoms(self, query=None):
def count(self, query=None):
query = parser(query)
- logger.info('query; {}'.format(query))
+ logger.info("query; {}".format(query))
if not query:
query = {}
@@ -254,61 +280,69 @@ def property(self, name, query=None):
query = parser(query)
pipeline = [
- {'$match': query},
- {'$match': {'{}'.format(name): {"$exists": True}}},
- {'$project': {'_id': False, 'data': '${}'.format(name)}}
+ {"$match": query},
+ {"$match": {"{}".format(name): {"$exists": True}}},
+ {"$project": {"_id": False, "data": "${}".format(name)}},
]
- return [val['data'] for val in self.db.atoms.aggregate(pipeline)]
+ return [val["data"] for val in self.db.atoms.aggregate(pipeline)]
def properties(self, query=None):
query = parser(query)
properties = {}
pipeline = [
- {'$match': query},
- {'$unwind': '$derived.info_keys'},
- {'$group': {'_id': '$derived.info_keys'}}
+ {"$match": query},
+ {"$unwind": "$derived.info_keys"},
+ {"$group": {"_id": "$derived.info_keys"}},
+ ]
+ properties["info"] = [
+ value["_id"] for value in self.db.atoms.aggregate(pipeline)
]
- properties['info'] = [value['_id'] for value in self.db.atoms.aggregate(pipeline)]
pipeline = [
- {'$match': query},
- {'$unwind': '$derived.arrays_keys'},
- {'$group': {'_id': '$derived.arrays_keys'}}
+ {"$match": query},
+ {"$unwind": "$derived.arrays_keys"},
+ {"$group": {"_id": "$derived.arrays_keys"}},
+ ]
+ properties["arrays"] = [
+ value["_id"] for value in self.db.atoms.aggregate(pipeline)
]
- properties['arrays'] = [value['_id'] for value in self.db.atoms.aggregate(pipeline)]
pipeline = [
- {'$match': query},
- {'$unwind': '$derived.derived_keys'},
- {'$group': {'_id': '$derived.derived_keys'}}
+ {"$match": query},
+ {"$unwind": "$derived.derived_keys"},
+ {"$group": {"_id": "$derived.derived_keys"}},
+ ]
+ properties["derived"] = [
+ value["_id"] for value in self.db.atoms.aggregate(pipeline)
]
- properties['derived'] = [value['_id'] for value in self.db.atoms.aggregate(pipeline)]
return properties
def get_type_of_property(self, prop, category):
# TODO: Probably it would be nicer to store the type info in the database from the beginning.
- atoms = self.db.atoms.find_one({prop: {'$exists': True}})
+ atoms = self.db.atoms.find_one({prop: {"$exists": True}})
data = atoms[prop]
- if category == 'arrays':
+ if category == "arrays":
if type(data[0]) == list:
- return 'array({}, N x {})'.format(map_types[type(data[0][0])], len(data[0]))
+ return "array({}, N x {})".format(
+ map_types[type(data[0][0])], len(data[0])
+ )
else:
- return 'vector({}, N)'.format(map_types[type(data[0])])
+ return "vector({}, N)".format(map_types[type(data[0])])
if type(data) == list:
if type(data[0]) == list:
if type(data[0][0]) == list:
- return 'list(list(...)'
+ return "list(list(...)"
else:
- return 'array({})'.format(map_types[type(data[0][0])])
+ return "array({})".format(map_types[type(data[0][0])])
else:
- return 'vector({})'.format(map_types[type(data[0])])
+ return "vector({})".format(map_types[type(data[0])])
else:
- return 'scalar({})'.format(map_types[type(data)])
+ return "scalar({})".format(map_types[type(data)])
def count_properties(self, query=None):
query = parser(query)
@@ -316,67 +350,69 @@ def count_properties(self, query=None):
properties = {}
pipeline = [
- {'$match': query},
- {'$unwind': '$derived.info_keys'},
- {'$group': {'_id': '$derived.info_keys', 'count': {'$sum': 1}}}
+ {"$match": query},
+ {"$unwind": "$derived.info_keys"},
+ {"$group": {"_id": "$derived.info_keys", "count": {"$sum": 1}}},
]
info_keys = self.db.atoms.aggregate(pipeline)
for val in info_keys:
- properties[val['_id']] = {
- 'count': val['count'],
- 'category': 'info',
- 'dtype': self.get_type_of_property(val['_id'], 'info')
+ properties[val["_id"]] = {
+ "count": val["count"],
+ "category": "info",
+ "dtype": self.get_type_of_property(val["_id"], "info"),
}
pipeline = [
- {'$match': query},
- {'$unwind': '$derived.arrays_keys'},
- {'$group': {'_id': '$derived.arrays_keys', 'count': {'$sum': 1}}}
+ {"$match": query},
+ {"$unwind": "$derived.arrays_keys"},
+ {"$group": {"_id": "$derived.arrays_keys", "count": {"$sum": 1}}},
]
arrays_keys = list(self.db.atoms.aggregate(pipeline))
for val in arrays_keys:
- properties[val['_id']] = {
- 'count': val['count'],
- 'category': 'arrays',
- 'dtype': self.get_type_of_property(val['_id'], 'arrays')
+ properties[val["_id"]] = {
+ "count": val["count"],
+ "category": "arrays",
+ "dtype": self.get_type_of_property(val["_id"], "arrays"),
}
pipeline = [
- {'$match': query},
- {'$unwind': '$derived.derived_keys'},
- {'$group': {'_id': '$derived.derived_keys', 'count': {'$sum': 1}}}
+ {"$match": query},
+ {"$unwind": "$derived.derived_keys"},
+ {"$group": {"_id": "$derived.derived_keys", "count": {"$sum": 1}}},
]
arrays_keys = list(self.db.atoms.aggregate(pipeline))
for val in arrays_keys:
- properties[val['_id']] = {
- 'count': val['count'],
- 'category': 'derived',
- 'dtype': self.get_type_of_property(val['_id'], 'derived')
+ properties[val["_id"]] = {
+ "count": val["count"],
+ "category": "derived",
+ "dtype": self.get_type_of_property(val["_id"], "derived"),
}
return properties
def add_property(self, data, query=None):
- logger.info('add: data={}, query={}'.format(data, query))
+ logger.info("add: data={}, query={}".format(data, query))
self.collection.update_many(
parser(query),
- {'$push': {'derived.info_keys': {'$each': list(data.keys())}},
- '$set': data})
+ {
+ "$push": {"derived.info_keys": {"$each": list(data.keys())}},
+ "$set": data,
+ },
+ )
def rename_property(self, name, new_name, query=None):
- logger.info('rename: query={}, old={}, new={}'.format(query, name, new_name))
+ logger.info("rename: query={}, old={}, new={}".format(query, name, new_name))
# TODO name in derived.info_keys OR name in derived.arrays_keys OR name in derived.derived_keys
self.collection.update_many(
- parser(query),
- {'$push': {'derived.info_keys': new_name}})
+ parser(query), {"$push": {"derived.info_keys": new_name}}
+ )
self.collection.update_many(
parser(query),
- {
- '$pull': {'derived.info_keys': name},
- '$rename': {name: new_name}})
+ {"$pull": {"derived.info_keys": name}, "$rename": {name: new_name}},
+ )
# self.collection.update_many(
# parser(query + ['arrays.{}'.format(name)]),
@@ -388,13 +424,15 @@ def rename_property(self, name, new_name, query=None):
# '$rename': {'arrays.{}'.format(name): 'arrays.{}'.format(new_name)}})
def delete_property(self, name, query=None):
- logger.info('delete: query={}, porperty={}'.format(name, query))
+ logger.info("delete: query={}, porperty={}".format(name, query))
self.collection.update_many(
parser(query),
- {'$pull': {'derived.info_keys': name,
- 'derived.arrays_keys': name},
- '$unset': {name: ''}})
+ {
+ "$pull": {"derived.info_keys": name, "derived.arrays_keys": name},
+ "$unset": {name: ""},
+ },
+ )
def hist(self, name, query=None, **kwargs):
@@ -411,21 +449,27 @@ def exec(self, code, query=None):
def __repr__(self):
host, port = self.client.address
- return '{}('.format(self.__class__.__name__) + \
- 'url={}:{}, '.format(host, port) + \
- 'db={}, '.format(self.db.name) + \
- 'collection={})'.format(self.collection.name)
+ return (
+ "{}(".format(self.__class__.__name__)
+ + "url={}:{}, ".format(host, port)
+ + "db={}, ".format(self.db.name)
+ + "collection={})".format(self.collection.name)
+ )
def _repr_html_(self):
"""Jupyter notebook representation"""
- return 'ABCD MongoDB database'
+ return "ABCD MongoDB database"
def print_info(self):
"""shows basic information about the connected database"""
- out = linesep.join(['{:=^50}'.format(' ABCD MongoDB '),
- '{:>10}: {}'.format('type', 'mongodb'),
- linesep.join('{:>10}: {}'.format(k, v) for k, v in self.info().items())])
+ out = linesep.join(
+ [
+ "{:=^50}".format(" ABCD MongoDB "),
+ "{:>10}: {}".format("type", "mongodb"),
+ linesep.join("{:>10}: {}".format(k, v) for k, v in self.info().items()),
+ ]
+ )
print(out)
@@ -449,26 +493,36 @@ def histogram(name, data, **kwargs):
return None
if ptype == float:
- bins = kwargs.get('bins', 10)
+ bins = kwargs.get("bins", 10)
return _hist_float(name, data, bins)
elif ptype == int:
- bins = kwargs.get('bins', 10)
+ bins = kwargs.get("bins", 10)
return _hist_int(name, data, bins)
elif ptype == str:
return _hist_str(name, data, **kwargs)
elif ptype == datetime:
- bins = kwargs.get('bins', 10)
+ bins = kwargs.get("bins", 10)
return _hist_date(name, data, bins)
else:
- print('{}: Histogram for list of {} types are not supported!'.format(name, type(data[0])))
- logger.info('{}: Histogram for list of {} types are not supported!'.format(name, type(data[0])))
+ print(
+ "{}: Histogram for list of {} types are not supported!".format(
+ name, type(data[0])
+ )
+ )
+ logger.info(
+ "{}: Histogram for list of {} types are not supported!".format(
+ name, type(data[0])
+ )
+ )
else:
- logger.info('{}: Histogram for {} types are not supported!'.format(name, type(data)))
+ logger.info(
+ "{}: Histogram for {} types are not supported!".format(name, type(data))
+ )
return None
@@ -477,16 +531,16 @@ def _hist_float(name, data, bins=10):
hist, bin_edges = np.histogram(data, bins=bins)
return {
- 'type': 'hist_float',
- 'name': name,
- 'bins': bins,
- 'edges': bin_edges,
- 'counts': hist,
- 'min': data.min(),
- 'max': data.max(),
- 'median': data.mean(),
- 'std': data.std(),
- 'var': data.var()
+ "type": "hist_float",
+ "name": name,
+ "bins": bins,
+ "edges": bin_edges,
+ "counts": hist,
+ "min": data.min(),
+ "max": data.max(),
+ "median": data.mean(),
+ "std": data.std(),
+ "var": data.var(),
}
@@ -497,16 +551,16 @@ def _hist_date(name, data, bins=10):
fromtimestamp = datetime.fromtimestamp
return {
- 'type': 'hist_date',
- 'name': name,
- 'bins': bins,
- 'edges': [fromtimestamp(d) for d in bin_edges],
- 'counts': hist,
- 'min': fromtimestamp(hist_data.min()),
- 'max': fromtimestamp(hist_data.max()),
- 'median': fromtimestamp(hist_data.mean()),
- 'std': fromtimestamp(hist_data.std()),
- 'var': fromtimestamp(hist_data.var())
+ "type": "hist_date",
+ "name": name,
+ "bins": bins,
+ "edges": [fromtimestamp(d) for d in bin_edges],
+ "counts": hist,
+ "min": fromtimestamp(hist_data.min()),
+ "max": fromtimestamp(hist_data.max()),
+ "median": fromtimestamp(hist_data.mean()),
+ "std": fromtimestamp(hist_data.std()),
+ "var": fromtimestamp(hist_data.var()),
}
@@ -520,16 +574,16 @@ def _hist_int(name, data, bins=10):
hist, bin_edges = np.histogram(data, bins=bins)
return {
- 'type': 'hist_int',
- 'name': name,
- 'bins': bins,
- 'edges': bin_edges,
- 'counts': hist,
- 'min': data.min(),
- 'max': data.max(),
- 'median': data.mean(),
- 'std': data.std(),
- 'var': data.var()
+ "type": "hist_int",
+ "name": name,
+ "bins": bins,
+ "edges": bin_edges,
+ "counts": hist,
+ "min": data.min(),
+ "max": data.max(),
+ "median": data.mean(),
+ "std": data.std(),
+ "var": data.var(),
}
@@ -538,7 +592,9 @@ def _hist_str(name, data, bins=10, truncate=20):
if truncate:
# data = (item[:truncate] for item in data)
- data = (item[:truncate] + '...' if len(item) > truncate else item for item in data)
+ data = (
+ item[:truncate] + "..." if len(item) > truncate else item for item in data
+ )
data = Counter(data)
@@ -548,27 +604,27 @@ def _hist_str(name, data, bins=10, truncate=20):
labels, counts = zip(*data.items())
return {
- 'type': 'hist_str',
- 'name': name,
- 'total': sum(data.values()),
- 'unique': n_unique,
- 'labels': labels[:bins],
- 'counts': counts[:bins]
+ "type": "hist_str",
+ "name": name,
+ "total": sum(data.values()),
+ "unique": n_unique,
+ "labels": labels[:bins],
+ "counts": counts[:bins],
}
-if __name__ == '__main__':
+if __name__ == "__main__":
# import json
# from ase.io import iread
# from pprint import pprint
# from server.styles.myjson import JSONEncoderOld, JSONDecoderOld, JSONEncoder
- print('hello')
- db = MongoDatabase(username='mongoadmin', password='secret')
+ print("hello")
+ db = MongoDatabase(username="mongoadmin", password="secret")
print(db.info())
print(db.count())
- print(db.hist('uploaded'))
+ print(db.hist("uploaded"))
# for atoms in iread('../../tutorials/data/bcc_bulk_54_expanded_2_high.xyz', index=slice(None)):
# # print(at)
diff --git a/abcd/frontends/commandline/commands.py b/abcd/frontends/commandline/commands.py
index ebe6a213..de158a5c 100644
--- a/abcd/frontends/commandline/commands.py
+++ b/abcd/frontends/commandline/commands.py
@@ -11,32 +11,37 @@
@init_config
def login(*, config, name, url, **kwargs):
logger.info(
- 'login args: \nconfig:{}, name:{}, url:{}, kwargs:{}'.format(config, name, url, kwargs))
+ "login args: \nconfig:{}, name:{}, url:{}, kwargs:{}".format(
+ config, name, url, kwargs
+ )
+ )
from abcd import ABCD
db = ABCD.from_url(url=url)
info = db.info()
- config['url'] = url
+ config["url"] = url
config.save()
- print('Successfully connected to the database!')
- print(" type: {type}\n"
- " hostname: {host}\n"
- " port: {port}\n"
- " database: {db}\n"
- " # of confs: {number of confs}".format(**info))
+ print("Successfully connected to the database!")
+ print(
+ " type: {type}\n"
+ " hostname: {host}\n"
+ " port: {port}\n"
+ " database: {db}\n"
+ " # of confs: {number of confs}".format(**info)
+ )
@init_config
@init_db
def download(*, db, query, fileformat, filename, **kwargs):
- logger.info('download\n kwargs: {}'.format(kwargs))
+ logger.info("download\n kwargs: {}".format(kwargs))
from ase.io import write
- if kwargs.pop('remote'):
- write('-', list(db.get_atoms(query=query)), format=fileformat)
+ if kwargs.pop("remote"):
+ write("-", list(db.get_atoms(query=query)), format=fileformat)
return
write(filename, list(db.get_atoms(query=query)), format=fileformat)
@@ -46,14 +51,18 @@ def download(*, db, query, fileformat, filename, **kwargs):
@init_db
@check_remote
def delete(*, db, query, yes, **kwargs):
- logger.info('delete\n kwargs: {}'.format(kwargs))
+ logger.info("delete\n kwargs: {}".format(kwargs))
if not yes:
- print('Please use --yes for deleting {} configurations'.format(db.count(query=query)))
+ print(
+ "Please use --yes for deleting {} configurations".format(
+ db.count(query=query)
+ )
+ )
exit(1)
count = db.delete(query=query)
- print('{} configuration has been deleted'.format(count))
+ print("{} configuration has been deleted".format(count))
@init_config
@@ -69,11 +78,11 @@ def upload(*, db, path, extra_infos, ignore_calc_results, **kwargs):
db.upload(path, extra_infos, store_calc=calculator)
elif path.is_dir():
- for file in path.glob('.xyz'):
- logger.info('Uploaded file: {}'.format(file))
+ for file in path.glob(".xyz"):
+ logger.info("Uploaded file: {}".format(file))
db.upload(file, extra_infos, store_calc=calculator)
else:
- logger.info('No file found: {}'.format(path))
+ logger.info("No file found: {}".format(path))
raise FileNotFoundError()
else:
@@ -83,8 +92,8 @@ def upload(*, db, path, extra_infos, ignore_calc_results, **kwargs):
@init_config
@init_db
def summary(*, db, query, print_all, bins, truncate, props, **kwargs):
- logger.info('summary\n kwargs: {}'.format(kwargs))
- logger.info('query: {}'.format(query))
+ logger.info("summary\n kwargs: {}".format(kwargs))
+ logger.info("query: {}".format(query))
if print_all:
truncate = None
@@ -97,15 +106,15 @@ def summary(*, db, query, print_all, bins, truncate, props, **kwargs):
props_list = []
for prop in props:
# TODO: Check that is this the right place?
- props_list.extend(re.split(r';\s*|,\s*|\s+', prop))
+ props_list.extend(re.split(r";\s*|,\s*|\s+", prop))
- if '*' in props_list:
- props_list = '*'
+ if "*" in props_list:
+ props_list = "*"
- logging.info('property list: {}'.format(props_list))
+ logging.info("property list: {}".format(props_list))
total = db.count(query)
- print('Total number of configurations: {}'.format(total))
+ print("Total number of configurations: {}".format(total))
if total == 0:
return
@@ -118,13 +127,13 @@ def summary(*, db, query, print_all, bins, truncate, props, **kwargs):
labels, categories, dtypes, counts = [], [], [], []
for k in sorted(props, key=str.lower):
labels.append(k)
- counts.append(props[k]['count'])
- categories.append(props[k]['category'])
- dtypes.append(props[k]['dtype'])
+ counts.append(props[k]["count"])
+ categories.append(props[k]["category"])
+ dtypes.append(props[k]["dtype"])
f.hist_labels(counts, categories, dtypes, labels)
- elif props_list == '*':
+ elif props_list == "*":
props = db.properties(query=query)
for ptype in props:
@@ -149,8 +158,8 @@ def summary(*, db, query, print_all, bins, truncate, props, **kwargs):
@init_config
@init_db
def show(*, db, query, print_all, props, **kwargs):
- logger.info('show\n kwargs: {}'.format(kwargs))
- logger.info('query: {}'.format(query))
+ logger.info("show\n kwargs: {}".format(kwargs))
+ logger.info("query: {}".format(query))
if not props:
print("Please define at least on property by using the -p option!")
@@ -162,7 +171,7 @@ def show(*, db, query, print_all, props, **kwargs):
for dct in islice(db.get_items(query), 0, limit):
print(" | ".join(str(dct.get(prop, None)) for prop in props))
- logging.info('property list: {}'.format(props))
+ logging.info("property list: {}".format(props))
@check_remote
@@ -171,17 +180,19 @@ def show(*, db, query, print_all, props, **kwargs):
def key_add(*, db, query, keys, **kwargs):
from abcd.parsers.extras import parser
- keys = ' '.join(keys)
+ keys = " ".join(keys)
data = parser.parse(keys)
if query:
- test = ('AND', query, ("OR", *(('NAME', key) for key in data.keys())))
+ test = ("AND", query, ("OR", *(("NAME", key) for key in data.keys())))
else:
- test = ("OR", *(('NAME', key) for key in data.keys()))
+ test = ("OR", *(("NAME", key) for key in data.keys()))
if db.count(query=test):
- print('The new key already exist for the given query! '
- 'Please make sure that the target key name don\'t exist')
+ print(
+ "The new key already exist for the given query! "
+ "Please make sure that the target key name don't exist"
+ )
exit(1)
db.add_property(data, query=query)
@@ -192,8 +203,10 @@ def key_add(*, db, query, keys, **kwargs):
@init_db
def key_rename(*, db, query, old_keys, new_keys, **kwargs):
if db.count(query=query + [old_keys, new_keys]):
- print('The new key already exist for the given query! '
- 'Please make sure that the target key name don\'t exist')
+ print(
+ "The new key already exist for the given query! "
+ "Please make sure that the target key name don't exist"
+ )
exit(1)
db.rename_property(old_keys, new_keys, query=query)
@@ -205,14 +218,17 @@ def key_rename(*, db, query, old_keys, new_keys, **kwargs):
def key_delete(*, db, query, yes, keys, **kwargs):
from abcd.parsers.extras import parser
- keys = ' '.join(keys)
+ keys = " ".join(keys)
data = parser.parse(keys)
- query = ('AND', query, ('OR', *(('NAME', key) for key in data.keys())))
+ query = ("AND", query, ("OR", *(("NAME", key) for key in data.keys())))
if not yes:
- print('Please use --yes for deleting keys from {} configurations'.format(
- db.count(query=query)))
+ print(
+ "Please use --yes for deleting keys from {} configurations".format(
+ db.count(query=query)
+ )
+ )
exit(1)
for k in keys:
@@ -224,8 +240,11 @@ def key_delete(*, db, query, yes, keys, **kwargs):
@init_db
def execute(*, db, query, yes, python_code, **kwargs):
if not yes:
- print('Please use --yes for executing code on {} configurations'.format(
- db.count(query=query)))
+ print(
+ "Please use --yes for executing code on {} configurations".format(
+ db.count(query=query)
+ )
+ )
exit(1)
db.exec(python_code, query)
@@ -236,7 +255,9 @@ def server(*, abcd_url, url, api_only, **kwargs):
from urllib.parse import urlparse
from abcd.server.app import create_app
- logger.info("SERVER - abcd: {}, url: {}, api_only:{}".format(abcd_url, url, api_only))
+ logger.info(
+ "SERVER - abcd: {}, url: {}, api_only:{}".format(abcd_url, url, api_only)
+ )
if api_only:
print("Not implemented yet!")
@@ -252,26 +273,37 @@ class Formater(object):
partialBlocks = ["▏", "▎", "▍", "▌", "▋", "▊", "▉", "█"] # char=pb
def title(self, title):
- print('', title, '=' * len(title), sep=os.linesep)
+ print("", title, "=" * len(title), sep=os.linesep)
def describe(self, data):
- if data['type'] == 'hist_float':
+ if data["type"] == "hist_float":
print(
- '{} count: {} min: {:11.4e} med: {:11.4e} max: {:11.4e} std: {:11.4e} var:{'
- ':11.4e}'.format(
- data["name"], sum(data["counts"]),
- data["min"], data["median"], data["max"],
- data["std"], data["var"])
+ "{} count: {} min: {:11.4e} med: {:11.4e} max: {:11.4e} std: {:11.4e} var:{"
+ ":11.4e}".format(
+ data["name"],
+ sum(data["counts"]),
+ data["min"],
+ data["median"],
+ data["max"],
+ data["std"],
+ data["var"],
+ )
)
- elif data['type'] == 'hist_int':
- print('{} count: {} '.format(data["name"], sum(data["counts"])),
- 'min: {:d} med: {:d} max: {:d} '.format(int(data["min"]), int(data["median"]),
- int(data["max"]))
- )
+ elif data["type"] == "hist_int":
+ print(
+ "{} count: {} ".format(data["name"], sum(data["counts"])),
+ "min: {:d} med: {:d} max: {:d} ".format(
+ int(data["min"]), int(data["median"]), int(data["max"])
+ ),
+ )
- elif data['type'] == 'hist_str':
- print('{} count: {} unique: {}'.format(data["name"], data["total"], data["unique"]))
+ elif data["type"] == "hist_str":
+ print(
+ "{} count: {} unique: {}".format(
+ data["name"], data["total"], data["unique"]
+ )
+ )
else:
pass
@@ -282,10 +314,11 @@ def hist_float(self, bin_edges, counts, width_hist=40):
for count, lower, upper in zip(counts, bin_edges[:-1], bin_edges[1:]):
scale = int(ratio * count)
- self.print('{:<{}} {:>{}d} [{: >11.4e}, {: >11.4f})'.format(
- "▉" * scale, width_hist,
- count, width_count,
- lower, upper))
+ self.print(
+ "{:<{}} {:>{}d} [{: >11.4e}, {: >11.4f})".format(
+ "▉" * scale, width_hist, count, width_count, lower, upper
+ )
+ )
def hist_int(self, bin_edges, counts, width_hist=40):
@@ -294,35 +327,50 @@ def hist_int(self, bin_edges, counts, width_hist=40):
for count, lower, upper in zip(counts, bin_edges[:-1], bin_edges[1:]):
scale = int(ratio * count)
- self.print('{:<{}} {:>{}d} [{:d}, {:d})'.format(
- "▉" * scale, width_hist,
- count, width_count,
- np.ceil(lower).astype(int), np.floor(upper).astype(int)))
+ self.print(
+ "{:<{}} {:>{}d} [{:d}, {:d})".format(
+ "▉" * scale,
+ width_hist,
+ count,
+ width_count,
+ np.ceil(lower).astype(int),
+ np.floor(upper).astype(int),
+ )
+ )
def hist_date(self, bin_edges, counts, width_hist=40):
- dateformat = '%y-%m-%d %H:%M'
+ dateformat = "%y-%m-%d %H:%M"
ratio = width_hist / max(counts)
width_count = len(str(max(counts)))
for count, lower, upper in zip(counts, bin_edges[:-1], bin_edges[1:]):
scale = int(ratio * count)
- self.print('{:<{}} {:>{}d} [{}, {})'.format(
- "▉" * scale, width_hist,
- count, width_count,
- lower.strftime(dateformat), upper.strftime(dateformat)))
+ self.print(
+ "{:<{}} {:>{}d} [{}, {})".format(
+ "▉" * scale,
+ width_hist,
+ count,
+ width_count,
+ lower.strftime(dateformat),
+ upper.strftime(dateformat),
+ )
+ )
def hist_str(self, total, counts, labels, width_hist=40):
remain = total - sum(counts)
if remain > 0:
counts = (*counts, remain)
- labels = (*labels, '...')
+ labels = (*labels, "...")
width_count = len(str(max(counts)))
ratio = width_hist / max(counts)
for label, count in zip(labels, counts):
scale = int(ratio * count)
self.print(
- '{:<{}} {:>{}d} {}'.format("▉" * scale, width_hist, count, width_count, label))
+ "{:<{}} {:>{}d} {}".format(
+ "▉" * scale, width_hist, count, width_count, label
+ )
+ )
def hist_labels(self, counts, categories, dtypes, labels, width_hist=40):
@@ -330,18 +378,21 @@ def hist_labels(self, counts, categories, dtypes, labels, width_hist=40):
ratio = width_hist / max(counts)
for label, count, dtype in zip(labels, counts, dtypes):
scale = int(ratio * count)
- self.print('{:<{}} {:<21} {:>{}d} {}'.format(
- "▉" * scale, width_hist, dtype, count, width_count, label))
+ self.print(
+ "{:<{}} {:<21} {:>{}d} {}".format(
+ "▉" * scale, width_hist, dtype, count, width_count, label
+ )
+ )
def hist(self, data: dict, width_hist=40):
- if data['type'] == 'hist_float':
- self.hist_float(data['edges'], data['counts'])
- elif data['type'] == 'hist_int':
- self.hist_int(data['edges'], data['counts'])
- elif data['type'] == 'hist_date':
- self.hist_date(data['edges'], data['counts'])
- elif data['type'] == 'hist_str':
- self.hist_str(data['total'], data['counts'], data['labels'])
+ if data["type"] == "hist_float":
+ self.hist_float(data["edges"], data["counts"])
+ elif data["type"] == "hist_int":
+ self.hist_int(data["edges"], data["counts"])
+ elif data["type"] == "hist_date":
+ self.hist_date(data["edges"], data["counts"])
+ elif data["type"] == "hist_str":
+ self.hist_str(data["total"], data["counts"], data["labels"])
else:
pass
@@ -350,4 +401,4 @@ def print(*args, **kwargs):
print(*args, **kwargs)
def _trunc(self, text, width=80):
- return text if len(text) < width else text[:width - 3] + '...'
+ return text if len(text) < width else text[: width - 3] + "..."
diff --git a/abcd/frontends/commandline/config.py b/abcd/frontends/commandline/config.py
index 96f1380c..3aa21bea 100644
--- a/abcd/frontends/commandline/config.py
+++ b/abcd/frontends/commandline/config.py
@@ -21,27 +21,33 @@ def from_json(cls, filename):
@classmethod
def load(cls):
- if os.environ.get('ABCD_CONFIG') and Path(os.environ.get('ABCD_CONFIG')).is_file():
- file = Path(os.environ.get('ABCD_CONFIG'))
- elif (Path.home() / '.abcd').is_file():
- file = Path.home() / '.abcd'
+ if (
+ os.environ.get("ABCD_CONFIG")
+ and Path(os.environ.get("ABCD_CONFIG")).is_file()
+ ):
+ file = Path(os.environ.get("ABCD_CONFIG"))
+ elif (Path.home() / ".abcd").is_file():
+ file = Path.home() / ".abcd"
else:
return cls()
- logger.info('Using config file: {}'.format(file))
+ logger.info("Using config file: {}".format(file))
config = cls.from_json(file)
return config
def save(self):
- file = Path(os.environ.get('ABCD_CONFIG')) if os.environ.get('ABCD_CONFIG') \
- else Path.home() / '.abcd'
+ file = (
+ Path(os.environ.get("ABCD_CONFIG"))
+ if os.environ.get("ABCD_CONFIG")
+ else Path.home() / ".abcd"
+ )
- logger.info('The saved config\'s file: {}'.format(file))
+ logger.info("The saved config's file: {}".format(file))
- with open(str(file), 'w') as file:
+ with open(str(file), "w") as file:
json.dump(self, file)
def __repr__(self):
- return '<{} {}>'.format(self.__class__.__name__, dict.__repr__(self))
+ return "<{} {}>".format(self.__class__.__name__, dict.__repr__(self))
diff --git a/abcd/frontends/commandline/decorators.py b/abcd/frontends/commandline/decorators.py
index d46e9ced..c2439be7 100644
--- a/abcd/frontends/commandline/decorators.py
+++ b/abcd/frontends/commandline/decorators.py
@@ -19,10 +19,10 @@ def wrapper(*args, **kwargs):
def init_db(func):
def wrapper(*args, config, **kwargs):
- url = config.get('url', None)
+ url = config.get("url", None)
if url is None:
- print('Please use abcd login first!')
+ print("Please use abcd login first!")
exit(1)
db = ABCD.from_url(url=url)
@@ -32,10 +32,10 @@ def wrapper(*args, config, **kwargs):
# TODO: better ast optimisation
query_list = []
- for q in kwargs.pop('default_query', []):
+ for q in kwargs.pop("default_query", []):
query_list.append(parser(q))
- for q in kwargs.pop('query', []):
+ for q in kwargs.pop("query", []):
query_list.append(parser(q))
if not query_list:
@@ -43,7 +43,7 @@ def wrapper(*args, config, **kwargs):
elif len(query_list) == 1:
query = query_list[0]
else:
- query = ('AND', *query_list)
+ query = ("AND", *query_list)
func(*args, db=db, query=query, **kwargs)
@@ -52,8 +52,8 @@ def wrapper(*args, config, **kwargs):
def check_remote(func):
def wrapper(*args, **kwargs):
- if kwargs.pop('remote'):
- print('In read only mode, you can\'t modify the data in the database')
+ if kwargs.pop("remote"):
+ print("In read only mode, you can't modify the data in the database")
exit(1)
func(*args, **kwargs)
diff --git a/abcd/frontends/commandline/parser.py b/abcd/frontends/commandline/parser.py
index 6856bd9e..9b2c1af2 100644
--- a/abcd/frontends/commandline/parser.py
+++ b/abcd/frontends/commandline/parser.py
@@ -5,135 +5,234 @@
logger = logging.getLogger(__name__)
-parser = ArgumentParser(description='Command line interface for ABCD database')
-parser.add_argument('-v', '--verbose', help='Enable verbose mode', action='store_true')
-parser.add_argument('-q', '--query', dest='default_query', action='append', help='Filtering extra quantities',
- default=[])
-
-parser.add_argument('--remote', help='Disables all the functions which would modify the database',
- action='store_true')
-
-subparsers = parser.add_subparsers(title='Commands', dest='command', parser_class=ArgumentParser)
-
-login_parser = subparsers.add_parser('login', help='login to the database')
+parser = ArgumentParser(description="Command line interface for ABCD database")
+parser.add_argument("-v", "--verbose", help="Enable verbose mode", action="store_true")
+parser.add_argument(
+ "-q",
+ "--query",
+ dest="default_query",
+ action="append",
+ help="Filtering extra quantities",
+ default=[],
+)
+
+parser.add_argument(
+ "--remote",
+ help="Disables all the functions which would modify the database",
+ action="store_true",
+)
+
+subparsers = parser.add_subparsers(
+ title="Commands", dest="command", parser_class=ArgumentParser
+)
+
+login_parser = subparsers.add_parser("login", help="login to the database")
login_parser.set_defaults(callback_func=commands.login)
-login_parser.add_argument('-n', '--name', help='name of the database', default='default')
-login_parser.add_argument(dest='url',
- help='url of abcd api (default: http://localhost)',
- default='http://localhost')
-
-download_parser = subparsers.add_parser('download', help='download data from the database')
+login_parser.add_argument(
+ "-n", "--name", help="name of the database", default="default"
+)
+login_parser.add_argument(
+ dest="url",
+ help="url of abcd api (default: http://localhost)",
+ default="http://localhost",
+)
+
+download_parser = subparsers.add_parser(
+ "download", help="download data from the database"
+)
download_parser.set_defaults(callback_func=commands.download)
-download_parser.add_argument('-q', '--query', action='append', help='Filtering extra quantities', default=[])
-download_parser.add_argument('-f', '--format', help='Valid ASE file format (optional)', dest='fileformat', default='extxyz')
-download_parser.add_argument(dest='filename', help='name of the file to store the configurations', nargs='?')
-
-upload_parser = subparsers.add_parser('upload', help='upload any ase supported files to the database')
-upload_parser.add_argument('-e', '--extra_infos', action='append', help='Adding extra quantities')
-upload_parser.add_argument('-i', '--ignore_calc_results', action='store_true',
- help='Ignore calculators results/parameters')
-upload_parser.add_argument('--upload-duplicates', action='store_true',
- help='Upload but still report all of the duplicates')
-upload_parser.add_argument('--upload-structure-duplicates', action='store_true',
- help='Ignore all the "exact" duplicates but store all of the structural duplicates')
-upload_parser.add_argument('--upload-duplicates-replace', action='store_true',
- help='Upload everything and duplicates overwrite previously existing data')
-upload_parser.add_argument(dest='path', help='Path to the file or folder.')
+download_parser.add_argument(
+ "-q", "--query", action="append", help="Filtering extra quantities", default=[]
+)
+download_parser.add_argument(
+ "-f",
+ "--format",
+ help="Valid ASE file format (optional)",
+ dest="fileformat",
+ default="extxyz",
+)
+download_parser.add_argument(
+ dest="filename", help="name of the file to store the configurations", nargs="?"
+)
+
+upload_parser = subparsers.add_parser(
+ "upload", help="upload any ase supported files to the database"
+)
+upload_parser.add_argument(
+ "-e", "--extra_infos", action="append", help="Adding extra quantities"
+)
+upload_parser.add_argument(
+ "-i",
+ "--ignore_calc_results",
+ action="store_true",
+ help="Ignore calculators results/parameters",
+)
+upload_parser.add_argument(
+ "--upload-duplicates",
+ action="store_true",
+ help="Upload but still report all of the duplicates",
+)
+upload_parser.add_argument(
+ "--upload-structure-duplicates",
+ action="store_true",
+ help='Ignore all the "exact" duplicates but store all of the structural duplicates',
+)
+upload_parser.add_argument(
+ "--upload-duplicates-replace",
+ action="store_true",
+ help="Upload everything and duplicates overwrite previously existing data",
+)
+upload_parser.add_argument(dest="path", help="Path to the file or folder.")
upload_parser.set_defaults(callback_func=commands.upload)
-summary_parser = subparsers.add_parser('summary', help='Discovery mode')
+summary_parser = subparsers.add_parser("summary", help="Discovery mode")
summary_parser.set_defaults(callback_func=commands.summary)
-summary_parser.add_argument('-q', '--query', action='append', help='Filtering extra quantities', default=[])
-summary_parser.add_argument('-p', '--props', action='append',
- help='Selecting properties for detailed description')
-summary_parser.add_argument('-a', '--all',
- help='Show everything without truncation of strings and limits of lines',
- action='store_true', dest='print_all')
-summary_parser.add_argument('-n', '--bins', help='The number of bins of the histogram', default=10, type=int)
-summary_parser.add_argument('-t', '--trunc',
- help='Length of string before truncation',
- default=20, type=int, dest='truncate')
-
-show_parser = subparsers.add_parser('show', help='shows the first 10 items')
+summary_parser.add_argument(
+ "-q", "--query", action="append", help="Filtering extra quantities", default=[]
+)
+summary_parser.add_argument(
+ "-p",
+ "--props",
+ action="append",
+ help="Selecting properties for detailed description",
+)
+summary_parser.add_argument(
+ "-a",
+ "--all",
+ help="Show everything without truncation of strings and limits of lines",
+ action="store_true",
+ dest="print_all",
+)
+summary_parser.add_argument(
+ "-n", "--bins", help="The number of bins of the histogram", default=10, type=int
+)
+summary_parser.add_argument(
+ "-t",
+ "--trunc",
+ help="Length of string before truncation",
+ default=20,
+ type=int,
+ dest="truncate",
+)
+
+show_parser = subparsers.add_parser("show", help="shows the first 10 items")
show_parser.set_defaults(callback_func=commands.show)
-show_parser.add_argument('-q', '--query', action='append', help='Filtering extra quantities', default=[])
-show_parser.add_argument('-p', '--props', action='append',
- help='Selecting properties for detailed description')
-show_parser.add_argument('-a', '--all',
- help='Show everything without truncation of strings and limits of lines',
- action='store_true', dest='print_all')
-
-delete_parser = subparsers.add_parser('delete', help='Delete configurations from the database')
+show_parser.add_argument(
+ "-q", "--query", action="append", help="Filtering extra quantities", default=[]
+)
+show_parser.add_argument(
+ "-p",
+ "--props",
+ action="append",
+ help="Selecting properties for detailed description",
+)
+show_parser.add_argument(
+ "-a",
+ "--all",
+ help="Show everything without truncation of strings and limits of lines",
+ action="store_true",
+ dest="print_all",
+)
+
+delete_parser = subparsers.add_parser(
+ "delete", help="Delete configurations from the database"
+)
delete_parser.set_defaults(callback_func=commands.delete)
-delete_parser.add_argument('-q', '--query', action='append', help='Filtering by a query', default=[])
-delete_parser.add_argument('-y', '--yes', action='store_true', help='Do the actual deletion.')
-
-key_add_parser = subparsers.add_parser('add-key', help='Adding new key value pairs for a given query')
+delete_parser.add_argument(
+ "-q", "--query", action="append", help="Filtering by a query", default=[]
+)
+delete_parser.add_argument(
+ "-y", "--yes", action="store_true", help="Do the actual deletion."
+)
+
+key_add_parser = subparsers.add_parser(
+ "add-key", help="Adding new key value pairs for a given query"
+)
key_add_parser.set_defaults(callback_func=commands.key_add)
-key_add_parser.add_argument('-q', '--query', action='append', help='Filtering by a query', default=[])
-key_add_parser.add_argument('-y', '--yes', action='store_true', help='Overwrite?')
-key_add_parser.add_argument('keys', help='keys(=value) pairs', nargs='+')
-
-key_rename_parser = subparsers.add_parser('rename-key', help='Rename a specific keys for a given query')
+key_add_parser.add_argument(
+ "-q", "--query", action="append", help="Filtering by a query", default=[]
+)
+key_add_parser.add_argument("-y", "--yes", action="store_true", help="Overwrite?")
+key_add_parser.add_argument("keys", help="keys(=value) pairs", nargs="+")
+
+key_rename_parser = subparsers.add_parser(
+ "rename-key", help="Rename a specific keys for a given query"
+)
key_rename_parser.set_defaults(callback_func=commands.key_rename)
-key_rename_parser.add_argument('-q', '--query', action='append', help='Filtering by a query', default=[])
-key_rename_parser.add_argument('-y', '--yes', action='store_true', help='Overwrite?')
-key_rename_parser.add_argument('old_keys', help='name of the old key')
-key_rename_parser.add_argument('new_keys', help='new name of the key')
-
-key_delete_parser = subparsers.add_parser('delete-key', help='Delete all the keys for a given query')
+key_rename_parser.add_argument(
+ "-q", "--query", action="append", help="Filtering by a query", default=[]
+)
+key_rename_parser.add_argument("-y", "--yes", action="store_true", help="Overwrite?")
+key_rename_parser.add_argument("old_keys", help="name of the old key")
+key_rename_parser.add_argument("new_keys", help="new name of the key")
+
+key_delete_parser = subparsers.add_parser(
+ "delete-key", help="Delete all the keys for a given query"
+)
key_delete_parser.set_defaults(callback_func=commands.key_delete)
-key_delete_parser.add_argument('-q', '--query', action='append', help='Filtering by a query', default=[])
-key_delete_parser.add_argument('-y', '--yes', action='store_true', help='Do the actual deletion.')
-key_delete_parser.add_argument('keys', help='keys(=value) data', nargs='+')
-
-exec_parser = subparsers.add_parser('exec', help='Running custom python code')
+key_delete_parser.add_argument(
+ "-q", "--query", action="append", help="Filtering by a query", default=[]
+)
+key_delete_parser.add_argument(
+ "-y", "--yes", action="store_true", help="Do the actual deletion."
+)
+key_delete_parser.add_argument("keys", help="keys(=value) data", nargs="+")
+
+exec_parser = subparsers.add_parser("exec", help="Running custom python code")
exec_parser.set_defaults(callback_func=commands.execute)
-exec_parser.add_argument('-q', '--query', action='append', help='Filtering by a query', default=[])
-exec_parser.add_argument('-y', '--yes', action='store_true', help='Do the actual execution.')
-exec_parser.add_argument('python_code', help='Selecting properties for detailed description')
-
-server = subparsers.add_parser('server', help='Running custom python code')
+exec_parser.add_argument(
+ "-q", "--query", action="append", help="Filtering by a query", default=[]
+)
+exec_parser.add_argument(
+ "-y", "--yes", action="store_true", help="Do the actual execution."
+)
+exec_parser.add_argument(
+ "python_code", help="Selecting properties for detailed description"
+)
+
+server = subparsers.add_parser("server", help="Running custom python code")
server.set_defaults(callback_func=commands.server)
-server.add_argument('abcd_url', help='Url for abcd database.')
-server.add_argument('--api-only', action='store_true', help='Running only the API.')
-server.add_argument('-u', '--url', help='Url to run the server.', default='http://localhost:5000')
+server.add_argument("abcd_url", help="Url for abcd database.")
+server.add_argument("--api-only", action="store_true", help="Running only the API.")
+server.add_argument(
+ "-u", "--url", help="Url to run the server.", default="http://localhost:5000"
+)
def main(args=None):
kwargs = parser.parse_args(args).__dict__
- if kwargs.pop('verbose'):
+ if kwargs.pop("verbose"):
# Remove all handlers associated with the root logger object.
# https://stackoverflow.com/questions/12158048/changing-loggings-basicconfig-which-is-already-set
for handler in logging.root.handlers[:]:
logging.root.removeHandler(handler)
logging.basicConfig(level=logging.INFO)
- logger.info('Verbose mode is active')
+ logger.info("Verbose mode is active")
- if not kwargs.pop('command'):
+ if not kwargs.pop("command"):
print(parser.format_help())
return
try:
- callback_func = kwargs.pop('callback_func')
+ callback_func = kwargs.pop("callback_func")
callback_func(**kwargs)
except URLError:
- print('Wrong connection: Please check the parameters of the url!')
+ print("Wrong connection: Please check the parameters of the url!")
exit(1)
except AuthenticationError:
- print('Authentication failed: Please check the parameters of the connection!')
+ print("Authentication failed: Please check the parameters of the connection!")
exit(1)
except TimeoutError:
print("Timeout: Please check the parameters of the connection!")
exit(1)
-if __name__ == '__main__':
- main(['summary'])
- main('delete-key -q pbc pbc'.split())
+if __name__ == "__main__":
+ main(["summary"])
+ main("delete-key -q pbc pbc".split())
# main('upload -e cas -i ../../../tutorials/GB_alphaFe_001/tilt/00110391110_v6bxv2_tv0.4bxv0.2_d1.6z_traj.xyz'.split())
# main('summary -q formula~"Si2"'.split())
# main('upload -e cas -i ../../../tutorials/GB_alphaFe_001/tilt/00110391110_v6bxv2_tv0.4bxv0.2_d1.6z_traj.xyz'.split())
@@ -142,7 +241,7 @@ def main(args=None):
# main('-v summary -p energy'.split())
# main('-v summary -p *'.split())
# main('add-key -q cas selected user="cas"'.split())
- main('delete-key user'.split())
+ main("delete-key user".split())
# main(['summary', '-p', '*'])
# main(['summary', '-p', 'info.config_name, info.energy'])
# main(['summary', '-p', 'info.config_name, info.energy,info.energy;info.energy info.energy'])
diff --git a/abcd/model.py b/abcd/model.py
index c62d14ec..f4c87b61 100644
--- a/abcd/model.py
+++ b/abcd/model.py
@@ -18,13 +18,13 @@ def __init__(self, method=md5()):
def update(self, value):
if isinstance(value, int):
- self.update(str(value).encode('ascii'))
+ self.update(str(value).encode("ascii"))
elif isinstance(value, str):
- self.update(value.encode('utf-8'))
+ self.update(value.encode("utf-8"))
elif isinstance(value, float):
- self.update('{:.8e}'.format(value).encode('ascii'))
+ self.update("{:.8e}".format(value).encode("ascii"))
elif isinstance(value, (tuple, list)):
for e in value:
@@ -33,7 +33,7 @@ def update(self, value):
elif isinstance(value, (dict, UserDict)):
keys = value.keys()
for k in sorted(keys):
- self.update(k.encode('utf-8'))
+ self.update(k.encode("utf-8"))
self.update(value[k])
elif isinstance(value, datetime.datetime):
@@ -43,7 +43,9 @@ def update(self, value):
self.method.update(value)
else:
- raise ValueError("The {} type cannot be hashed! (Value: {})", format(type(value), value))
+ raise ValueError(
+ "The {} type cannot be hashed! (Value: {})", format(type(value), value)
+ )
def __call__(self):
"""Retrieve the digest of the hash."""
@@ -51,7 +53,14 @@ def __call__(self):
class AbstractModel(UserDict):
- reserved_keys = {'n_atoms', 'cell', 'pbc', 'calculator_name', 'calculator_parameters', 'derived'}
+ reserved_keys = {
+ "n_atoms",
+ "cell",
+ "pbc",
+ "calculator_name",
+ "calculator_parameters",
+ "derived",
+ }
def __init__(self, dict=None, **kwargs):
self.arrays_keys = []
@@ -64,22 +73,22 @@ def __init__(self, dict=None, **kwargs):
@property
def derived(self):
return {
- 'arrays_keys': self.arrays_keys,
- 'info_keys': self.info_keys,
- 'results_keys': self.results_keys,
- 'derived_keys': self.derived_keys
+ "arrays_keys": self.arrays_keys,
+ "info_keys": self.info_keys,
+ "results_keys": self.results_keys,
+ "derived_keys": self.derived_keys,
}
def __getitem__(self, key):
- if key == 'derived':
+ if key == "derived":
return self.derived
return super().__getitem__(key)
def __setitem__(self, key, value):
- if key == 'derived':
+ if key == "derived":
# raise KeyError('Please do not use "derived" as key because it is protected!')
# Silent return to avoid raising error in pymongo package
return
@@ -99,31 +108,31 @@ def convert(self, value):
def update_key_category(self, key, value):
- if key == '_id':
+ if key == "_id":
# raise KeyError('Please do not use "derived" as key because it is protected!')
return
- for category in ('arrays_keys', 'info_keys', 'results_keys', 'derived_keys'):
+ for category in ("arrays_keys", "info_keys", "results_keys", "derived_keys"):
if key in self.derived[category]:
return
- if key in ('positions', 'forces'):
- self.derived['arrays_keys'].append(key)
+ if key in ("positions", "forces"):
+ self.derived["arrays_keys"].append(key)
return
- if key in ('n_atoms', 'cell', 'pbc'):
- self.derived['info_keys'].append(key)
+ if key in ("n_atoms", "cell", "pbc"):
+ self.derived["info_keys"].append(key)
return
# Guess the category based in the shape of the value
- n_atoms = self['n_atoms']
+ n_atoms = self["n_atoms"]
if isinstance(value, (np.ndarray, list)) and len(value) == n_atoms:
- self.derived['arrays_keys'].append(key)
+ self.derived["arrays_keys"].append(key)
else:
- self.derived['info_keys'].append(key)
+ self.derived["info_keys"].append(key)
def __delitem__(self, key):
- for category in ('arrays_keys', 'info_keys', 'results_keys', 'derived_keys'):
+ for category in ("arrays_keys", "info_keys", "results_keys", "derived_keys"):
if key in self.derived[category]:
self.derived[category].remove(key)
break
@@ -133,34 +142,44 @@ def __delitem__(self, key):
def __iter__(self):
for item in super().__iter__():
yield item
- yield 'derived'
+ yield "derived"
@classmethod
def from_atoms(cls, atoms: Atoms, extra_info=None, store_calc=True):
"""ASE's original implementation"""
- reserved_keys = {'n_atoms', 'cell', 'pbc', 'calculator_name', 'calculator_parameters', 'derived', 'formula'}
+ reserved_keys = {
+ "n_atoms",
+ "cell",
+ "pbc",
+ "calculator_name",
+ "calculator_parameters",
+ "derived",
+ "formula",
+ }
arrays_keys = set(atoms.arrays.keys())
info_keys = set(atoms.info.keys())
- results_keys = set(atoms.calc.results.keys()) if store_calc and atoms.calc else {}
+ results_keys = (
+ set(atoms.calc.results.keys()) if store_calc and atoms.calc else {}
+ )
all_keys = (reserved_keys, arrays_keys, info_keys, results_keys)
if len(set.union(*all_keys)) != sum(map(len, all_keys)):
print(all_keys)
- raise ValueError('All the keys must be unique!')
+ raise ValueError("All the keys must be unique!")
item = cls()
n_atoms = len(atoms)
dct = {
- 'n_atoms': n_atoms,
- 'cell': atoms.cell.tolist(),
- 'pbc': atoms.pbc.tolist(),
- 'formula': atoms.get_chemical_formula()
+ "n_atoms": n_atoms,
+ "cell": atoms.cell.tolist(),
+ "pbc": atoms.pbc.tolist(),
+ "formula": atoms.get_chemical_formula(),
}
- info_keys.update({'n_atoms', 'cell', 'pbc', 'formula'})
+ info_keys.update({"n_atoms", "cell", "pbc", "formula"})
for key, value in atoms.arrays.items():
if isinstance(value, np.ndarray):
@@ -175,9 +194,9 @@ def from_atoms(cls, atoms: Atoms, extra_info=None, store_calc=True):
dct[key] = value
if store_calc and atoms.calc:
- dct['calculator_name'] = atoms.calc.__class__.__name__
- dct['calculator_parameters'] = atoms.calc.todict()
- info_keys.update({'calculator_name', 'calculator_parameters'})
+ dct["calculator_name"] = atoms.calc.__class__.__name__
+ dct["calculator_parameters"] = atoms.calc.todict()
+ info_keys.update({"calculator_name", "calculator_parameters"})
for key, value in atoms.calc.results.items():
@@ -205,26 +224,22 @@ def to_ase(self):
arrays_keys = set(self.arrays_keys)
info_keys = set(self.info_keys)
- cell = self.pop('cell', None)
- pbc = self.pop('pbc', None)
- numbers = self.pop('numbers', None)
- positions = self.pop('positions', None)
- results_keys = self.derived['results_keys']
+ cell = self.pop("cell", None)
+ pbc = self.pop("pbc", None)
+ numbers = self.pop("numbers", None)
+ positions = self.pop("positions", None)
+ results_keys = self.derived["results_keys"]
- info_keys -= {'cell', 'pbc'}
- arrays_keys -= {'numbers', 'positions'}
+ info_keys -= {"cell", "pbc"}
+ arrays_keys -= {"numbers", "positions"}
- atoms = Atoms(
- cell=cell,
- pbc=pbc,
- numbers=numbers,
- positions=positions)
+ atoms = Atoms(cell=cell, pbc=pbc, numbers=numbers, positions=positions)
- if 'calculator_name' in self:
+ if "calculator_name" in self:
# calculator_name = self['info'].pop('calculator_name')
# atoms.calc = get_calculator(data['results']['calculator_name'])(**params)
- params = self.pop('calculator_parameters', {})
+ params = self.pop("calculator_parameters", {})
atoms.calc = SinglePointCalculator(atoms, **params)
atoms.calc.results.update((key, self[key]) for key in results_keys)
@@ -235,38 +250,38 @@ def to_ase(self):
return atoms
def pre_save(self):
- self.derived_keys = ['elements', 'username', 'uploaded', 'modified']
+ self.derived_keys = ["elements", "username", "uploaded", "modified"]
- cell = self['cell']
+ cell = self["cell"]
if cell:
volume = abs(np.linalg.det(cell)) # atoms.get_volume()
- self['volume'] = volume
- self.derived_keys.append('volume')
+ self["volume"] = volume
+ self.derived_keys.append("volume")
- virial = self.get('virial')
+ virial = self.get("virial")
if virial:
# pressure P = -1/3 Tr(stress) = -1/3 Tr(virials/volume)
- self['pressure'] = -1 / 3 * np.trace(virial / volume)
- self.derived_keys.append('pressure')
+ self["pressure"] = -1 / 3 * np.trace(virial / volume)
+ self.derived_keys.append("pressure")
# 'elements': Counter(atoms.get_chemical_symbols()),
- self['elements'] = Counter(str(element) for element in self['numbers'])
+ self["elements"] = Counter(str(element) for element in self["numbers"])
- self['username'] = getpass.getuser()
+ self["username"] = getpass.getuser()
- if not self.get('uploaded'):
- self['uploaded'] = datetime.datetime.utcnow()
+ if not self.get("uploaded"):
+ self["uploaded"] = datetime.datetime.utcnow()
- self['modified'] = datetime.datetime.utcnow()
+ self["modified"] = datetime.datetime.utcnow()
m = Hasher()
- for key in ('numbers', 'positions', 'cell', 'pbc'):
+ for key in ("numbers", "positions", "cell", "pbc"):
m.update(self[key])
- self.derived_keys.append('hash_structure')
- self['hash_structure'] = m()
+ self.derived_keys.append("hash_structure")
+ self["hash_structure"] = m()
m = Hasher()
for key in self.arrays_keys:
@@ -274,11 +289,11 @@ def pre_save(self):
for key in self.info_keys:
m.update(self[key])
- self.derived_keys.append('hash')
- self['hash'] = m()
+ self.derived_keys.append("hash")
+ self["hash"] = m()
-if __name__ == '__main__':
+if __name__ == "__main__":
import io
from pprint import pprint
from ase.io import read
@@ -286,7 +301,7 @@ def pre_save(self):
logging.basicConfig(level=logging.INFO)
# from ase.io import jsonio
- atoms = read('test.xyz', format='xyz', index=0)
+ atoms = read("test.xyz", format="xyz", index=0)
atoms.set_cell([1, 1, 1])
print(atoms)
diff --git a/abcd/parsers/extras.py b/abcd/parsers/extras.py
index 939210bb..c007acf6 100644
--- a/abcd/parsers/extras.py
+++ b/abcd/parsers/extras.py
@@ -90,40 +90,43 @@ def string(self, s):
# parser = Lark(grammar, parser='lalr', lexer='contextual', debug=False)
-parser = Lark(grammar, parser='lalr', lexer='contextual', transformer=TreeToDict(), debug=False)
-
-if __name__ == '__main__':
- test_string = ' '.join([
- ' ' # start with a separator
- 'flag',
- 'quotedd_string="quoteddd value"',
- r'quotedddd_string_escaped="esc\"aped"',
- 'false_value = F',
- 'integer=22',
- 'floating=1.1',
- 'int_array={1 2 3}',
- 'scientific_float=1.2e7',
- 'scientific_float_2=5e-6',
- 'scientific_float_array="1.2 2.2e3 4e1 3.3e-1 2e-2"',
- 'not_array="1.2 3.4 text"',
- 'array_nested=[[1,2],[3,4]] ' # gets flattented if not 3x3
- 'array_many_other_quotes=({[4 8 12]})',
- 'array_boolean={T F T F}',
- 'array_boolean_2=" T, F, T " ' # leading spaces
- # 'not_bool_array=[T F S]',
- # # read and write
- # u'\xfcnicode_key=val\xfce',
- # # u'unquoted_special_value=a_to_Z_$%%^&*\xfc\u2615',
- # '2body=33.3',
- 'hyphen-ated',
- # # # parse only
- 'comma_separated="7, 4, -1"',
- 'array_bool_commas=[T, T, F, T]',
- # # 'Properties=species:S:1:pos:R:3',
- # 'double_equals=abc=xyz',
- 'multiple_separators ',
- 'trailing'
- ])
+parser = Lark(
+ grammar, parser="lalr", lexer="contextual", transformer=TreeToDict(), debug=False
+)
+
+if __name__ == "__main__":
+ test_string = " ".join(
+ [
+ " " "flag", # start with a separator
+ 'quotedd_string="quoteddd value"',
+ r'quotedddd_string_escaped="esc\"aped"',
+ "false_value = F",
+ "integer=22",
+ "floating=1.1",
+ "int_array={1 2 3}",
+ "scientific_float=1.2e7",
+ "scientific_float_2=5e-6",
+ 'scientific_float_array="1.2 2.2e3 4e1 3.3e-1 2e-2"',
+ 'not_array="1.2 3.4 text"',
+ "array_nested=[[1,2],[3,4]] " # gets flattented if not 3x3
+ "array_many_other_quotes=({[4 8 12]})",
+ "array_boolean={T F T F}",
+ 'array_boolean_2=" T, F, T " ' # leading spaces
+ # 'not_bool_array=[T F S]',
+ # # read and write
+ # u'\xfcnicode_key=val\xfce',
+ # # u'unquoted_special_value=a_to_Z_$%%^&*\xfc\u2615',
+ # '2body=33.3',
+ "hyphen-ated",
+ # # # parse only
+ 'comma_separated="7, 4, -1"',
+ "array_bool_commas=[T, T, F, T]",
+ # # 'Properties=species:S:1:pos:R:3',
+ # 'double_equals=abc=xyz',
+ "multiple_separators ",
+ "trailing",
+ ]
+ )
j = parser.parse(test_string)
diff --git a/abcd/parsers/queries.py b/abcd/parsers/queries.py
index 707fd4bc..41ef1c33 100644
--- a/abcd/parsers/queries.py
+++ b/abcd/parsers/queries.py
@@ -90,17 +90,17 @@ class TreeTransformer(Transformer):
def array(self, *items):
return list(items)
- true = lambda _: ('VALUE', True)
- false = lambda _: ('VALUE', False)
+ true = lambda _: ("VALUE", True)
+ false = lambda _: ("VALUE", False)
def float(self, number):
- return 'NUMBER', float(number)
+ return "NUMBER", float(number)
def int(self, number):
- return 'NUMBER', int(number)
+ return "NUMBER", int(number)
def string(self, s):
- return 'STRING', s[1:-1].replace('\\"', '"')
+ return "STRING", s[1:-1].replace('\\"', '"')
def single_statement(self, expression):
return expression
@@ -109,24 +109,24 @@ def multi_statement(self, statement, operator, expression):
return operator.type, statement, expression
def single_expression(self, name):
- return 'NAME', str(name)
+ return "NAME", str(name)
def grouped_expression(self, statement):
# return statement
- return 'GROUP', statement
+ return "GROUP", statement
def operator_expression(self, name, operator, value):
- return operator.type, ('NAME', str(name)), value
+ return operator.type, ("NAME", str(name)), value
def reversed_expression(self, value, operator, name):
- return operator.type, ('NAME', str(name)), value
+ return operator.type, ("NAME", str(name)), value
def negation_expression(self, operator, expression):
return operator.type, expression
class Parser:
- parser = Lark(grammar, start='statement')
+ parser = Lark(grammar, start="statement")
transformer = TreeTransformer()
def parse(self, string):
@@ -138,29 +138,29 @@ def __call__(self, string):
parser = Parser()
-if __name__ == '__main__':
+if __name__ == "__main__":
# logging.basicConfig(level=logging.DEBUG)
logging.basicConfig(level=logging.INFO)
queries = (
- ' ',
- 'single',
- 'not single',
- 'operator_gt > 23 ',
- 'operator_gt > -2.31e-5 ',
+ " ",
+ "single",
+ "not single",
+ "operator_gt > 23 ",
+ "operator_gt > -2.31e-5 ",
'string = "some string"',
'regexp ~ ".*H"',
- 'aa & not bb',
- 'aa & bb > 23.54 | cc & dd',
+ "aa & not bb",
+ "aa & bb > 23.54 | cc & dd",
# 'aa bb > 22 cc > 33 dd > 44 ',
- 'aa and bb > 22 and cc > 33 and dd > 44 ',
- '((aa and bb > 22) and cc > 33) and dd > 44 ',
- '(aa and bb > 22) and (cc > 33 and dd > 44) ',
- '(aa and bb > 22 and cc > 33 and dd > 44) ',
- 'aa and bb > 23.54 or 22 in cc and dd',
- 'aa & bb > 23.54 | (22 in cc & dd)',
- 'aa and bb > 23.54 or (22 in cc and dd)',
- 'aa and not (bb > 23.54 or (22 in cc and dd))',
+ "aa and bb > 22 and cc > 33 and dd > 44 ",
+ "((aa and bb > 22) and cc > 33) and dd > 44 ",
+ "(aa and bb > 22) and (cc > 33 and dd > 44) ",
+ "(aa and bb > 22 and cc > 33 and dd > 44) ",
+ "aa and bb > 23.54 or 22 in cc and dd",
+ "aa & bb > 23.54 | (22 in cc & dd)",
+ "aa and bb > 23.54 or (22 in cc and dd)",
+ "aa and not (bb > 23.54 or (22 in cc and dd))",
# 'expression = (bb/3-1)*cc',
# 'energy/n_atoms > 3',
# '1=3',
@@ -176,7 +176,7 @@ def __call__(self, string):
# print(parser.parse(query).pretty())
try:
tree = parser.parse(query)
- logger.info('=> tree: {}'.format(tree))
- logger.info('==> ast: {}'.format(parser(query)))
+ logger.info("=> tree: {}".format(tree))
+ logger.info("==> ast: {}".format(parser(query)))
except LarkError:
raise NotImplementedError
diff --git a/abcd/parsers/queries_new.py b/abcd/parsers/queries_new.py
index 8f176d0c..22004d11 100644
--- a/abcd/parsers/queries_new.py
+++ b/abcd/parsers/queries_new.py
@@ -8,21 +8,21 @@
# TODO: Reversed operator in the grammar (value op prop VS prop op value VS IN)
-class DebugTransformer(Transformer): # pragma: no cover
+class DebugTransformer(Transformer): # pragma: no cover
def __init__(self):
super().__init__()
def __default__(self, data, children, meta):
- print('Node: ', data, children)
+ print("Node: ", data, children)
return data
-if __name__ == '__main__':
+if __name__ == "__main__":
logging.basicConfig(level=logging.INFO)
- with open('query_new.lark') as file:
- parser = Lark(file.read(), start='expression')
+ with open("query_new.lark") as file:
+ parser = Lark(file.read(), start="expression")
transformer = DebugTransformer()
@@ -72,30 +72,30 @@ def __default__(self, data, children, meta):
# )
queries = (
- ' ',
- 'single',
- 'not single',
- 'operator_gt > 23 ',
- 'operator_gt > -2.31e-5 ',
+ " ",
+ "single",
+ "not single",
+ "operator_gt > 23 ",
+ "operator_gt > -2.31e-5 ",
'string = "some string"',
'regexp ~ ".*H"',
- 'aa & not bb',
- 'aa & bb > 23.54 | cc & dd',
- 'aa and bb > 22 and cc > 33 and dd > 44 ',
- '((aa and bb > 22) and cc > 33) and dd > 44 ',
- '(aa and bb > 22) and (cc > 33 and dd > 44) ',
- '(aa and bb > 22 and cc > 33 and dd > 44) ',
- 'aa and bb > 23.54 or 22 in cc and dd',
- 'aa & bb > 23.54 | (22 in cc & dd)',
- 'aa and bb > 23.54 or (22 in cc and dd)',
- 'aa and not (bb > 23.54 or (22 in cc and dd))',
- 'expression = (bb/3-1)*cc',
- 'energy/n_atoms > 3',
- '1=3',
- 'all(aa) > 3',
- 'any(aa) > 3',
- 'aa = False',
- 'aa = [True, True, True]',
+ "aa & not bb",
+ "aa & bb > 23.54 | cc & dd",
+ "aa and bb > 22 and cc > 33 and dd > 44 ",
+ "((aa and bb > 22) and cc > 33) and dd > 44 ",
+ "(aa and bb > 22) and (cc > 33 and dd > 44) ",
+ "(aa and bb > 22 and cc > 33 and dd > 44) ",
+ "aa and bb > 23.54 or 22 in cc and dd",
+ "aa & bb > 23.54 | (22 in cc & dd)",
+ "aa and bb > 23.54 or (22 in cc and dd)",
+ "aa and not (bb > 23.54 or (22 in cc and dd))",
+ "expression = (bb/3-1)*cc",
+ "energy/n_atoms > 3",
+ "1=3",
+ "all(aa) > 3",
+ "any(aa) > 3",
+ "aa = False",
+ "aa = [True, True, True]",
)
for query in queries:
diff --git a/abcd/server/__init__.py b/abcd/server/__init__.py
index 86601b85..3b21e3bb 100644
--- a/abcd/server/__init__.py
+++ b/abcd/server/__init__.py
@@ -2,5 +2,5 @@
app = create_app()
-if __name__ == '__main__':
- app.run(host='0.0.0.0')
+if __name__ == "__main__":
+ app.run(host="0.0.0.0")
diff --git a/abcd/server/app/__init__.py b/abcd/server/app/__init__.py
index bc4614f6..d7176a42 100644
--- a/abcd/server/app/__init__.py
+++ b/abcd/server/app/__init__.py
@@ -11,34 +11,34 @@
def create_app(abcd_url=None):
# Define the WSGI application object
app = Flask(__name__)
- app.logger.info('Creating an application')
+ app.logger.info("Creating an application")
- app.config['SECRET_KEY'] = os.getenv('SECRET_KEY', os.urandom(12).hex())
- app.config['ABCD_URL'] = os.getenv('ABCD_URL', 'mongodb://localhost:27017/abcd')
+ app.config["SECRET_KEY"] = os.getenv("SECRET_KEY", os.urandom(12).hex())
+ app.config["ABCD_URL"] = os.getenv("ABCD_URL", "mongodb://localhost:27017/abcd")
# Initialize extensions/add-ons/plugins.
nav.init_app(app)
- register_renderer(app, 'BootstrapRenderer', BootstrapRenderer)
- register_renderer(app, 'DatabaseNav', DatabaseNav)
+ register_renderer(app, "BootstrapRenderer", BootstrapRenderer)
+ register_renderer(app, "DatabaseNav", DatabaseNav)
db.init_app(app)
# Setup redirects and register blueprints.
app.register_blueprint(index.bp)
- app.register_blueprint(database.bp, url_prefix='/db')
- app.register_blueprint(api.bp, url_prefix='/api')
+ app.register_blueprint(database.bp, url_prefix="/db")
+ app.register_blueprint(api.bp, url_prefix="/api")
@app.errorhandler(404)
def not_found(error):
- return render_template('404.html'), 404
+ return render_template("404.html"), 404
return app
-if __name__ == '__main__':
+if __name__ == "__main__":
import logging
logging.basicConfig(level=logging.DEBUG)
app = create_app()
- app.run(host='0.0.0.0', debug=True)
+ app.run(host="0.0.0.0", debug=True)
diff --git a/abcd/server/app/nav.py b/abcd/server/app/nav.py
index 09af4385..ec8ea2d1 100644
--- a/abcd/server/app/nav.py
+++ b/abcd/server/app/nav.py
@@ -16,29 +16,29 @@ def __init__(self, title, *args, **kwargs):
@nav.navigation()
def main_navbar():
return TopNavbar(
- 'ABCD',
- View('Home', 'index.index'),
- View('API', 'api.index'),
- View('Databases', 'database.database', database_name='default'),
- Link('Docs', 'https://libatoms.github.io/abcd/'),
- Link('Github', 'https://github.com/libatoms/abcd'),
+ "ABCD",
+ View("Home", "index.index"),
+ View("API", "api.index"),
+ View("Databases", "database.database", database_name="default"),
+ Link("Docs", "https://libatoms.github.io/abcd/"),
+ Link("Github", "https://github.com/libatoms/abcd"),
)
@nav.navigation()
def database_navbar():
return Navbar(
- '',
- View('Database', 'database.database'),
- Link('Collections', '#'),
- Link('History', '#'),
- Link('Statistics', '#'),
- View('Settings', 'database.settings'),
+ "",
+ View("Database", "database.database"),
+ Link("Collections", "#"),
+ Link("History", "#"),
+ Link("Statistics", "#"),
+ View("Settings", "database.settings"),
)
class DatabaseNav(Renderer):
- def __init__(self, database_name='atoms'):
+ def __init__(self, database_name="atoms"):
self.database_name = database_name
def visit_Navbar(self, node):
@@ -50,22 +50,24 @@ def visit_Navbar(self, node):
return root
def visit_Text(self, node):
- return tags.li(tags.a(node.text, _class='nav-link disabled'), _class="nav-item")
+ return tags.li(tags.a(node.text, _class="nav-link disabled"), _class="nav-item")
def visit_Link(self, node):
- item = tags.li(_class='nav-item')
- item.add(tags.a(node.text, href=node.get_url(), _class='nav-link'))
+ item = tags.li(_class="nav-item")
+ item.add(tags.a(node.text, href=node.get_url(), _class="nav-link"))
return item
def visit_View(self, node):
# Dinamically modify the url
- node.url_for_kwargs.update({'database_name': self.database_name})
+ node.url_for_kwargs.update({"database_name": self.database_name})
- item = tags.li(_class='nav-item')
- item.add(tags.a(node.text, href=node.get_url(), title=node.text, _class='nav-link'))
+ item = tags.li(_class="nav-item")
+ item.add(
+ tags.a(node.text, href=node.get_url(), title=node.text, _class="nav-link")
+ )
if node.active:
- item['class'] = 'nav-item active'
+ item["class"] = "nav-item active"
return item
@@ -78,34 +80,41 @@ def __init__(self, nav_id=None):
def visit_Navbar(self, node):
node_id = self.id or sha1(str(id(node)).encode()).hexdigest()
- root = tags.nav(_class='navbar navbar-expand-md navbar-dark bg-dark fixed-top')
+ root = tags.nav(_class="navbar navbar-expand-md navbar-dark bg-dark fixed-top")
# title may also have a 'get_url()' method, in which case we render
# a brand-link
if node.title is not None:
- if hasattr(node.title, 'get_url'):
- root.add(tags.a(node.title.text, _class='navbar-brand',
- href=node.title.get_url()))
+ if hasattr(node.title, "get_url"):
+ root.add(
+ tags.a(
+ node.title.text,
+ _class="navbar-brand",
+ href=node.title.get_url(),
+ )
+ )
else:
- root.add(tags.span(node.title, _class='navbar-brand'))
+ root.add(tags.span(node.title, _class="navbar-brand"))
btn = root.add(tags.button())
- btn['type'] = 'button'
- btn['class'] = 'navbar-toggler'
- btn['data-toggle'] = 'collapse'
- btn['data-target'] = '#' + node_id
- btn['aria-controls'] = 'navbarCollapse'
- btn['aria-expanded'] = 'false'
- btn['aria-label'] = "Toggle navigation"
+ btn["type"] = "button"
+ btn["class"] = "navbar-toggler"
+ btn["data-toggle"] = "collapse"
+ btn["data-target"] = "#" + node_id
+ btn["aria-controls"] = "navbarCollapse"
+ btn["aria-expanded"] = "false"
+ btn["aria-label"] = "Toggle navigation"
- btn.add(tags.span('', _class='navbar-toggler-icon'))
+ btn.add(tags.span("", _class="navbar-toggler-icon"))
- bar = root.add(tags.div(
- _class='navbar-collapse collapse',
- id=node_id,
- ))
+ bar = root.add(
+ tags.div(
+ _class="navbar-collapse collapse",
+ id=node_id,
+ )
+ )
- bar_list = bar.add(tags.ul(_class='navbar-nav mr-auto'))
+ bar_list = bar.add(tags.ul(_class="navbar-nav mr-auto"))
for item in node.items:
bar_list.add(self.visit(item))
@@ -117,56 +126,58 @@ def visit_Navbar(self, node):
# search_input['aria-label'] = "Search"
search_btn = search_form.add(tags.button(_class="btn btn-success my-2 my-sm-0"))
- search_btn['type'] = "submit"
- search_btn.add_raw_string('+')
+ search_btn["type"] = "submit"
+ search_btn.add_raw_string("+")
search_btn = search_form.add(tags.button(_class="btn btn-success my-2 my-sm-0"))
- search_btn['type'] = "submit"
- search_btn.add_raw_string('Login')
+ search_btn["type"] = "submit"
+ search_btn.add_raw_string("Login")
return root
def visit_Text(self, node):
if self._in_dropdown:
- return tags.a(node.text, _class='dropdown-item disabled')
+ return tags.a(node.text, _class="dropdown-item disabled")
- return tags.li(tags.a(node.text, _class='nav-link disabled'), _class="nav-item")
+ return tags.li(tags.a(node.text, _class="nav-link disabled"), _class="nav-item")
def visit_Link(self, node):
if self._in_dropdown:
- return tags.a(node.text, href=node.get_url(), _class='dropdown-item')
+ return tags.a(node.text, href=node.get_url(), _class="dropdown-item")
- item = tags.li(_class='nav-item')
- item.add(tags.a(node.text, href=node.get_url(), _class='nav-link'))
+ item = tags.li(_class="nav-item")
+ item.add(tags.a(node.text, href=node.get_url(), _class="nav-link"))
return item
def visit_View(self, node):
if self._in_dropdown:
- return tags.a(node.text, href=node.get_url(), _class='dropdown-item')
+ return tags.a(node.text, href=node.get_url(), _class="dropdown-item")
- item = tags.li(_class='nav-item')
- item.add(tags.a(node.text, href=node.get_url(), title=node.text, _class='nav-link'))
+ item = tags.li(_class="nav-item")
+ item.add(
+ tags.a(node.text, href=node.get_url(), title=node.text, _class="nav-link")
+ )
if node.active:
- item['class'] = 'nav-item active'
+ item["class"] = "nav-item active"
return item
def visit_Subgroup(self, node):
if self._in_dropdown:
- raise RuntimeError('Cannot render nested Subgroups')
+ raise RuntimeError("Cannot render nested Subgroups")
- li = tags.li(_class='nav-item dropdown')
+ li = tags.li(_class="nav-item dropdown")
if node.active:
- li['class'] = 'nav-item dropdown active'
+ li["class"] = "nav-item dropdown active"
- a = li.add(tags.a(node.title, href='#', _class='nav-link dropdown-toggle'))
- a['data-toggle'] = 'dropdown'
- a['aria-haspopup'] = 'true'
- a['aria-expanded'] = 'false'
+ a = li.add(tags.a(node.title, href="#", _class="nav-link dropdown-toggle"))
+ a["data-toggle"] = "dropdown"
+ a["aria-haspopup"] = "true"
+ a["aria-expanded"] = "false"
- dropdown_div = li.add(tags.div(_class='dropdown-menu'))
+ dropdown_div = li.add(tags.div(_class="dropdown-menu"))
self._in_dropdown = True
for item in node.items:
@@ -177,6 +188,6 @@ def visit_Subgroup(self, node):
def visit_Separator(self, node):
if self._in_dropdown:
- return tags.div(_class='dropdown-divider')
+ return tags.div(_class="dropdown-divider")
- raise RuntimeError('Cannot render separator outside Subgroup.')
+ raise RuntimeError("Cannot render separator outside Subgroup.")
diff --git a/abcd/server/app/views/api.py b/abcd/server/app/views/api.py
index 7a02d307..b3aa6d25 100644
--- a/abcd/server/app/views/api.py
+++ b/abcd/server/app/views/api.py
@@ -1,21 +1,20 @@
from flask import Blueprint, Response, make_response, jsonify, request
-bp = Blueprint('api', __name__)
+bp = Blueprint("api", __name__)
-@bp.route('/')
+@bp.route("/")
def index():
- return Response('ok', status=200)
+ return Response("ok", status=200)
+
# endpoint to create new user
@bp.route("/calculation", methods=["POST"])
def query_calculation():
- response = {
- 'query': request.json,
- 'results': []
- }
+ response = {"query": request.json, "results": []}
return jsonify(response)
+
# # endpoint to show all users
# @bp.route("/calculation", methods=["GET"])
# def get_calculation():
diff --git a/abcd/server/app/views/database.py b/abcd/server/app/views/database.py
index 52dc0588..8569dcd2 100644
--- a/abcd/server/app/views/database.py
+++ b/abcd/server/app/views/database.py
@@ -1,31 +1,31 @@
from flask import Blueprint, render_template, request
from flask import Response
-bp = Blueprint('database', __name__)
+bp = Blueprint("database", __name__)
-@bp.route('/')
+@bp.route("/")
def index():
return Response(status=200)
# Our index-page just shows a quick explanation. Check out the template
# "templates/index.html" documentation for more details.
-@bp.route('//', methods=['GET'])
+@bp.route("//", methods=["GET"])
def database(database_name):
# data = Atoms.objects()
# list(Atoms.objects.aggregate({'$unwind': '$derived.arrays_keys'}, {'$group': {'_id': '$derived.arrays_keys', 'count': {'$sum': 1}}}))
- if request.method == 'POST':
- print('POST')
+ if request.method == "POST":
+ print("POST")
info = {
- 'name': database_name,
- 'description': 'Vivamus sagittis lacus vel augue laoreet rutrum faucibus dolor auctor. Duis mollis, est non commodo luctus.',
- 'columns': [
- {'slug': 'formula', 'name': 'Formula'},
- {'slug': 'energy', 'name': 'Energy'},
- {'slug': 'derived.n_atoms', 'name': "# of atoms"}
+ "name": database_name,
+ "description": "Vivamus sagittis lacus vel augue laoreet rutrum faucibus dolor auctor. Duis mollis, est non commodo luctus.",
+ "columns": [
+ {"slug": "formula", "name": "Formula"},
+ {"slug": "energy", "name": "Energy"},
+ {"slug": "derived.n_atoms", "name": "# of atoms"},
],
}
@@ -34,23 +34,23 @@ def database(database_name):
# page = request.args.get('page', 1, type=int)
# paginated_atoms = atoms.paginate(page, per_page=10)
- return render_template("database/database.html",
- info=info,
- atoms=paginated_atoms)
+ return render_template("database/database.html", info=info, atoms=paginated_atoms)
# Our index-page just shows a quick explanation. Check out the template
# "templates/index.html" documentation for more details.
-@bp.route('//settings')
+@bp.route("//settings")
def settings(database_name):
info = {
- 'name': database_name,
- 'description': 'Vivamus sagittis lacus vel augue laoreet rutrum faucibus dolor auctor. Duis mollis, est non commodo luctus.',
- 'columns': [
- {'slug': 'formula', 'name': 'Formula'},
- {'slug': 'energy', 'name': 'Energy'},
- {'slug': 'derived.n_atoms', 'name': "# of atoms"}
+ "name": database_name,
+ "description": "Vivamus sagittis lacus vel augue laoreet rutrum faucibus dolor auctor. Duis mollis, est non commodo luctus.",
+ "columns": [
+ {"slug": "formula", "name": "Formula"},
+ {"slug": "energy", "name": "Energy"},
+ {"slug": "derived.n_atoms", "name": "# of atoms"},
],
- 'public': True
+ "public": True,
}
- return render_template("database/settings.html", database_name=database_name, info=info)
+ return render_template(
+ "database/settings.html", database_name=database_name, info=info
+ )
diff --git a/abcd/server/app/views/index.py b/abcd/server/app/views/index.py
index b283d999..37faa52a 100644
--- a/abcd/server/app/views/index.py
+++ b/abcd/server/app/views/index.py
@@ -4,24 +4,24 @@
# from flask import jsonify
# import requests
-bp = Blueprint('index', __name__)
+bp = Blueprint("index", __name__)
-@bp.route('/')
+@bp.route("/")
def index():
return render_template("index.html")
-@bp.route('/login/')
+@bp.route("/login/")
def login():
return render_template("login.html")
-@bp.route('/new/')
+@bp.route("/new/")
def new():
return render_template("new.html")
-@bp.route('/graphql')
+@bp.route("/graphql")
def graphql():
- return render_template("graphql.html", url=url_for('api.graphql'))
+ return render_template("graphql.html", url=url_for("api.graphql"))
diff --git a/tests/test_database.py b/tests/test_database.py
index 09340849..82cddfab 100644
--- a/tests/test_database.py
+++ b/tests/test_database.py
@@ -8,9 +8,9 @@
@pytest.fixture
-@mongomock.patch(servers=(('localhost', 27017),))
+@mongomock.patch(servers=(("localhost", 27017),))
def abcd_mongodb():
- url = 'mongodb://localhost'
+ url = "mongodb://localhost"
abcd = ABCD.from_url(url)
abcd.print_info()
@@ -22,13 +22,15 @@ def test_thing(abcd_mongodb):
def test_push(abcd_mongodb):
- xyz = StringIO("""2
+ xyz = StringIO(
+ """2
Properties=species:S:1:pos:R:3 s="sadf" _vtk_test="t e s t _ s t r" pbc="F F F"
-Si 0.00000000 0.00000000 0.00000000
-Si 0.00000000 0.00000000 0.00000000
-""")
+Si 0.00000000 0.00000000 0.00000000
+Si 0.00000000 0.00000000 0.00000000
+"""
+ )
- atoms = read(xyz, format='extxyz')
+ atoms = read(xyz, format="extxyz")
atoms.set_cell([1, 1, 1])
abcd_mongodb.destroy()
diff --git a/tests/test_parsers.py b/tests/test_parsers.py
index 02a6d98a..5a2211e0 100644
--- a/tests/test_parsers.py
+++ b/tests/test_parsers.py
@@ -4,7 +4,6 @@
class TestParsingExtras:
-
@pytest.fixture
def parser(self):
return extras_parser
@@ -140,16 +139,15 @@ def test_composite(self, parser):
'a3x3_array="1 4 7 2 5 8 3 6 9" ' # fortran ordering
'Lattice=" 4.3 0.0 0.0 0.0 3.3 0.0 0.0 0.0 7.0 " ' # spaces in array
'comma_separated="7, 4, -1"',
- 'array_boolean_2=" T, F, T " ' # leading spaces
- 'not_array="1.2 3.4 text"',
+ 'array_boolean_2=" T, F, T " ' 'not_array="1.2 3.4 text"', # leading spaces
"not_bool_array=[T F S]",
],
)
- def test_missing(self, string): ...
+ def test_missing(self, string):
+ ...
class TestParsingQueries:
-
@pytest.fixture
def parser(self):
return queries_parser
@@ -190,7 +188,8 @@ def test_combination(self, parser, string, expected):
("any(aa) > 3", {}),
],
)
- def test_expressions(self, case): ...
+ def test_expressions(self, case):
+ ...
@pytest.mark.skip("known issues / future features")
@pytest.mark.parametrize(
@@ -203,4 +202,5 @@ def test_expressions(self, case): ...
("aa and (bb > 23.54 or (22 in cc and dd))", {}),
],
)
- def test_expressions(self, case): ...
+ def test_expressions(self, case):
+ ...
diff --git a/tutorials/gb_upload.py b/tutorials/gb_upload.py
index 1b0680d8..3d276de2 100644
--- a/tutorials/gb_upload.py
+++ b/tutorials/gb_upload.py
@@ -1,23 +1,18 @@
import sys
from pathlib import Path
-sys.path.append('..')
+sys.path.append("..")
from abcd import ABCD
from utils.ext_xyz import XYZReader
-if __name__ == '__main__':
+if __name__ == "__main__":
- url = 'mongodb://localhost:27017'
+ url = "mongodb://localhost:27017"
abcd = ABCD(url)
- for file in Path('GB_alphaFe_001/tilt/').glob('*.xyz'):
+ for file in Path("GB_alphaFe_001/tilt/").glob("*.xyz"):
print(file)
- gb_params = {
- 'name': 'alphaFe',
- 'type': 'tilt',
- 'params': file.name[:-4]
-
- }
+ gb_params = {"name": "alphaFe", "type": "tilt", "params": file.name[:-4]}
with abcd as db, XYZReader(file) as reader:
- db.push(reader.read_atoms(), extra_info={'GB_params': gb_params})
+ db.push(reader.read_atoms(), extra_info={"GB_params": gb_params})
diff --git a/tutorials/scripts/Preprocess.py b/tutorials/scripts/Preprocess.py
index 6b29c474..49a315d8 100644
--- a/tutorials/scripts/Preprocess.py
+++ b/tutorials/scripts/Preprocess.py
@@ -4,6 +4,7 @@
from ase.io import read, write
from ase.geometry import crystal_structure_from_cell
import numpy as np
+
# import numpy.linalg as la
import matplotlib.pyplot as plt
@@ -12,7 +13,7 @@
class Calculation(object):
def __init__(self, *args, **kwargs):
- self.filepath = kwargs.pop('filepath', None)
+ self.filepath = kwargs.pop("filepath", None)
self.parameters = kwargs
def get_data(self, index=-1):
@@ -20,20 +21,19 @@ def get_data(self, index=-1):
@classmethod
def from_path(cls, path: Path):
- with (path / 'gb.json').open() as data_file:
+ with (path / "gb.json").open() as data_file:
gb_data = json.load(data_file)
- with (path / 'subgb.json').open() as data_file:
+ with (path / "subgb.json").open() as data_file:
subgb_data = json.load(data_file)
# print(gb_data['angle'])
- filename = subgb_data['name'] + "_traj.xyz"
+ filename = subgb_data["name"] + "_traj.xyz"
filepath = (path / filename).resolve()
# configuration = read(str((path / filename).resolve()), index=index)
# # gb = read(str((path / filename).resolve()), index=-1)
-
# print('{:=^60}'.format(' '+str(path)+' '))
#
# print('{:-^40}'.format(' gb.json '))
@@ -60,27 +60,34 @@ def from_path(cls, path: Path):
# print('Force mean: {:f}, std: {:f}'.format(force_mean, force_std))
# pprint(gb_final.calc.results)
-
return cls(**{**gb_data, **subgb_data}, filepath=filepath)
-if __name__ == '__main__':
+if __name__ == "__main__":
# Read grain boundary database
- dirpath = Path('../GB_alphaFe_001')
+ dirpath = Path("../GB_alphaFe_001")
calculations = {
- 'tilt': [Calculation.from_path(calc_dir) for calc_dir in (dirpath / 'tilt').iterdir() if calc_dir.is_dir()],
- 'twist': [Calculation.from_path(calc_dir) for calc_dir in (dirpath / 'twist').iterdir() if calc_dir.is_dir()]
+ "tilt": [
+ Calculation.from_path(calc_dir)
+ for calc_dir in (dirpath / "tilt").iterdir()
+ if calc_dir.is_dir()
+ ],
+ "twist": [
+ Calculation.from_path(calc_dir)
+ for calc_dir in (dirpath / "twist").iterdir()
+ if calc_dir.is_dir()
+ ],
}
# potential energy of the perfect crystal according to a specific potential
- potential_energy_per_atom = -4.01298214176 # alpha-Fe PotBH
+ potential_energy_per_atom = -4.01298214176 # alpha-Fe PotBH
eV = 1.6021766208e-19
- Angstrom = 1.e-10
+ Angstrom = 1.0e-10
angles, energies = [], []
- for calc in sorted(calculations['tilt'], key=lambda item: item.parameters['angle']):
+ for calc in sorted(calculations["tilt"], key=lambda item: item.parameters["angle"]):
# E_gb = calc.parameters.get('E_gb', None)
#
@@ -92,7 +99,7 @@ def from_path(cls, path: Path):
# energy = 16.02 / (2 * calc.parameters['A'] ) * \
# (E_gb - potential_energy_per_atom * calc.parameters['n_at'])
- if calc.parameters.get('converged', None):
+ if calc.parameters.get("converged", None):
# energy = 16.02 / (2 * calc.parameters['A'] ) * \
# (calc.parameters.get('E_gb') - potential_energy_per_atom * calc.parameters['n_at'])
#
@@ -101,24 +108,26 @@ def from_path(cls, path: Path):
A = cell[0, 0] * cell[1, 1]
energy = (
- eV / Angstrom**2 /
- (2 * A) *
- (atoms.get_potential_energy() - potential_energy_per_atom * len(atoms))
+ eV
+ / Angstrom**2
+ / (2 * A)
+ * (
+ atoms.get_potential_energy()
+ - potential_energy_per_atom * len(atoms)
+ )
)
write(calc.filepath.name, atoms)
-
# print(energy)
# print(calc.parameters['converged'])
# print(data.get_potential_energy()) # data.get_total_energy() == data.get_potential_energy()
# energies.append(calc.parameters['E_gb'] - data.get_total_energy())
energies.append(energy)
- angles.append(calc.parameters['angle'] * 180.0 / np.pi)
+ angles.append(calc.parameters["angle"] * 180.0 / np.pi)
else:
print("not converged: ", calc.filepath)
-
plt.bar(angles, energies)
# x_smooth = np.linspace(min(angles), max(angles), 1000, endpoint=True)
@@ -126,5 +135,3 @@ def from_path(cls, path: Path):
# plt.plot(x_smooth, f(x_smooth), '-')
plt.show()
-
-
diff --git a/tutorials/scripts/Reader.py b/tutorials/scripts/Reader.py
index 3bd0ff3f..83d6be52 100644
--- a/tutorials/scripts/Reader.py
+++ b/tutorials/scripts/Reader.py
@@ -4,6 +4,7 @@
from ase.io import read, write
from ase.geometry import crystal_structure_from_cell
import numpy as np
+
# import numpy.linalg as la
import matplotlib.pyplot as plt
@@ -12,7 +13,7 @@
class Calculation(object):
def __init__(self, *args, **kwargs):
- self.filepath = kwargs.pop('filepath', None)
+ self.filepath = kwargs.pop("filepath", None)
self.parameters = kwargs
def get_data(self, index=-1):
@@ -20,20 +21,19 @@ def get_data(self, index=-1):
@classmethod
def from_path(cls, path: Path, index=-1):
- with (path / 'gb.json').open() as data_file:
+ with (path / "gb.json").open() as data_file:
gb_data = json.load(data_file)
- with (path / 'subgb.json').open() as data_file:
+ with (path / "subgb.json").open() as data_file:
subgb_data = json.load(data_file)
# print(gb_data['angle'])
- filename = subgb_data['name'] + "_traj.xyz"
+ filename = subgb_data["name"] + "_traj.xyz"
filepath = (path / filename).resolve()
# configuration = read(str((path / filename).resolve()), index=index)
# # gb = read(str((path / filename).resolve()), index=-1)
-
# print('{:=^60}'.format(' '+str(path)+' '))
#
# print('{:-^40}'.format(' gb.json '))
@@ -60,41 +60,54 @@ def from_path(cls, path: Path, index=-1):
# print('Force mean: {:f}, std: {:f}'.format(force_mean, force_std))
# pprint(gb_final.calc.results)
-
return cls(**{**gb_data, **subgb_data}, filepath=filepath)
-if __name__ == '__main__':
+if __name__ == "__main__":
# Read grain boundary database
- dirpath = Path('../GB_alphaFe_001')
+ dirpath = Path("../GB_alphaFe_001")
calculations = {
- 'tilt': [Calculation.from_path(calc_dir) for calc_dir in (dirpath / 'tilt').iterdir() if calc_dir.is_dir()],
- 'twist': [Calculation.from_path(calc_dir) for calc_dir in (dirpath / 'twist').iterdir() if calc_dir.is_dir()]
+ "tilt": [
+ Calculation.from_path(calc_dir)
+ for calc_dir in (dirpath / "tilt").iterdir()
+ if calc_dir.is_dir()
+ ],
+ "twist": [
+ Calculation.from_path(calc_dir)
+ for calc_dir in (dirpath / "twist").iterdir()
+ if calc_dir.is_dir()
+ ],
}
# potential energy of the perfect crystal according to a specific potential
- potential_energy_per_atom = -4.01298214176 # alpha-Fe PotBH
+ potential_energy_per_atom = -4.01298214176 # alpha-Fe PotBH
eV = 1.6021766208e-19
- Angstrom = 1.e-10
+ Angstrom = 1.0e-10
angles, energies = [], []
- for calc in sorted(calculations['tilt'], key=lambda item: item.parameters['angle']):
- angles.append(calc.parameters['angle'] * 180.0 / np.pi)
-
+ for calc in sorted(calculations["tilt"], key=lambda item: item.parameters["angle"]):
+ angles.append(calc.parameters["angle"] * 180.0 / np.pi)
- energy = 16.02 / (2 * calc.parameters['A'] ) * \
- (calc.parameters['E_gb'] - potential_energy_per_atom * calc.parameters['n_at'])
+ energy = (
+ 16.02
+ / (2 * calc.parameters["A"])
+ * (
+ calc.parameters["E_gb"]
+ - potential_energy_per_atom * calc.parameters["n_at"]
+ )
+ )
atoms = calc.get_data()
cell = atoms.get_cell()
A = cell[0, 0] * cell[1, 1]
energy = (
- eV / Angstrom**2 /
- (2 * A) *
- (atoms.get_total_energy() - potential_energy_per_atom * len(atoms))
+ eV
+ / Angstrom**2
+ / (2 * A)
+ * (atoms.get_total_energy() - potential_energy_per_atom * len(atoms))
)
print(energy)
@@ -102,7 +115,6 @@ def from_path(cls, path: Path, index=-1):
# energies.append(calc.parameters['E_gb'] - data.get_total_energy())
energies.append(energy)
-
plt.bar(angles, energies)
# x_smooth = np.linspace(min(angles), max(angles), 1000, endpoint=True)
@@ -186,7 +198,7 @@ def from_path(cls, path: Path, index=-1):
# print at.get_potential_energy()
# print E_gb, 'eV/A^2'
# E_gb = 16.02*(at.get_potential_energy()-(at.n*(E_bulk)))/(2.*A)
- # print E_gb, 'J/m^2'
+# print E_gb, 'J/m^2'
# return E_gb
#
# relax.py
@@ -194,4 +206,4 @@ def from_path(cls, path: Path, index=-1):
# E_gb = grain.get_potential_energy()
#
#
-# ener_per_atom = -4.01298214176
\ No newline at end of file
+# ener_per_atom = -4.01298214176
diff --git a/tutorials/scripts/Visualise.py b/tutorials/scripts/Visualise.py
index 1ed65c2d..308e76fb 100644
--- a/tutorials/scripts/Visualise.py
+++ b/tutorials/scripts/Visualise.py
@@ -7,13 +7,12 @@
import matplotlib.pyplot as plt
import numpy as np
-
-@register_backend('ase')
+@register_backend("ase")
class MyASEStructure(Structure):
def __init__(self, atoms, bfactor=[], occupancy=[]):
# super(MyASEStructure, self).__init__()
- self.ext = 'pdb'
+ self.ext = "pdb"
self.params = {}
self._atoms = atoms
self.bfactor = bfactor # [min, max]
@@ -21,7 +20,7 @@ def __init__(self, atoms, bfactor=[], occupancy=[]):
self.id = str(uuid.uuid4())
def get_structure_string(self):
- """ PDB file format:
+ """PDB file format:
CRYST1 16.980 62.517 124.864 90.00 90.00 90.00 P 1
MODEL 1
ATOM 0 Fe MOL 1 15.431 60.277 6.801 1.00 0.00 FE
@@ -36,12 +35,12 @@ def get_structure_string(self):
if self._atoms.get_pbc().any():
cellpar = self._atoms.get_cell_lengths_and_angles()
- str_format = 'CRYST1' + '{:9.3f}' * 3 + '{:7.2f}' * 3 + ' P 1\n'
+ str_format = "CRYST1" + "{:9.3f}" * 3 + "{:7.2f}" * 3 + " P 1\n"
data += str_format.format(*cellpar.tolist())
- data += 'MODEL 1\n'
+ data += "MODEL 1\n"
- str_format = 'ATOM {:5d} {:>4s} MOL 1 {:8.3f}{:8.3f}{:8.3f}{:6.2f}{:6.2f} {:2s}\n'
+ str_format = "ATOM {:5d} {:>4s} MOL 1 {:8.3f}{:8.3f}{:8.3f}{:6.2f}{:6.2f} {:2s}\n"
for index, atom in enumerate(self._atoms):
data += str_format.format(
index,
@@ -51,10 +50,10 @@ def get_structure_string(self):
atom.position[2].tolist(),
self.occupancy[index] if index <= len(self.occupancy) - 1 else 1.0,
self.bfactor[index] if index <= len(self.bfactor) - 1 else 1.0,
- atom.symbol.upper()
+ atom.symbol.upper(),
)
- data += 'ENDMDL\n'
+ data += "ENDMDL\n"
return data
@@ -63,12 +62,11 @@ def ViewStructure(atoms):
import nglview
view = nglview.NGLWidget()
-
+
structure = MyASEStructure(atoms)
view.add_structure(structure)
-
- return view
+ return view
class AtomViewer(object):
@@ -76,38 +74,35 @@ def __init__(self, atoms, data=[], xsize=1000, ysize=500):
self.view = self._init_nglview(atoms, data, xsize, ysize)
self.widgets = {
- 'radius': FloatSlider(
- value=0.8, min=0.0, max=1.5, step=0.01,
- description='Ball size'
+ "radius": FloatSlider(
+ value=0.8, min=0.0, max=1.5, step=0.01, description="Ball size"
),
- 'color_scheme': Dropdown(description='Solor scheme:'),
- 'colorbar': Output()
+ "color_scheme": Dropdown(description="Solor scheme:"),
+ "colorbar": Output(),
}
self.show_colorbar(data)
- self.widgets['radius'].observe(self._update_repr)
+ self.widgets["radius"].observe(self._update_repr)
- self.gui = VBox([
- self.view,
- self.widgets['colorbar'],
- self.widgets['radius']])
+ self.gui = VBox([self.view, self.widgets["colorbar"], self.widgets["radius"]])
def _update_repr(self, chg=None):
self.view.update_spacefill(
- radiusType='radius',
- radius=self.widgets['radius'].value
+ radiusType="radius", radius=self.widgets["radius"].value
)
def show_colorbar(self, data):
- with self.widgets['colorbar']:
+ with self.widgets["colorbar"]:
# Have colormaps separated into categories:
# http://matplotlib.org/examples/color/colormaps_reference.html
- cmap = 'rainbow'
+ cmap = "rainbow"
fig, (ax1, ax2) = plt.subplots(2, 1, figsize=(16, 2))
- img = ax1.imshow([[min(data), max(data)]], aspect='auto', cmap=plt.get_cmap(cmap))
+ img = ax1.imshow(
+ [[min(data), max(data)]], aspect="auto", cmap=plt.get_cmap(cmap)
+ )
ax1.remove()
- cbar = fig.colorbar(img, cax=ax2, orientation='horizontal')
+ cbar = fig.colorbar(img, cax=ax2, orientation="horizontal")
plt.show()
@@ -117,15 +112,12 @@ def _init_nglview(atoms, data, xsize, ysize):
view = nglview.NGLWidget(gui=False)
view._remote_call(
- 'setSize',
- target='Widget',
- args=[
- '{:d}px'.format(xsize),
- '{:d}px'.format(ysize)
- ]
+ "setSize",
+ target="Widget",
+ args=["{:d}px".format(xsize), "{:d}px".format(ysize)],
)
- data = np.max(data)-data
+ data = np.max(data) - data
structure = MyASEStructure(atoms, bfactor=data)
view.add_structure(structure)
@@ -136,19 +128,15 @@ def _init_nglview(atoms, data, xsize, ysize):
view.add_spacefill(
# radiusType='radius',
# radius=1.0,
- color_scheme='bfactor',
- color_scale='rainbow'
- )
- view.update_spacefill(
- radiusType='radius',
- radius=1.0
+ color_scheme="bfactor",
+ color_scale="rainbow",
)
-
+ view.update_spacefill(radiusType="radius", radius=1.0)
# update camera type
view.control.spin([1, 0, 0], np.pi / 2)
view.control.spin([0, 0, 1], np.pi / 2)
- view.camera = 'orthographic'
+ view.camera = "orthographic"
view.center()
return view
diff --git a/tutorials/scripts/Visualise_quip.py b/tutorials/scripts/Visualise_quip.py
index 1ed65c2d..308e76fb 100644
--- a/tutorials/scripts/Visualise_quip.py
+++ b/tutorials/scripts/Visualise_quip.py
@@ -7,13 +7,12 @@
import matplotlib.pyplot as plt
import numpy as np
-
-@register_backend('ase')
+@register_backend("ase")
class MyASEStructure(Structure):
def __init__(self, atoms, bfactor=[], occupancy=[]):
# super(MyASEStructure, self).__init__()
- self.ext = 'pdb'
+ self.ext = "pdb"
self.params = {}
self._atoms = atoms
self.bfactor = bfactor # [min, max]
@@ -21,7 +20,7 @@ def __init__(self, atoms, bfactor=[], occupancy=[]):
self.id = str(uuid.uuid4())
def get_structure_string(self):
- """ PDB file format:
+ """PDB file format:
CRYST1 16.980 62.517 124.864 90.00 90.00 90.00 P 1
MODEL 1
ATOM 0 Fe MOL 1 15.431 60.277 6.801 1.00 0.00 FE
@@ -36,12 +35,12 @@ def get_structure_string(self):
if self._atoms.get_pbc().any():
cellpar = self._atoms.get_cell_lengths_and_angles()
- str_format = 'CRYST1' + '{:9.3f}' * 3 + '{:7.2f}' * 3 + ' P 1\n'
+ str_format = "CRYST1" + "{:9.3f}" * 3 + "{:7.2f}" * 3 + " P 1\n"
data += str_format.format(*cellpar.tolist())
- data += 'MODEL 1\n'
+ data += "MODEL 1\n"
- str_format = 'ATOM {:5d} {:>4s} MOL 1 {:8.3f}{:8.3f}{:8.3f}{:6.2f}{:6.2f} {:2s}\n'
+ str_format = "ATOM {:5d} {:>4s} MOL 1 {:8.3f}{:8.3f}{:8.3f}{:6.2f}{:6.2f} {:2s}\n"
for index, atom in enumerate(self._atoms):
data += str_format.format(
index,
@@ -51,10 +50,10 @@ def get_structure_string(self):
atom.position[2].tolist(),
self.occupancy[index] if index <= len(self.occupancy) - 1 else 1.0,
self.bfactor[index] if index <= len(self.bfactor) - 1 else 1.0,
- atom.symbol.upper()
+ atom.symbol.upper(),
)
- data += 'ENDMDL\n'
+ data += "ENDMDL\n"
return data
@@ -63,12 +62,11 @@ def ViewStructure(atoms):
import nglview
view = nglview.NGLWidget()
-
+
structure = MyASEStructure(atoms)
view.add_structure(structure)
-
- return view
+ return view
class AtomViewer(object):
@@ -76,38 +74,35 @@ def __init__(self, atoms, data=[], xsize=1000, ysize=500):
self.view = self._init_nglview(atoms, data, xsize, ysize)
self.widgets = {
- 'radius': FloatSlider(
- value=0.8, min=0.0, max=1.5, step=0.01,
- description='Ball size'
+ "radius": FloatSlider(
+ value=0.8, min=0.0, max=1.5, step=0.01, description="Ball size"
),
- 'color_scheme': Dropdown(description='Solor scheme:'),
- 'colorbar': Output()
+ "color_scheme": Dropdown(description="Solor scheme:"),
+ "colorbar": Output(),
}
self.show_colorbar(data)
- self.widgets['radius'].observe(self._update_repr)
+ self.widgets["radius"].observe(self._update_repr)
- self.gui = VBox([
- self.view,
- self.widgets['colorbar'],
- self.widgets['radius']])
+ self.gui = VBox([self.view, self.widgets["colorbar"], self.widgets["radius"]])
def _update_repr(self, chg=None):
self.view.update_spacefill(
- radiusType='radius',
- radius=self.widgets['radius'].value
+ radiusType="radius", radius=self.widgets["radius"].value
)
def show_colorbar(self, data):
- with self.widgets['colorbar']:
+ with self.widgets["colorbar"]:
# Have colormaps separated into categories:
# http://matplotlib.org/examples/color/colormaps_reference.html
- cmap = 'rainbow'
+ cmap = "rainbow"
fig, (ax1, ax2) = plt.subplots(2, 1, figsize=(16, 2))
- img = ax1.imshow([[min(data), max(data)]], aspect='auto', cmap=plt.get_cmap(cmap))
+ img = ax1.imshow(
+ [[min(data), max(data)]], aspect="auto", cmap=plt.get_cmap(cmap)
+ )
ax1.remove()
- cbar = fig.colorbar(img, cax=ax2, orientation='horizontal')
+ cbar = fig.colorbar(img, cax=ax2, orientation="horizontal")
plt.show()
@@ -117,15 +112,12 @@ def _init_nglview(atoms, data, xsize, ysize):
view = nglview.NGLWidget(gui=False)
view._remote_call(
- 'setSize',
- target='Widget',
- args=[
- '{:d}px'.format(xsize),
- '{:d}px'.format(ysize)
- ]
+ "setSize",
+ target="Widget",
+ args=["{:d}px".format(xsize), "{:d}px".format(ysize)],
)
- data = np.max(data)-data
+ data = np.max(data) - data
structure = MyASEStructure(atoms, bfactor=data)
view.add_structure(structure)
@@ -136,19 +128,15 @@ def _init_nglview(atoms, data, xsize, ysize):
view.add_spacefill(
# radiusType='radius',
# radius=1.0,
- color_scheme='bfactor',
- color_scale='rainbow'
- )
- view.update_spacefill(
- radiusType='radius',
- radius=1.0
+ color_scheme="bfactor",
+ color_scale="rainbow",
)
-
+ view.update_spacefill(radiusType="radius", radius=1.0)
# update camera type
view.control.spin([1, 0, 0], np.pi / 2)
view.control.spin([0, 0, 1], np.pi / 2)
- view.camera = 'orthographic'
+ view.camera = "orthographic"
view.center()
return view
diff --git a/tutorials/test_db.py b/tutorials/test_db.py
index 8f1fb542..8a8f34f8 100644
--- a/tutorials/test_db.py
+++ b/tutorials/test_db.py
@@ -3,7 +3,7 @@
from abcd import ABCD
-if __name__ == '__main__':
+if __name__ == "__main__":
# http requests
# url = 'http://localhost:5000/api'
@@ -12,9 +12,9 @@
# Mongoengine
# https://stackoverflow.com/questions/36200288/mongolab-pymongo-connection-error
- url = 'mongodb://root:example@localhost:27018/?authSource=admin'
+ url = "mongodb://root:example@localhost:27018/?authSource=admin"
- abcd = ABCD(url, db='abcd', collection='default')
+ abcd = ABCD(url, db="abcd", collection="default")
print(abcd)
abcd.print_info()
@@ -24,8 +24,8 @@
abcd.destroy()
- direcotry = Path('../tutorials/data/')
- file = direcotry / 'bcc_bulk_54_expanded_2_high.xyz'
+ direcotry = Path("../tutorials/data/")
+ file = direcotry / "bcc_bulk_54_expanded_2_high.xyz"
# file = direcotry / 'GAP_6.xyz'
traj = read(file.as_posix(), index=slice(None))
diff --git a/tutorials/test_upload.py b/tutorials/test_upload.py
index 0f3a34bd..c5fe8e91 100644
--- a/tutorials/test_upload.py
+++ b/tutorials/test_upload.py
@@ -3,7 +3,7 @@
from abcd import ABCD
-if __name__ == '__main__':
- abcd = ABCD(url='mongodb://localhost:27017')
+if __name__ == "__main__":
+ abcd = ABCD(url="mongodb://localhost:27017")
print(abcd)
abcd.print_info()