Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

FigureData wrapper for figures #814

Merged
merged 26 commits into from
Jun 24, 2022
Merged

Conversation

gadial
Copy link
Contributor

@gadial gadial commented Jun 2, 2022

Summary

This PR adds a FigureData wrapper around figures added to ExperimentData to enable saving additional metadata (e.g. qubits) with the figure. This data is currently not saved in the result database.

Details and comments

@gadial gadial changed the title FigureData wrapper for figures [WIP] FigureData wrapper for figures Jun 2, 2022
@gadial gadial requested a review from yaelbh June 6, 2022 06:51
Copy link
Collaborator

@yaelbh yaelbh left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Still missing:

  1. Save the figure and not the data class (I could run in my environment after I made a dirty fix).
  2. Sync the name data member with the figure_names array. For example, when the user modifies the figure's name, by modifying one of them. Alternatively block such modifications.
  3. Tests.

@gadial
Copy link
Contributor Author

gadial commented Jun 7, 2022

Still missing:

  1. Save the figure and not the data class (I could run in my environment after I made a dirty fix).
  2. Sync the name data member with the figure_names array. For example, when the user modifies the figure's name, by modifying one of them. Alternatively block such modifications.
  3. Tests.

All fixed. For name I chose to make it read-only for now; if we want change to name to change the containing ExperimentData figure list, we can, but it's better to avoid doing it if there is no need for this feature.

@gadial gadial changed the title [WIP] FigureData wrapper for figures FigureData wrapper for figures Jun 7, 2022
qiskit_experiments/database_service/db_experiment_data.py Outdated Show resolved Hide resolved
self._name = name
self.metadata = metadata if metadata is not None else {}

# name is read only
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are there potential issues with the user modifying figure_names after a figure was created?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was wondering this too. In theory couldn't you rename a figure by using this?

I expect it would have some headaches in how you propagate it through to the DB since you would likely need to track the old figure name to delete when saving under the new figure name.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree; so for now, the user cannot change name.

Copy link
Collaborator

@chriseclectic chriseclectic left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A few suggestions, but I think this is headed in the right direction

qiskit_experiments/database_service/db_experiment_data.py Outdated Show resolved Hide resolved
self._name = name
self.metadata = metadata if metadata is not None else {}

# name is read only
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was wondering this too. In theory couldn't you rename a figure by using this?

I expect it would have some headaches in how you propagate it through to the DB since you would likely need to track the old figure name to delete when saving under the new figure name.

@@ -130,6 +129,36 @@ def __json_encode__(self):
return self.__getstate__()


class FigureData:
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This could probably just be called Figure.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would also add a __repr__ or _repr_html_ method that prints the name and displays the actual figure (not exactly sure how to do this). This should work whether the figure is a MPL figure, or a loaded SVG from database (later can is IPython.SVG I think).

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Figure is already taken by pyplot

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Using Figure will cause confusion with pyplot's figure; also, this requires changing the way figures are referenced all over the code.

def copy(self, new_name: Optional[str] = None):
"""Creates a deep copy of the figure data"""
name = new_name if new_name is not None else self.name
return FigureData(figure=self.figure, name=name, metadata=copy.deepcopy(self.metadata))
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This isn't a deep copy if the figure isn't also deep copied. If you want a deepcopy why not just do copy.deepcopy(self) ?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This method is needed since we don't allow yet changing the name, but when copying might want to change it. I'll deepcopy the figure as well.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just tried it and deep copying the figure causes matplotlib errors.

"""Initialize a figure data from the json representation"""
return cls(**args)


Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it could be nice to add a save and load method to this class for saving the the figure to a file or the DB.

While this wouldn't be supported yet a save method for DB would be useful if you change filename or add metadata and to save to update DB.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't understand what those functions should do now. Their existence might trick the user to think the data is saved to the DB.

self._figures[figure_key] = figure_data

if figure_data is None:
raise DbExperimentEntryNotFound(f"Figure {figure_key} not found.")

if file_name:
with open(file_name, "wb") as output:
num_bytes = output.write(figure_data)
num_bytes = output.write(figure_data.figure)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This writing to binary could be moved to the save method of Figure class

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There should also be some LOG.warning msgs that if you try and save a figure with metadata that the metadata is not currently saved to DB

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added a warning, not sure about the save method; in this context we are saving the figure obtained from the DB into a file on the local computer.

