Skip to content

Commit

Permalink
Resolve bug with searching databases and impact categories (#1153)
Browse files Browse the repository at this point in the history
* Delete .github/workflows/install-canary.yaml

* GH action script for commenting on issues when related milestone is closed.

* update bw2io requirement

* Update readme install instructions

* Update meta.yaml

* Update README.md

* Change requirement to brightway2 instead of bw2io

* Update README.md

* Increases the flexibility of the excel importer so it can also handle the parameter scenario files.

* Update README.md

How to install using Mamba

* Set up signals infrascructure

* Implement dialog

* Implement finding of suitable alternatives and proper management of failing that.

* Implement actual relinking

* Minor documentation improvements + store exchanges to remove instead of to keep.

* Minor documentation + code improvements

* Resolve #782 + improve documentation of `Contributions` class

* Minor documentation + code improvements

* Fix type hinting error

* Update README.md

* Update README.md

* Update README.md

* Updates to the functions get_relevant_flows and get_relevant_activities (#1069)

* Updates to the functions from de Koning (get_relevant_flows and get_relevant_activities), avoiding use of pandas apply and using python map functionality for splitting pandas dataframes.

* Update the use of DataFrame.applymap to DataFrame.map in the excel file importer module.

* Multiple sdf update (#1083)

* Updates to the logging system to avoid the print statement. Merging with
commits for corrections to the uncertainty distributions that were
included within the same branch. Improves thread safety in logging
increasing stability in the multi-threaded processes.

* Changes the creation of log files. Amends the uncertainty wizard test to correct for use of logging over the standard console. Adds the .logs to .gitignore.

* 1) Simplifies the use of the python logging facility with QtThreads (as published on the python docs for logging), improving the stability of logging.\n2) Changes the generation of the logging object that is more compatible with pytest, supporting the existing test procedures.

* Provides an update to the Model for the calculation_setup Methods table, corrects for deletion of impact assessment methods. Updates the routine to the lca_setup module for calling the logger.

* Updates to the logger module, what was the ABLogger class is renamed to ABHandler. The module now contains the formats and settings for the Stream handlers. Calling of the methods in the other AB modules requires passing a logging.Logger instance and the name of the calling module. These are then incorporated into the wrapping routines in the class. Additions:\n 1) An error wrapper is provided and also explicitly provides a trace of the error.\n 2) A timestamp routine is provided and used for providing the file names for log files.\n 3) A standard location based on appdirs is provided and used for log file locations.

* Corrections to the setup of the logger in the test_uncertainty_wizard module

* Includes a change to the type used for the scenario columns when using multiple files with the combined (combinatoric) approach for the scenarios. Includes respective changes to the boolean tests applied to such Indexes.

* Minor corrections to local repository branch, to keep changes aligned

* Alterations to keep minor changes aligned with master branch and fork

---------

Co-authored-by: zoo <[email protected]>

* Use node16 actions for main pipeline

* Use node16 actions in release pipeline

* Use latest version of release-changelog-builder

* Use node16 action for install canary pipeline

* Remove the special build for arm architecture

This was added before brightway2 supported multi-arch builds on
conda-forge. It is not needed anymore, because the normal AB is now also
installable on arm arch.

