-
Notifications
You must be signed in to change notification settings - Fork 192
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Fix autocompletion for LinkManager and AttributeManager
In the managers, exceptions were not caught properly. For instance, `getattr(calc.inputs, 'value')` was not return an AttributeError if `'value'` does not exist, and as a consequence `getattr(calc.inputs, 'value', None)` was raising instead of returning `None`. Similarly, I fixed `calc.inputs['value']` to raise a KeyError for not-existing `value`. This is now addressed by defining a new compound exception, that inherits both from NotExistent (for backwards-compatible reasons) and from the correct base exception of python (AttributeError or KeyError). The AttributeManager was not having Tab completion for a different reason, `__dir__` was returning tuples of dict items, rather than just the keys. Now these are fixed and tests are added (I cannot really test the TAB-completion functionality, but at least I test that the values of `dir()` and the exceptions raised are the correct ones). Fixes #3984
- Loading branch information
1 parent
6220e85
commit ee01669
Showing
5 changed files
with
182 additions
and
21 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,137 @@ | ||
# -*- coding: utf-8 -*- | ||
########################################################################### | ||
# Copyright (c), The AiiDA team. All rights reserved. # | ||
# This file is part of the AiiDA code. # | ||
# # | ||
# The code is hosted on GitHub at https://github.com/aiidateam/aiida-core # | ||
# For further information on the license, see the LICENSE.txt file # | ||
# For further information please visit http://www.aiida.net # | ||
########################################################################### | ||
"""Tests for the various node managers (.inputs, .outputs, .dict, ...).""" | ||
# pylint: disable=unused-argument | ||
|
||
import pytest | ||
|
||
from aiida import orm | ||
from aiida.common.exceptions import NotExistent, NotExistentAttributeError, NotExistentKeyError | ||
from aiida.common import LinkType | ||
|
||
|
||
def test_dot_dict_manager(clear_database_before_test): | ||
"""Verify that the Dict.dict manager behaves as intended.""" | ||
dict_content = {'a': True, 'b': 1, 'c': 'Some string'} | ||
dict_node = orm.Dict(dict=dict_content) | ||
|
||
# Check that dir() return all keys and nothing else, important | ||
# for tab competion | ||
assert len(dir(dict_node.dict)) == len(dict_content) | ||
assert set(dir(dict_node.dict)) == set(dict_content) | ||
# Check that it works also as an iterator | ||
assert len(list(dict_node.dict)) == len(dict_content) | ||
assert set(dict_node.dict) == set(dict_content) | ||
|
||
for key, val in dict_content.items(): | ||
# dict_node.dict.a == True, ... | ||
assert getattr(dict_node.dict, key) == val | ||
# dict_node.dict['a'] == True, ... | ||
assert dict_node.dict[key] == val | ||
|
||
# I check the attribute fetching directly | ||
assert dict_node.dict.b == 1 | ||
|
||
# Must raise a AttributeError, otherwise tab competion will not work | ||
with pytest.raises(AttributeError): | ||
getattr(dict_node.dict, 'NotExistentKey') | ||
|
||
# Must raise a KeyError | ||
with pytest.raises(KeyError): | ||
_ = dict_node.dict['NotExistentKey'] | ||
|
||
|
||
def test_link_manager(clear_database_before_test): | ||
"""Test the LinkManager via .inputs and .outputs from a ProcessNode.""" | ||
# I first create a calculation with two inputs and two outputs | ||
|
||
# Create inputs | ||
inp1 = orm.Data() | ||
inp1.store() | ||
inp2 = orm.Data() | ||
inp2.store() | ||
|
||
# Create calc with inputs | ||
calc = orm.CalculationNode() | ||
calc.add_incoming(inp1, link_type=LinkType.INPUT_CALC, link_label='inp1label') | ||
calc.add_incoming(inp2, link_type=LinkType.INPUT_CALC, link_label='inp2label') | ||
calc.store() | ||
|
||
# Attach outputs | ||
out1 = orm.Data() | ||
out2 = orm.Data() | ||
out1.add_incoming(calc, link_type=LinkType.CREATE, link_label='out1label') | ||
out1.store() | ||
out2.add_incoming(calc, link_type=LinkType.CREATE, link_label='out2label') | ||
out2.store() | ||
|
||
expected_inputs = {'inp1label': inp1.uuid, 'inp2label': inp2.uuid} | ||
expected_outputs = {'out1label': out1.uuid, 'out2label': out2.uuid} | ||
|
||
#### Check the 'inputs' manager ### | ||
# Check that dir() return all keys and nothing else, important | ||
# for tab competion (we skip anything that starts with an underscore) | ||
assert len([key for key in dir(calc.inputs) if not key.startswith('_')]) == len(expected_inputs) | ||
assert set(key for key in dir(calc.inputs) if not key.startswith('_')) == set(expected_inputs) | ||
# Check that it works also as an iterator | ||
assert len(list(calc.inputs)) == len(expected_inputs) | ||
assert set(calc.inputs) == set(expected_inputs) | ||
|
||
for key, val in expected_inputs.items(): | ||
# calc.inputs.a.uuid == ..., ... | ||
assert getattr(calc.inputs, key).uuid == val | ||
# calc.inputs['a'].uuid == ..., ... | ||
assert calc.inputs[key].uuid == val | ||
|
||
# I check the attribute fetching directly | ||
assert calc.inputs.inp1label.uuid == expected_inputs['inp1label'] | ||
|
||
## Check for not-existing links | ||
# - Must raise a AttributeError, otherwise tab competion will not work | ||
# - Actually raises a NotExistentAttributeError | ||
# - NotExistentAttributeError should also be caught by NotExistent, | ||
# for backwards-compatibility for AiiDA 1.0, 1.1, 1.2 | ||
for exception in [AttributeError, NotExistent, NotExistentAttributeError]: | ||
with pytest.raises(exception): | ||
getattr(calc.inputs, 'NotExistentLabel') | ||
|
||
# - Must raise a KeyError to behave like a dictionary | ||
# - Actually raises a NotExistentKeyError | ||
# - NotExistentKeyError should also be caught by NotExistent, | ||
# for backwards-compatibility for AiiDA 1.0, 1.1, 1.2 | ||
for exception in [KeyError, NotExistent, NotExistentKeyError]: | ||
with pytest.raises(exception): | ||
_ = calc.inputs['NotExistentLabel'] | ||
|
||
#### Check the 'outputs' manager ### | ||
# Check that dir() return all keys and nothing else, important | ||
# for tab competion (we skip anything that starts with an underscore) | ||
assert len([key for key in dir(calc.outputs) if not key.startswith('_')]) == len(expected_outputs) | ||
assert set(key for key in dir(calc.outputs) if not key.startswith('_')) == set(expected_outputs) | ||
# Check that it works also as an iterator | ||
assert len(list(calc.outputs)) == len(expected_outputs) | ||
assert set(calc.outputs) == set(expected_outputs) | ||
|
||
for key, val in expected_outputs.items(): | ||
# calc.outputs.a.uuid == ..., ... | ||
assert getattr(calc.outputs, key).uuid == val | ||
# calc.outputs['a'].uuid == ..., ... | ||
assert calc.outputs[key].uuid == val | ||
|
||
# I check the attribute fetching directly | ||
assert calc.outputs.out1label.uuid == expected_outputs['out1label'] | ||
|
||
# Must raise a AttributeError, otherwise tab competion will not work | ||
with pytest.raises(AttributeError): | ||
getattr(calc.outputs, 'NotExistentLabel') | ||
|
||
# Must raise a KeyError | ||
with pytest.raises(KeyError): | ||
_ = calc.outputs['NotExistentLabel'] |