qiskit_experiments/database_service/db_experiment_data.py Outdated Show resolved Hide resolved
qiskit_experiments/database_service/db_experiment_data.py Outdated Show resolved Hide resolved
---
features:
- |
Figures added to `ExperimentData` are now wrapped with a `FigureData` class
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This might be better as an upgrade note. Something like:

   Adds a :class:`.FigureData` class for adding metadata to a analysis result figures. Figures added to 
   `ExperimentData` are now stored using this class. The raw image object (SVG or matplotlib.Figure)
   can be accessed using the :attr:`.FigureData.figure` attribute.

    Note that currently metadata is only stored locally and will be discarded when saved to the cloud
    experiment service database.

Copy link
Collaborator

@chriseclectic chriseclectic left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good. Couple minor changes inc adding the new class to the qiskit_experiments.framework API docs.

qiskit_experiments/database_service/db_experiment_data.py Outdated Show resolved Hide resolved
"""Initialize a figure data from the json representation"""
return cls(**args)

def _repr_png_(self):
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What is the reason for having both png and svg reprs? Does svg repr not work well for MPL figures?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, with svg sometimes there were missing things from the diagram. I played with it and it seems to me png was the best method I could find to consistently give good results.


def _repr_png_(self):
if isinstance(self.figure, MatplotlibFigure):
b = io.BytesIO()
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wonder if this should be cached so the conversion doesn't happen everytime you view it.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure we need this - it is very quick, and we don't display many images at once, certainty not several times for the same image.

qiskit_experiments/database_service/db_experiment_data.py Outdated Show resolved Hide resolved
# currently only the figure and its name are stored in the database
if isinstance(figure, FigureData):
figure = figure.figure
LOG.warning("Figure metadata is currently not saved to the database")
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
LOG.warning("Figure metadata is currently not saved to the database")
if figure.metadata:
LOG.warning(f"Figure {figure.name} metadata cannot be saved to the experiment service.")

This only needs warning if the metadata is not empty, otherwise there will be warnings raised everywhere.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Technically you are correct, but in practice the metadata will never be empty, since it has a qubits field with the physical_qubits from the experiment itself (this was needed for qiskit-monitoring, the initial motivation for this PR). Also, it probably should always be initialized and not None even without it.

So right now we will get warnings everywhere, even with your suggested fix. Not sure what's the best way to go about this - I'm not sure we need this warning.

releasenotes/notes/figure_data-ecf5a82c95844b6a.yaml Outdated Show resolved Hide resolved
releasenotes/notes/figure_data-ecf5a82c95844b6a.yaml Outdated Show resolved Hide resolved
Copy link
Collaborator

@chriseclectic chriseclectic left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like the wrong logger msg was changed from warning to debug

qiskit_experiments/database_service/db_experiment_data.py Outdated Show resolved Hide resolved
qiskit_experiments/database_service/db_experiment_data.py Outdated Show resolved Hide resolved
@chriseclectic chriseclectic merged commit 1b1ea01 into qiskit-community:main Jun 24, 2022
@gadial gadial deleted the figure_data branch June 24, 2022 15:57
gadial added a commit to gadial/qiskit-experiments that referenced this pull request Jun 29, 2022
* Added FigureData wrapper for figures
* FigureData is now kept when adding the figure into a composite analysis result
* Update qiskit_experiments/database_service/db_experiment_data.py

Co-authored-by: Yael Ben-Haim <[email protected]>
Co-authored-by: Christopher J. Wood <[email protected]>
paco-ri pushed a commit to paco-ri/qiskit-experiments that referenced this pull request Jul 11, 2022
* Added FigureData wrapper for figures
* FigureData is now kept when adding the figure into a composite analysis result
* Update qiskit_experiments/database_service/db_experiment_data.py

Co-authored-by: Yael Ben-Haim <[email protected]>
Co-authored-by: Christopher J. Wood <[email protected]>
@coruscating coruscating modified the milestone: Release 0.4 Jul 12, 2022
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants