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

Return created parameter from add_parameter #4412

Merged
merged 8 commits into from
May 22, 2024

Conversation

jenshnielsen
Copy link
Collaborator

@jenshnielsen jenshnielsen commented Jul 23, 2022

This means that you will be able to do

        self.resolution = self.add_parameter(
            "resolution",
            get_cmd="VOLT:DC:RES?",
            get_parser=float,
            set_cmd=self._set_resolution,
            label="Resolution",
            unit="V",
        )
        """Parameter to control voltage resolution """

which means that you get the benefit of the self.resolution = Parameter(..., instrument=self) form without having to
remember to pass the instrument to Parameter

In documentation this currently (with the qcodes extension) looks like this:

image

Currently it looks like this

image

If we go this way we need to

  • Update docs and examples to reflect this
  • Add tests
  • News fragment

This has tradeoffs between duplication of the parameter name as docstring vs statically defined the parameter.
We have decided that the benefit of a statically defined parameter outweighs the issues. Furthermore, we provide a tool to generate the additional names and if there is a wish we can extend the tool to also lint these things in existing code similar to how the standard library requires that a typevar and its assigned variable is named the same T= TypeVar("T")

An example of applying this can be found in #6089

@codecov
Copy link

codecov bot commented Jul 23, 2022

Codecov Report

Merging #4412 (220df47) into master (220df47) will not change coverage.
The diff coverage is n/a.

❗ Current head 220df47 differs from pull request most recent head 1881499. Consider uploading reports for the commit 1881499 to get more accurate results

@@           Coverage Diff           @@
##           master    #4412   +/-   ##
=======================================
  Coverage   68.24%   68.24%           
=======================================
  Files         339      339           
  Lines       31781    31781           
=======================================
  Hits        21688    21688           
  Misses      10093    10093           

Copy link
Contributor

@astafan8 astafan8 left a comment

Choose a reason for hiding this comment

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

this looks good to me, brings more life to add_parameter syntax :)

the only question i have is what happens with docstring argument of add_parameter and the docstring that comes right after the attribute declaration - the latter takes over, right? in any case, this probably deserves a separate PR anyway to clarify what method of providing a docstring for a parameter to use when and how.

@jenshnielsen
Copy link
Collaborator Author

the only question i have is what happens with docstring argument of add_parameter and the docstring that comes right after the attribute declaration - the latter takes over, right? in any case, this probably deserves a separate PR anyway to clarify what method of providing a docstring for a parameter to use when and how.

As I understand it the upshot of this is that

  • The docstring after is used for sphinx (it does not run the code so it cannot know that the docstring arg is assigned to __doc__ Without that sphinx will simply skip the attribute
  • The docstring below the attribute is not assigned to __doc__ so we need to do that manually. Perhaps some introspection hack could make this possible to get from the sphinx docstring but it does not seems worth it

@astafan8
Copy link
Contributor

  • The docstring after is used for sphinx (it does not run the code so it cannot know that the docstring arg is assigned to doc Without that sphinx will simply skip the attribute
  • The docstring below the attribute is not assigned to doc so we need to do that manually. Perhaps some introspection hack could make this possible to get from the sphinx docstring but it does not seems worth it

so we are bound to have to duplicate the parameter docstrings? :( one for shpinx and another one for e.g. help(..) in vs code, jupyter, etc?

@jenshnielsen
Copy link
Collaborator Author

so we are bound to have to duplicate the parameter docstrings? :( one for shpinx and another one for e.g. help(..) in vs code, jupyter, etc?

I am afraid so unless we come up with a hack of some sort

@jenshnielsen
Copy link
Collaborator Author

@astafan8 Does not look like there is a way to get the attribute docstring at runtime https://stackoverflow.com/questions/55672640/how-can-i-get-doc-string-of-the-class-attribute-in-python

@mgunyho
Copy link
Contributor

mgunyho commented Sep 22, 2022

Hi, I came across this PR and would like to comment.

I see the benefit of self.param_name = ... for autocompletion and static analysis, and I think it should be done that way. However, I don't like that with this change, you have to type the parameter name twice. This feels redundant and can cause problems with e.g. typos. Typos can be detected at runtime by checking that the parameter name is correct in setattr, but still, it feels wrong.

I also think that having the dosctring as add_parameter(..., docstring="...") is better. Having the docstring as a separate string looks strange and feels a bit too magical to me. Having to write the docstring twice would be really bad. Do I understand correctly that Sphinx can only parse the separately added string? Would it be possible to do this using a decorator somehow? Or using ast as suggested in #4458.

@jenshnielsen
Copy link
Collaborator Author

I see the benefit of self.param_name = ... for autocompletion and static analysis, and I think it should be done that way. However, I don't like that with this change, you have to type the parameter name twice. This feels redundant and can cause problems with e.g. typos. Typos can be detected at runtime by checking that the parameter name is correct in setattr, but still, it feels wrong.

I agree this is suboptimal and ideally, we really should get away from having to give name which has always been strange and ugly looking. This may be possible using inspection or by implementing parameters using the descriptor protocol. Regardless I do feel like the gain of making parameters static outweighs the issue by a significant margin

I also think that having the dosctring as add_parameter(..., docstring="...") is better. Having the docstring as a separate string looks strange and feels a bit too magical to me. Having to write the docstring twice would be really bad. Do I understand correctly that Sphinx can only parse the separately added string? Would it be possible to do this using a decorator somehow? Or using ast as suggested in #4458.

The thing is that having a string following an attribute is a well established thing with support in python and in Sphinx which experienced python users will know about but the docstring argument is something that qcodes invented with no support elsewhere.

@mgunyho
Copy link
Contributor

mgunyho commented Sep 22, 2022

I do feel like the gain of making parameters static outweighs the issue by a significant margin

I agree, having to type the name twice is not too bad. But the docstring is too much IMO.

having a string following an attribute is a well established thing with support in python and in Sphinx which experienced python users will know about

Interesting, I haven't seen this before and was not aware of this, maybe I'm not an experienced Python user then :P

But in any case, I think the runtime availability of docstrings is very important. As mentioned in the Stackoverflow link above (and see also discussion on Python mailing list, PEP 257), attribute docstrings are ignored by the the Python interpreter and will never be available at runtime.

Maybe one option to consider would then be to initialize parameters using a decorator, something like

class MyInstrument(...):

    @parameter(
    	unit="V",
    	label="Resolution",
    	# not sure if set_cmd/get_cmd should go here, or should we have @frequency.setter?
    )
    def resolution(self):
    	"""
    	Parameter to control voltage resolution
    	"""

Which I guess is pretty similar to the descriptor idea. This would be quite a major change, but it would solve all of the problems; autocomplete and documentation work while not having to write anything twice.

@mgunyho
Copy link
Contributor

mgunyho commented Sep 22, 2022

Hmm, maybe this is not perfect either, you can't access self in the decorator.

@jenshnielsen
Copy link
Collaborator Author

@mgunyho Thanks for your input. Good to know that you use the runtime docstring. I had not seem many users make use of it. I will investigate if there are ways to write a sphinx extension that parses that our when generating the docs.

I will also see if there is a way for the parameter to get its way with introspection

@jenshnielsen
Copy link
Collaborator Author

A few random thoughts that came up on this

  • Perhaps we can generate the external docstring for the attribute automatically in a precommit hook?
  • If we can get sphinx to document the attributes even without docs it should be possible to write a plugin that takes out the docstring

@jenshnielsen
Copy link
Collaborator Author

Spend a bit of time today exploring what is possible and found out that Sphinx will document attributes if they have a type annotation. This will remove the need to include a docstring to have a parameter documented which was one of the concerns raised above.

@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch from f25a452 to fa26504 Compare April 14, 2024 13:07
Copy link

codecov bot commented Apr 14, 2024

Codecov Report

Attention: Patch coverage is 74.13793% with 30 lines in your changes are missing coverage. Please review.

Project coverage is 67.67%. Comparing base (4081238) to head (76b3c27).
Report is 8 commits behind head on main.

Files Patch % Lines
src/qcodes/extensions/_refactor.py 72.72% 30 Missing ⚠️
Additional details and impacted files
@@            Coverage Diff             @@
##             main    #4412      +/-   ##
==========================================
+ Coverage   67.65%   67.67%   +0.02%     
==========================================
  Files         351      352       +1     
  Lines       30591    30704     +113     
==========================================
+ Hits        20697    20780      +83     
- Misses       9894     9924      +30     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

@jenshnielsen
Copy link
Collaborator Author

I also experimented with looking up the name from the previous stack frame using some very hacky code. While this is definitely possible I am not sure if I like this better than having to write the name twice

@jenshnielsen
Copy link
Collaborator Author

jenshnielsen commented Apr 14, 2024

An alternative could be to overwrite __setattr__ to validate that the attribute name and the parameter short name are the same. This would at least catch any issue at runtime but also make it impossible to assign a parameter to an alias attribute name which I don't think we want.

@jenshnielsen
Copy link
Collaborator Author

This may also be interesting to improve the way parameters are documented https://www.sphinx-doc.org/en/master/development/tutorials/autodoc_ext.html#autodoc-ext-tutorial

@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch 2 times, most recently from 34f1a76 to e84598e Compare May 3, 2024 11:13
@jenshnielsen
Copy link
Collaborator Author

jenshnielsen commented May 3, 2024

For reference this is a libcst transform that can perform this rewrite at least in a simple example

#%%
import libcst as cst
from libcst.codemod import CodemodContext, VisitorBasedCodemodCommand


class AddParameterTransformer(VisitorBasedCodemodCommand):
    def leave_SimpleStatementLine(self, original_node: cst.SimpleStatementLine, updated_node: cst.SimpleStatementLine) -> cst.SimpleStatementLine:
        if isinstance(updated_node.body[0], cst.Expr):
            call_node  = updated_node.body[0].value
        else:
            return updated_node
        if isinstance(call_node, cst.Call):
            func_node = call_node.func
        else:
            return updated_node

        if isinstance(func_node, cst.Attribute) and func_node.attr.value == "add_parameter":
            arg = call_node.args[0].value
            if isinstance(arg, cst.SimpleString):
                var_name = arg.value.strip('"')
                stm = cst.parse_statement(f"self.{var_name}: Parameter = x")
                new_node = stm.body[0]
                new_node = new_node.with_changes(value=call_node)
                return updated_node.with_changes(body=[new_node])
        return updated_node

def transform_code(code: str) -> str:
    transformer = AddParameterTransformer(CodemodContext())
    module = cst.parse_module(code)
    new_module = module.visit(transformer)
    return new_module.code


# Test the transformation
code = """
self.add_parameter(
    "cool_time",
    label="Cooling Time",
    unit="s",
    get_cmd="PS:CTIME?",
    get_parser=int,
    set_cmd="CONF:PS:CTIME {}",
    vals=Ints(5, 3600),
)
"""

#%%
a = transform_code(code)
print(a)
# %%
import libcst as cst
from libcst.codemod import CodemodContext, VisitorBasedCodemodCommand


class AddParameterTransformer(VisitorBasedCodemodCommand):
    def leave_SimpleStatementLine(self, original_node: cst.SimpleStatementLine, updated_node: cst.SimpleStatementLine) -> cst.SimpleStatementLine:
        if isinstance(updated_node.body[0], cst.Expr):
            call_node  = updated_node.body[0].value
        else:
            return updated_node
        if isinstance(call_node, cst.Call):
            func_node = call_node.func
        else:
            return updated_node

        if isinstance(func_node, cst.Attribute) and func_node.attr.value == "add_parameter":
            arg = call_node.args[0].value
            if isinstance(arg, cst.SimpleString):
                var_name = arg.value.strip('"')

                new_node = cst.AnnAssign(target=cst.Attribute(
                    value=cst.Name(
                        value='self',
                        lpar=[],
                        rpar=[],
                    ),
                    attr=cst.Name(
                        value=var_name,
                        lpar=[],
                        rpar=[],
                    ),
                    dot=cst.Dot(
                        whitespace_before=cst.SimpleWhitespace(
                            value='',
                        ),
                        whitespace_after=cst.SimpleWhitespace(
                            value='',
                        ),
                    ),
                    lpar=[],
                    rpar=[],
                    ),
                    annotation=cst.Annotation(
                        annotation=cst.Name(
                            value='Parameter',
                            lpar=[],
                            rpar=[],
                        ),
                        whitespace_before_indicator=cst.SimpleWhitespace(
                            value='',
                        ),
                        whitespace_after_indicator=cst.SimpleWhitespace(
                            value=' ',
                        ),
                    ),
                    value=call_node
                    )
                return updated_node.with_changes(body=[new_node])
        return updated_node

def transform_code(code: str) -> str:
    transformer = AddParameterTransformer(CodemodContext())
    module = cst.parse_module(code)
    new_module = module.visit(transformer)
    return new_module.code


# Test the transformation
code = """
self.add_parameter(
    "cool_time",
    label="Cooling Time",
    unit="s",
    get_cmd="PS:CTIME?",
    get_parser=int,
    set_cmd="CONF:PS:CTIME {}",
    vals=Ints(5, 3600),
)
"""

#%%
a = transform_code(code)
print(a)

@jenshnielsen jenshnielsen mentioned this pull request May 5, 2024
@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch 2 times, most recently from 9d66f36 to e51c0b7 Compare May 13, 2024 08:27
@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch from e51c0b7 to 8fa757a Compare May 17, 2024 13:22
@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch from 8fa757a to 204d701 Compare May 17, 2024 13:24
@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch from 25723ff to 715b109 Compare May 18, 2024 07:29
@jenshnielsen jenshnielsen marked this pull request as ready for review May 19, 2024 08:53
@jenshnielsen jenshnielsen requested a review from a team as a code owner May 19, 2024 08:53
@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch 2 times, most recently from 835a6fb to 3611268 Compare May 19, 2024 17:14
@jenshnielsen jenshnielsen mentioned this pull request May 19, 2024
1 task
@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch from 3611268 to ef0a7e2 Compare May 21, 2024 12:11
Copy link
Contributor

@astafan8 astafan8 left a comment

Choose a reason for hiding this comment

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

And kudos for the refactor tool!!!

src/qcodes/extensions/_refactor.py Outdated Show resolved Hide resolved
@jenshnielsen jenshnielsen force-pushed the add_parameter_return branch from ef0a7e2 to 36cc036 Compare May 21, 2024 14:56
@jenshnielsen jenshnielsen enabled auto-merge May 22, 2024 07:27
@jenshnielsen jenshnielsen added this pull request to the merge queue May 22, 2024
Merged via the queue into microsoft:main with commit 8713dbc May 22, 2024
20 checks passed
@jenshnielsen jenshnielsen deleted the add_parameter_return branch May 22, 2024 08:12
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