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

Driver for Cryogenic sms120c #819

Merged

Conversation

peendebak
Copy link
Contributor

Copy link
Contributor

@WilliamHPNielsen WilliamHPNielsen left a comment

Choose a reason for hiding this comment

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

Looks like a great driver for a highly non-standard instrument (gotta love the if m[2] == '------->':).

A few issues on style:

  • Delete outcommented lines of code
  • Use module level logging (log = logging.getLogger(__name__)) and change print to log.error, log.warning, or log.debug where it is appropriate

Optionally, we would be happy if you would use type hints (including return value) for the function.

Is it correct that the parameters can't have units because the instrument swicthes between "Ampere state" and "Tesla state"? If not, the parameters should have a unit.

@layeoh
Copy link

layeoh commented Oct 27, 2017 via email

@peendebak
Copy link
Contributor Author

@WilliamHPNielsen Driver now passes tests. @layeoh Can you test the driver?

@layeoh
Copy link

layeoh commented Oct 29, 2017 via email

# Created on Fri 29 Nov 2017
# @author: lyeoh

# Last modified by lyeoh 27/10/2017
Copy link
Contributor

Choose a reason for hiding this comment

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

It's kind of redundant to maintain such information in the file since we are using git.

if self._get_rampRate() <= self._current_ramp_limit:
if state == 2: # Quench
log.error(
__name__ + ': Magnet quench detected - please check magnet status before ramping.')
Copy link
Contributor

Choose a reason for hiding this comment

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

SInce you are now using module level logging (log = logging.getLogger(__name__)), there is no reason to prepend the __name__.