* Install canary updates (#1093)

* Increase timeout to 30 min

Originally the idea was that installation should never take longer than
12 min, otherwise we should get a warning with the failed pipeline. But
unfortuantely conda currently takes longer than 12 minutes to solve
the environment.

* Add mamba install canary

* Download artifacts to start implementing env comparison

* Looks like on linux it can take more than 30 minutes :-(

* 60min not enough on linux, increasing to 120

* Completely remove timeout, default is 6 hours

* Use solver libmamba option

* Add diff step to compare installations

* Use node16 actions

* Re-add the 12 timeout

* Split diff into separate steps

* Yq action only runs on linux

* debug yq formatting step

* Simplify

* more fighting with yq action

* artifacts are apparently directories in this case

* Try again with while loop

* Run on all os, but only 3.9

* ignore diff exit code

* Run canary install for 3.8 and 3.9 again

---------

Co-authored-by: haasad <[email protected]>

* Adding error message if no scenario file is loaded in a scenario LCA (#1085)

* Updated and improved contributing file

* Improve text on dependencies

* Fix canary link

* Add good/bad examples to CONTRIBUTING.md for commit messages and PR titles

* Add more specific requirements on communicating work on issues in CONTRIBUTING.md

* Minor updates to CONTRIBUTING.md and README.md

* Minor improvements to CONTRIBUTING.md

* Fix formatting error in PULL_REQUEST_TEMPLATE.md

* Resolve #1105

* Update copyright and license information to only link to one place.

* Resolve 3 bugs for location linking (#1051): 1) database of own activity was not always added (in case none of the exchanges were in own database) 2) improve handling of multiple locations. 3) replaced exchanges were deleted from the old process instead of the new one.

* Improve choosing and handling of alternative locations for location choosing

* Resolve readonly database bug for activity duplication to new activity.

* Store AB version in logfile

* Remove deprecated channels from conda envs

* Remove experimental conda environment creation and upload

I started this mostly as an experimental alternative install method with
`conda env create`. However since the channel handling isn't such a pain
anymore now that everything is on conda-forge this is obsolete.

* Move conda-envs to .github directory

* Move changelog config to .github

* Move dev recipe to .github directory

* Move stable recipe to recipe/meta.yaml

* Update dev recipe

* Use micromamba instead of miniconda to speed up tests

* Update install instructions to use libmamba solver with conda

* Fix path for dev recipe

* Added tooltip for table content

* Added docstring for data method

* Cleaning-up documentation

* Update README.md

Resolve #1127

* Add - copy on impact category copy (#1136)

* Add - copy on impact category copy #1133

* Reorder impact category tree context menu

* Resolve bug with switching project on delete (#1138)

* Switch project on delete #1075

* Cleaning up ProjectController.change_project

* Resolving comments by marc-vdm

* GH action script for commenting on issues when related milestone is closed.

* GH action script for commenting on issues when related milestone is closed.

* GH action script for commenting on issues when related milestone is closed.

* Strip whitespace from search fields + implement auto-search for search boxes in databases and impact categories + remove some unused code...

* review instructions bot

* review instructions bot

* Assertion for biosphere install in test

* Added test for project deletion

* Disasbled biosphere search test

* Updated search biosphere test

* Changed a logger to reduce console spam during search

---------

Co-authored-by: bsteubing <[email protected]>
Co-authored-by: zoophobus <[email protected]>
Co-authored-by: Jonathan Kidner <[email protected]>
Co-authored-by: zoo <[email protected]>
Co-authored-by: haasad <[email protected]>
Co-authored-by: Adrian Haas <[email protected]>
Co-authored-by: Marin Visscher <[email protected]>
Co-authored-by: Marin Visscher <[email protected]>
  • Loading branch information
9 people authored Dec 20, 2023
1 parent 4baf486 commit d010b1e
Show file tree
Hide file tree
Showing 7 changed files with 64 additions and 36 deletions.
40 changes: 23 additions & 17 deletions activity_browser/layouts/tabs/impact_categories.py
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,13 @@ def __init__(self, parent):
self.table = MethodsTable(self)
self.table.setToolTip(
"Drag (groups of) impact categories to the calculation setup")

# auto-search
self.debounce_search = QtCore.QTimer()
self.debounce_search.setInterval(300)
self.debounce_search.setSingleShot(True)
self.debounce_search.timeout.connect(self.set_search_term)

#
self.search_box = QtWidgets.QLineEdit()
self.search_box.setPlaceholderText("Search impact categories")
Expand Down Expand Up @@ -122,34 +129,33 @@ def __init__(self, parent):
self.table.setVisible(False)
self.setLayout(container)

self.reset_search_button.clicked.connect(self.table.sync)
self.reset_search_button.clicked.connect(self.tree.model.sync)

self.search_button.clicked.connect(lambda: self.table.sync(query=self.search_box.text()))
self.search_button.clicked.connect(lambda: self.tree.model.sync(query=self.search_box.text()))
self.reset_search_button.clicked.connect(self.search_box.clear)
self.search_box.returnPressed.connect(lambda: self.table.sync(query=self.search_box.text()))
self.search_box.returnPressed.connect(lambda: self.tree.model.sync(query=self.search_box.text()))

signals.project_selected.connect(self.search_box.clear)
self.connect_signals()

def connect_signals(self):
self.mode_radio_list.toggled.connect(self.update_view)

@QtCore.Slot(tuple, name="searchCopiedMethod")
def method_copied(self, method: tuple) -> None:
"""If a method is successfully copied, we might want to filter, but for now do nothing."""
# """If a method is successfully copied, sync and filter for new name."""
# query = ", ".join(method)
# self.search_box.setText(query)
self.reset_search_button.clicked.connect(self.table.sync)
self.reset_search_button.clicked.connect(self.tree.model.sync)

self.search_button.clicked.connect(self.set_search_term)
self.search_button.clicked.connect(self.set_search_term)
self.search_box.returnPressed.connect(self.set_search_term)
self.search_box.returnPressed.connect(self.set_search_term)
self.search_box.textChanged.connect(self.debounce_search.start)

self.reset_search_button.clicked.connect(self.search_box.clear)
signals.project_selected.connect(self.search_box.clear)

@QtCore.Slot(bool, name="isListToggled")
def update_view(self, toggled: bool):
self.tree.setVisible(not toggled)
# self.tree_settings_layout_container.setVisible(not toggled)
self.table.setVisible(toggled)

def set_search_term(self):
search_term = self.search_box.text().strip()
self.table.sync(query=search_term)
self.tree.model.sync(query=search_term)


class CharacterizationFactorsTab(ABTab):
def __init__(self, parent=None):
Expand Down
9 changes: 8 additions & 1 deletion activity_browser/layouts/tabs/project_manager.py
Original file line number Diff line number Diff line change
Expand Up @@ -212,6 +212,12 @@ def __init__(self, parent):
self.header_layout.setAlignment(QtCore.Qt.AlignLeft)
self.header_widget.setLayout(self.header_layout)

# auto-search
self.debounce_search = QtCore.QTimer()
self.debounce_search.setInterval(300)
self.debounce_search.setSingleShot(True)
self.debounce_search.timeout.connect(self.set_search_term)

self.setup_search()

# Overall Layout
Expand All @@ -234,6 +240,7 @@ def setup_search(self):
# 1st search box
self.search_box = QtWidgets.QLineEdit()
self.search_box.setPlaceholderText("Search")
self.search_box.textChanged.connect(self.debounce_search.start)
self.search_box.returnPressed.connect(self.set_search_term)

# search
Expand All @@ -256,5 +263,5 @@ def setup_search(self):
self.header_layout.addWidget(self.reset_search_button)

def set_search_term(self):
search_term = self.search_box.text()
search_term = self.search_box.text().strip()
self.table.search(search_term)
2 changes: 1 addition & 1 deletion activity_browser/ui/tables/models/inventory.py
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ def df_from_metadata(self, db_name: str) -> pd.DataFrame:
def sync(self, db_name: str, df: pd.DataFrame = None) -> None:
if df is not None:
# skip the rest of the sync here if a dataframe is directly supplied
log.info("Pandas Dataframe passed to sync.", df.shape)
log.debug("Pandas Dataframe passed to sync.", df.shape)
self._dataframe = df
self.updated.emit()
return
Expand Down
2 changes: 1 addition & 1 deletion activity_browser/ui/tables/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -296,7 +296,7 @@ def update_proxy_model(self) -> None:

def quick_filter(self) -> None:
# remove weird whitespace from input
query = self.input_line.text().translate(str.maketrans('', '', '\n\t\r'))
query = self.input_line.text().translate(str.maketrans('', '', '\n\t\r')).strip()

# convert to filter
col_name = {v: k for k, v in self.model.filterable_columns.items()}[self.selected_column]
Expand Down
4 changes: 2 additions & 2 deletions activity_browser/ui/widgets/dialog.py
Original file line number Diff line number Diff line change
Expand Up @@ -947,7 +947,7 @@ def __init__(self, idx: int,
@property
def get_state(self) -> tuple:
# remove weird whitespace from input
query_line = self.filter_query_line.text().translate(str.maketrans('', '', '\n\t\r'))
query_line = self.filter_query_line.text().translate(str.maketrans('', '', '\n\t\r')).strip()
# if valid, return a tuple with the state, otherwise, return None
if query_line == '':
return None
Expand Down Expand Up @@ -1010,7 +1010,7 @@ def __init__(self, idx: int,
@property
def get_state(self) -> tuple:
# remove weird whitespace from input
query_line = self.filter_query_line.text().translate(str.maketrans('', '', ' \n\t\r'))
query_line = self.filter_query_line.text().translate(str.maketrans('', '', ' \n\t\r')).strip()
# if valid, return a tuple with the state, otherwise, return None
if query_line == '':
return None
Expand Down
5 changes: 5 additions & 0 deletions tests/test_add_default_data.py
Original file line number Diff line number Diff line change
Expand Up @@ -76,10 +76,15 @@ def test_search_biosphere(qtbot, ab_app):
with qtbot.waitSignal(act_bio_widget.search_box.returnPressed, timeout=1000):
qtbot.keyClicks(act_bio_widget.search_box, 'Pentanol')
qtbot.keyPress(act_bio_widget.search_box, QtCore.Qt.Key_Return)

# Reset search box & timer so it doesn't auto-search after this test
act_bio_widget.search_box.clear()
act_bio_widget.debounce_search.stop()
# We found some results!
assert act_bio_widget.table.rowCount() > 0
# And the table is now definitely smaller than it was.
assert act_bio_widget.table.rowCount() < initial_amount



def test_fail_open_biosphere(ab_app):
Expand Down
38 changes: 24 additions & 14 deletions tests/test_projects.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@
import brightway2 as bw
from PySide2 import QtCore, QtWidgets

from activity_browser.ui.widgets.dialog import ProjectDeletionDialog


def test_new_project(qtbot, ab_app, monkeypatch):
qtbot.waitForWindowShown(ab_app.main_window)
Expand Down Expand Up @@ -30,17 +32,25 @@ def test_change_project(qtbot, ab_app):
assert bw.projects.current == 'pytest_project_del'


#def test_delete_project(qtbot, ab_app, monkeypatch):
# qtbot.waitForWindowShown(ab_app.main_window)
# assert bw.projects.current == 'pytest_project_del'
# monkeypatch.setattr(
# QtWidgets.QMessageBox, "question",
# staticmethod(lambda *args: QtWidgets.QMessageBox.Yes)
# )
# project_tab = ab_app.main_window.left_panel.tabs['Project']
# qtbot.mouseClick(
# project_tab.projects_widget.delete_project_button,
# QtCore.Qt.LeftButton
# )
#
# assert bw.projects.current == 'default'
def test_delete_project(qtbot, ab_app, monkeypatch):
qtbot.waitForWindowShown(ab_app.main_window)
assert bw.projects.current == 'pytest_project_del'
monkeypatch.setattr(
ProjectDeletionDialog, "exec_",
staticmethod(lambda *args: ProjectDeletionDialog.Accepted)
)
monkeypatch.setattr(
ProjectDeletionDialog, "deletion_warning_checked",
staticmethod(lambda *args: True)
)
monkeypatch.setattr(
QtWidgets.QMessageBox, "information",
staticmethod(lambda *args: True)
)
project_tab = ab_app.main_window.left_panel.tabs['Project']
qtbot.mouseClick(
project_tab.projects_widget.delete_project_button,
QtCore.Qt.LeftButton
)

assert bw.projects.current == 'default'

0 comments on commit d010b1e

Please sign in to comment.