@@ -71,6 +71,7 @@ def __init__(self, name, address, coil_constant=0.113375, current_rating=105.84,
idn = self.IDN.get()
print(idn)
Copy link
Contributor

Choose a reason for hiding this comment

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

This should be handled by the connect_message method of the instrument. Unless the IDN response is too non-standard.

units = self._get_unit()
if units == 1:
print("Magnet in persistent mode, at a field of %f T" % field)
print("Magnet in persistent mode, at a field of %f T" %
Copy link
Contributor

Choose a reason for hiding this comment

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

This print call (and many more below in the driver) should be converted to logging calls with the appropriate level.

@layeoh
Copy link

layeoh commented Nov 19, 2017 via email

@peendebak
Copy link
Contributor Author

@WilliamHPNielsen Driver should be fine now.

@WilliamHPNielsen
Copy link
Contributor

@layeoh sorry, but I don't really see the requested changes implemented? Perhaps I am missing something, but it looks like you simply left all calls to print as they were?

In short, there should not be any print calls unless in functions that explicitly request it such as print_readable_snapshot or print_error_messages. There should, however, always be logging of such important feedback as "magnet currently holding" etc. It is not up to driver to decide how the user should handle this information, the driver should simply make it available to the user. And printed messages are not available in all circumstances. If somebody is writing automated procedures using an instrument, the output of print is basically lost to them, whereas the logging system messages are easily retrieved. The driver should not make the assumption that users always sit at a terminal and execute driver commands one-by-one.

Finally, I don't understand the purpose of setting the log level to info sometimes. Can you explain the idea to me?

@layeoh
Copy link

layeoh commented Nov 22, 2017 via email

@WilliamHPNielsen
Copy link
Contributor

@layeoh I'm glad we agree about the usefulness of logging over prints! And I can fully get behind always informing about what is happening in the driver, especially if it takes more than a few ms (like moving into persistent mode).

I don't quite understand the way you handle the logging level, though. Basically, I would say that one should never set the logging levels in the driver. If you don't set any levels, all log messages are piped on in the system. Filtering should happen later by setting the level of loghandlers and/or the loggers at runtime.

I plainly don't understand how setting the loglevel to INFO changes anything. Do you actually not get these messages if you don't do that? How do you usually read out logging information during measurement operation?

@layeoh
Copy link

layeoh commented Nov 22, 2017 via email

@WilliamHPNielsen
Copy link
Contributor

WilliamHPNielsen commented Nov 23, 2017

@layeoh That's a pretty impressive first driver!

Regarding the handling of logging messages, I would recommend the following:

  • Delete the setting of logger level in the driver (no surprises here!)
  • Do the following (or something very similar) in your scripts/notebooks:
logger = logging.getLogger('<name-of-driver-module>')  # the name is probably 'qcodes/instrument_drivers/cryogenic/CryogenicSMS120C'
handler = logging.StreamHandler(sys.stdout)
logger.setLevel(logging.INFO)  # set the logger's level
handler.setLevel(logging.INFO)  # here you get a second chance to discriminate
formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')
handler.setFormatter(formatter)
logger.addHandler(handler)

It's probably a bit longer than what you are doing now, but I think that this is more correct. It's also easy to modify to log to a file plus you get to set the format which can be extremely helpful, e.g. by adding information on which function is producing the message (check out https://docs.python.org/3/library/logging.html#logrecord-attributes to see what' spossible).

Does the above help you/make sense?

@peendebak
Copy link
Contributor Author

@WilliamHPNielsen @layeoh The lines of code above will print each info message twice, but I get the idea.
I still think this is too much code for a typical user to place in any script. The defaults should work for 99% of the users. What about adding

self.handler = logging.StreamHandler(sys.stdout)
if verbose_logging:
   self.handler.setLevel(logging.INFO)  # here you get a second chance to discriminate
formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')
self.handler.setFormatter(formatter)
logger.addHandler(handler)

to the driver init?

@WilliamHPNielsen
Copy link
Contributor

WilliamHPNielsen commented Nov 23, 2017

@peendebak It will only print twice if your root logger (or any parent of the logger) has a handler that prints. But if that is the case, then you should be home safe by simply doing this in your script/notebook:

logger = logging.getLogger('<name-of-driver-module>')  # the name is probably 'qcodes/instrument_drivers/cryogenic/CryogenicSMS120C'
logger.setLevel(logging.INFO)

Does that work for you? @peendebak / @layeoh

@layeoh
Copy link

layeoh commented Nov 28, 2017 via email

@WilliamHPNielsen
Copy link
Contributor

Looks good to me, I'll just wait for travis and then merge. Thanks for the discussion about logging!

@WilliamHPNielsen WilliamHPNielsen merged commit ce3ee86 into microsoft:master Nov 30, 2017
giulioungaretti pushed a commit that referenced this pull request Nov 30, 2017
Author: peendebak <P.T.eendebak@tudelft.nl>

    Driver for Cryogenic sms120c (#819)
@layeoh
Copy link

layeoh commented Nov 30, 2017

Great! Thanks very much for your help and feedback @WilliamHPNielsen and @peendebak !

@layeoh layeoh deleted the feat/crygenicSMS120c branch November 30, 2017 15:19
bors bot added a commit that referenced this pull request Dec 29, 2021
3746: Bump types-docutils from 0.17.1 to 0.17.2 r=jenshnielsen a=dependabot[bot]

Bumps [types-docutils](https://github.com/python/typeshed) from 0.17.1 to 0.17.2.
<details>
<summary>Commits</summary>
<ul>
<li>See full diff in <a href="https://github.com/python/typeshed/commits">compare view</a></li>
</ul>
</details>
<br />


[![Dependabot compatibility score](https://dependabot-badges.githubapp.com/badges/compatibility_score?dependency-name=types-docutils&package-manager=pip&previous-version=0.17.1&new-version=0.17.2)](https://docs.github.com/en/github/managing-security-vulnerabilities/about-dependabot-security-updates#about-compatibility-scores)

You can trigger a rebase of this PR by commenting ``@dependabot` rebase`.

[//]: # (dependabot-automerge-start)
[//]: # (dependabot-automerge-end)

---

<details>
<summary>Dependabot commands and options</summary>
<br />

You can trigger Dependabot actions by commenting on this PR:
- ``@dependabot` rebase` will rebase this PR
- ``@dependabot` recreate` will recreate this PR, overwriting any edits that have been made to it
- ``@dependabot` merge` will merge this PR after your CI passes on it
- ``@dependabot` squash and merge` will squash and merge this PR after your CI passes on it
- ``@dependabot` cancel merge` will cancel a previously requested merge and block automerging
- ``@dependabot` reopen` will reopen this PR if it is closed
- ``@dependabot` close` will close this PR and stop Dependabot recreating it. You can achieve the same result by closing it manually
- ``@dependabot` ignore this major version` will close this PR and stop Dependabot creating any more for this major version (unless you reopen the PR or upgrade to it yourself)
- ``@dependabot` ignore this minor version` will close this PR and stop Dependabot creating any more for this minor version (unless you reopen the PR or upgrade to it yourself)
- ``@dependabot` ignore this dependency` will close this PR and stop Dependabot creating any more for this dependency (unless you reopen the PR or upgrade to it yourself)


</details>

3747: Bump attrs from 21.2.0 to 21.3.0 r=jenshnielsen a=dependabot[bot]

Bumps [attrs](https://github.com/python-attrs/attrs) from 21.2.0 to 21.3.0.
<details>
<summary>Release notes</summary>
<p><em>Sourced from <a href="https://github.com/python-attrs/attrs/releases">attrs's releases</a>.</em></p>
<blockquote>
<h2>21.3.0</h2>
<p>This is a big release in the history of <code>attrs</code> and finishes an arc that took way too long and also delayed this very overdue release. But it's done: <code>import attrs</code> that has been talked about for years[^issue], but fell victim to “just this one more thing” has <em>finally</em> landed.</p>
<p>From now on, modern <code>attrs</code> code looks like this:</p>
<pre lang="python"><code>from attrs import define
<p><a href="https://github.com/define"><code>`@​define</code></a>`
class HelloWorld:
modern: bool = True
</code></pre></p>
<p>The <code>define</code>/<code>field</code> APIs have been around for over a year and were very popular, now the rest of the package followed suit. I'm very excited that <code>attrs</code> remains relevant and keeps evolving over now more than half a decade. If you're curious about some of the background, the docs now contain a short <a href="https://www.attrs.org/en/stable/names.html">explanation and history lesson</a>. As long as our users keep pushing us, we will keep pushing forward class generation in Python!</p>
<p>Big thanks to my <a href="https://github.com/sponsors/hynek/dashboard">GitHub Sponsors</a>, <a href="https://tidelift.com/subscription/pkg/pypi-attrs?utm_source=pypi-attrs&amp;utm_medium=referral&amp;utm_campaign=enterprise&amp;utm_term=repo">Tidelift subscribers</a>, and <a href="https://ko-fi.com/the_hynekl">Ko-fi buyers</a> that help me mustering the motivation for such long-running project!</p>
<hr />
<p>Since the release took so long, there's more highlights than we can enumerate here, we'd just like to point out a breaking change in the new APIs: converters now run on setting attributes by default. If this is causing problems to you, you can disable that behavior by setting <code>`@define(on_setattr=[])</code>.</p>`
<p>[^issue]:  I have an issue from 2018 that I wanted to &quot;come back the moment this lands&quot;.</p>
<h1>Full Changelog</h1>
<h2>Backward-incompatible Changes</h2>
<ul>
<li>
<p>When using <code>`@define</code>,` converters are now run by default when setting an attribute on an instance -- additionally to validators. I.e. the new default is <code>on_setattr=[attrs.setters.convert, attrs.setters.validate]</code>.</p>
<p>This is unfortunately a breaking change, but it was an oversight, impossible to raise a <code>DeprecationWarning</code> about, and it's better to fix it now while the APIs are very fresh with few users. <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/835">#835</a>, <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/886">#886</a></p>
</li>
<li>
<p><code>import attrs</code> has finally landed! As of this release, you can finally import <code>attrs</code> using its proper name.</p>
<p>Not all names from the <code>attr</code> namespace have been transferred; most notably <code>attr.s</code> and <code>attr.ib</code> are missing. See <code>attrs.define</code> and <code>attrs.field</code> if you haven't seen our next-generation APIs yet. A more elaborate explanation can be found <a href="https://www.attrs.org/en/latest/names.html">On The Core API Names</a></p>
<p>This feature is at least for one release <strong>provisional</strong>. We don't <em>plan</em> on changing anything, but such a big change is unlikely to go perfectly on the first strike.</p>
<p>The API docs have been mostly updated, but it will be an ongoing effort to change everything to the new APIs. Please note that we have <strong>not</strong> moved -- or even removed -- anything from <code>attr</code>!</p>
<p>Please do report any bugs or documentation inconsistencies! <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/887">#887</a></p>
</li>
</ul>
<h2>Changes</h2>
<ul>
<li><code>attr.asdict(retain_collection_types=False)</code> (default) dumps collection-esque keys as tuples. <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/646">#646</a>, <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/888">#888</a></li>
<li><code>__match_args__</code> are now generated to support Python 3.10's <a href="https://docs.python.org/3.10/whatsnew/3.10.html#pep-634-structural-pattern-matching">Structural Pattern Matching</a>. This can be controlled by the <code>match_args</code> argument to the class decorators on Python 3.10 and later. On older versions, it is never added and the argument is ignored. <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/815">#815</a></li>
<li>If the class-level <em>on_setattr</em> is set to <code>attrs.setters.validate</code> (default in <code>`@define</code>` and <code>`@mutable</code>)` but no field defines a validator, pretend that it's not set. <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/817">#817</a></li>
<li>The generated <code>__repr__</code> is significantly faster on Pythons with f-strings. <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/819">#819</a></li>
<li>Attributes transformed via <code>field_transformer</code> are wrapped with <code>AttrsClass</code> again. <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/824">#824</a></li>
<li>Generated source code is now cached more efficiently for identical classes. <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/828">#828</a></li>
<li>Added <code>attrs.converters.to_bool()</code>. <a href="https://github-redirect.dependabot.com/python-attrs/attrs/issues/830">#830</a></li>
</ul>
<!-- raw HTML omitted -->
</blockquote>
<p>... (truncated)</p>
</details>
<details>
<summary>Changelog</summary>
<p><em>Sourced from <a href="https://github.com/python-attrs/attrs/blob/main/CHANGELOG.rst">attrs's changelog</a>.</em></p>
<blockquote>
<h2>21.3.0 (2021-12-28)</h2>
<p>Backward-incompatible Changes
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^</p>
<ul>
<li>
<p>When using <code>`@define</code>,` converters are now run by default when setting an attribute on an instance -- additionally to validators.
I.e. the new default is <code>on_setattr=[attrs.setters.convert, attrs.setters.validate]</code>.</p>
<p>This is unfortunately a breaking change, but it was an oversight, impossible to raise a <code>DeprecationWarning</code> about, and it's better to fix it now while the APIs are very fresh with few users.
<code>[#835](python-attrs/attrs#835) &lt;https://github.com/python-attrs/attrs/issues/835&gt;</code><em>,
<code>[#886](python-attrs/attrs#886) &lt;https://github.com/python-attrs/attrs/issues/886&gt;</code></em></p>
</li>
<li>
<p><code>import attrs</code> has finally landed!
As of this release, you can finally import <code>attrs</code> using its proper name.</p>
<p>Not all names from the <code>attr</code> namespace have been transferred; most notably <code>attr.s</code> and <code>attr.ib</code> are missing.
See <code>attrs.define</code> and <code>attrs.field</code> if you haven't seen our next-generation APIs yet.
A more elaborate explanation can be found <code>On The Core API Names &lt;https://www.attrs.org/en/latest/names.html&gt;</code>_</p>
<p>This feature is at least for one release <strong>provisional</strong>.
We don't <em>plan</em> on changing anything, but such a big change is unlikely to go perfectly on the first strike.</p>
<p>The API docs have been mostly updated, but it will be an ongoing effort to change everything to the new APIs.
Please note that we have <strong>not</strong> moved -- or even removed -- anything from <code>attr</code>!</p>
<p>Please do report any bugs or documentation inconsistencies!
<code>[#887](python-attrs/attrs#887) &lt;https://github.com/python-attrs/attrs/issues/887&gt;</code>_</p>
</li>
</ul>
<p>Changes
^^^^^^^</p>
<ul>
<li><code>attr.asdict(retain_collection_types=False)</code> (default) dumps collection-esque keys as tuples.
<code>[#646](python-attrs/attrs#646) &lt;https://github.com/python-attrs/attrs/issues/646&gt;</code><em>,
<code>[#888](python-attrs/attrs#888) &lt;https://github.com/python-attrs/attrs/issues/888&gt;</code></em></li>
<li><code>__match_args__</code> are now generated to support Python 3.10's
<code>Structural Pattern Matching &lt;https://docs.python.org/3.10/whatsnew/3.10.html#pep-634-structural-pattern-matching&gt;</code><em>.
This can be controlled by the <code>match_args</code> argument to the class decorators on Python 3.10 and later.
On older versions, it is never added and the argument is ignored.
<code>[#815](python-attrs/attrs#815) &lt;https://github.com/python-attrs/attrs/issues/815&gt;</code></em></li>
<li>If the class-level <em>on_setattr</em> is set to <code>attrs.setters.validate</code> (default in <code>`@define</code>` and <code>`@mutable</code>)` but no field defines a validator, pretend that it's not set.
<code>[#817](python-attrs/attrs#817) &lt;https://github.com/python-attrs/attrs/issues/817&gt;</code>_</li>
<li>The generated <code>__repr__</code> is significantly faster on Pythons with f-strings.
<code>[#819](python-attrs/attrs#819) &lt;https://github.com/python-attrs/attrs/issues/819&gt;</code>_</li>
<li>Attributes transformed via <code>field_transformer</code> are wrapped with <code>AttrsClass</code> again.
<code>[#824](python-attrs/attrs#824) &lt;https://github.com/python-attrs/attrs/issues/824&gt;</code>_</li>
<li>Generated source code is now cached more efficiently for identical classes.
<code>[#828](python-attrs/attrs#828) &lt;https://github.com/python-attrs/attrs/issues/828&gt;</code>_</li>
<li>Added <code>attrs.converters.to_bool()</code>.
<code>[#830](python-attrs/attrs#830) &lt;https://github.com/python-attrs/attrs/issues/830&gt;</code>_</li>
</ul>
<!-- raw HTML omitted -->
</blockquote>
<p>... (truncated)</p>
</details>
<details>
<summary>Commits</summary>
<ul>
<li><a href="https://github.com/python-attrs/attrs/commit/dd26edd68e12879f716c6554f25d957af299b801"><code>dd26edd</code></a> Prepare 21.3.0</li>
<li><a href="https://github.com/python-attrs/attrs/commit/20bf4b6e54a75201b378cff8e6dd9521d2da28f1"><code>20bf4b6</code></a> Go over CONTRIBUTING.md</li>
<li><a href="https://github.com/python-attrs/attrs/commit/d528dd425980eff3f43b0e29b0ce4dc81ecd8d84"><code>d528dd4</code></a> Fix more links</li>
<li><a href="https://github.com/python-attrs/attrs/commit/fcfb5a692cc8c9f8fde8e39bbd2c5733a47fb1e7"><code>fcfb5a6</code></a> Last pass over changelogs</li>
<li><a href="https://github.com/python-attrs/attrs/commit/e09873485e14e9b11d5d590a55280894df367d92"><code>e098734</code></a> Add logo to PyPI description</li>
<li><a href="https://github.com/python-attrs/attrs/commit/26c0cef8e48bd131d062d45bdaa0c949d4a2d035"><code>26c0cef</code></a> Streamline workflow</li>
<li><a href="https://github.com/python-attrs/attrs/commit/3333e749781a107c829717f7bc0382d33b538b6e"><code>3333e74</code></a> Remove dead achor</li>
<li><a href="https://github.com/python-attrs/attrs/commit/2d77d83d4e3ceadbf4414da5963623a20564c415"><code>2d77d83</code></a> Fix dataclass_transform links</li>
<li><a href="https://github.com/python-attrs/attrs/commit/b4dc9b07c70c16848960da077fc7ac18fe5e9bc8"><code>b4dc9b0</code></a> Better 2.7 example</li>
<li><a href="https://github.com/python-attrs/attrs/commit/d4e32209dc5855796e57c2b08bdc1c1702d051ab"><code>d4e3220</code></a> Use attrs namespace throughout examples.rst</li>
<li>Additional commits viewable in <a href="https://github.com/python-attrs/attrs/compare/21.2.0...21.3.0">compare view</a></li>
</ul>
</details>
<br />


[![Dependabot compatibility score](https://dependabot-badges.githubapp.com/badges/compatibility_score?dependency-name=attrs&package-manager=pip&previous-version=21.2.0&new-version=21.3.0)](https://docs.github.com/en/github/managing-security-vulnerabilities/about-dependabot-security-updates#about-compatibility-scores)

You can trigger a rebase of this PR by commenting ``@dependabot` rebase`.

[//]: # (dependabot-automerge-start)
[//]: # (dependabot-automerge-end)

---

<details>
<summary>Dependabot commands and options</summary>
<br />

You can trigger Dependabot actions by commenting on this PR:
- ``@dependabot` rebase` will rebase this PR
- ``@dependabot` recreate` will recreate this PR, overwriting any edits that have been made to it
- ``@dependabot` merge` will merge this PR after your CI passes on it
- ``@dependabot` squash and merge` will squash and merge this PR after your CI passes on it
- ``@dependabot` cancel merge` will cancel a previously requested merge and block automerging
- ``@dependabot` reopen` will reopen this PR if it is closed
- ``@dependabot` close` will close this PR and stop Dependabot recreating it. You can achieve the same result by closing it manually
- ``@dependabot` ignore this major version` will close this PR and stop Dependabot creating any more for this major version (unless you reopen the PR or upgrade to it yourself)
- ``@dependabot` ignore this minor version` will close this PR and stop Dependabot creating any more for this minor version (unless you reopen the PR or upgrade to it yourself)
- ``@dependabot` ignore this dependency` will close this PR and stop Dependabot creating any more for this dependency (unless you reopen the PR or upgrade to it yourself)


</details>

Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
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.

None yet

4 participants