rowan-python 3.0.7__py3-none-any.whl → 3.0.8__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
rowan/__init__.py CHANGED
@@ -29,6 +29,7 @@ from stjames.optimization.freezing_string_method import (
29
29
  api_key: str | None = None
30
30
  project_uuid: str | None = None
31
31
 
32
+ from .api_keys import *
32
33
  from .calculation import *
33
34
  from .folder import *
34
35
  from .molecule import *
rowan/api_keys.py ADDED
@@ -0,0 +1,144 @@
1
+ import uuid
2
+ from datetime import datetime
3
+ from typing import Literal, Self
4
+
5
+ from pydantic import BaseModel
6
+
7
+ from .utils import api_client
8
+
9
+ APIKeyScope = Literal["read", "read_write", "read_write_delete"]
10
+
11
+
12
+ class APIKey(BaseModel):
13
+ """
14
+ Rowan API key.
15
+
16
+ :ivar uuid: UUID of the API key.
17
+ :ivar name: Human-readable name of the API key.
18
+ :ivar created_at: When the key was created.
19
+ :ivar expires_at: When the key expires.
20
+ :ivar is_expired: Whether the key has expired.
21
+ :ivar is_revoked: Whether the key has been revoked.
22
+ :ivar scope: Permission scope ("read", "read_write", or "read_write_delete").
23
+ :ivar can_manage_api_keys: Whether this key can create/list/revoke other API keys.
24
+ :ivar scoped_project_uuid: If set, the key can only access this project.
25
+ :ivar created_by_key_uuid: UUID of the API key used to create this one (if any).
26
+ :ivar revoked_at: When the key was revoked, if applicable.
27
+ :ivar last_used_at: When the key was last used, if known.
28
+ """
29
+
30
+ uuid: str
31
+ name: str
32
+ created_at: datetime
33
+ expires_at: datetime
34
+ is_expired: bool
35
+ is_revoked: bool
36
+ scope: str
37
+ can_manage_api_keys: bool
38
+ scoped_project_uuid: str | None = None
39
+ created_by_key_uuid: str | None = None
40
+ revoked_at: datetime | None = None
41
+ last_used_at: datetime | None = None
42
+
43
+ def __repr__(self) -> str:
44
+ return (
45
+ f"<APIKey name='{self.name}' scope='{self.scope}' "
46
+ f"scoped_project_uuid={self.scoped_project_uuid!r} uuid='{self.uuid}'>"
47
+ )
48
+
49
+ def revoke(self) -> Self:
50
+ """
51
+ Revoke this API key.
52
+
53
+ :returns: Updated APIKey object.
54
+ """
55
+ with api_client() as client:
56
+ response = client.post(f"/api_key/{self.uuid}/revoke")
57
+ response.raise_for_status()
58
+ return type(self)(**response.json())
59
+
60
+
61
+ class CreatedAPIKey(BaseModel):
62
+ """
63
+ Result of creating a new API key.
64
+
65
+ The plaintext ``key`` is only available at creation time — store it now,
66
+ it cannot be retrieved later.
67
+
68
+ :ivar key: Plaintext API key. Save this; it is only returned once.
69
+ :ivar api_key: Metadata for the newly-created key.
70
+ """
71
+
72
+ key: str
73
+ api_key: APIKey
74
+
75
+
76
+ def create_api_key(
77
+ name: str = "api-key",
78
+ scope: APIKeyScope = "read_write",
79
+ valid_days: int = 365,
80
+ scoped_project_uuid: str | None = None,
81
+ ) -> CreatedAPIKey:
82
+ """
83
+ Create a new API key.
84
+
85
+ The caller must currently authenticate with an unscoped key that has
86
+ ``can_manage_api_keys`` permission.
87
+
88
+ :param name: Human-readable name for the key.
89
+ :param scope: Permission scope. One of "read", "read_write", "read_write_delete".
90
+ :param valid_days: Number of days until the key expires.
91
+ :param scoped_project_uuid: If provided, restrict the key to a single project.
92
+ :returns: plaintext key together with its metadata; the plaintext key is
93
+ only returned once — store it immediately.
94
+ """
95
+ plaintext_key = f"rowan-sk{uuid.uuid4()}"
96
+ payload = {
97
+ "api_key": plaintext_key,
98
+ "name": name,
99
+ "scope": scope,
100
+ "valid_days": valid_days,
101
+ "scoped_project_uuid": scoped_project_uuid,
102
+ }
103
+ with api_client() as client:
104
+ response = client.post("/api_key", json=payload)
105
+ if response.status_code == 403:
106
+ raise PermissionError(
107
+ "API key creation rejected by the server (403). The key you are "
108
+ "authenticating with must be unscoped and have `can_manage_api_keys` "
109
+ "permission. Create a manager key from the Rowan web UI "
110
+ "(Account → API keys) and retry with that key."
111
+ )
112
+ response.raise_for_status()
113
+ return CreatedAPIKey(key=plaintext_key, api_key=APIKey(**response.json()))
114
+
115
+
116
+ def list_api_keys(active: bool | None = True) -> list[APIKey]:
117
+ """
118
+ List API keys belonging to the current user.
119
+
120
+ :param active: If True (default), only return non-revoked, non-expired keys.
121
+ If False, only return revoked or expired keys. If None, return all keys.
122
+ :returns: List of APIKey objects.
123
+ """
124
+ params: dict[str, str] = {}
125
+ if active is not None:
126
+ params["active"] = "true" if active else "false"
127
+
128
+ with api_client() as client:
129
+ response = client.get("/api_key", params=params)
130
+ response.raise_for_status()
131
+ return [APIKey(**item) for item in response.json()]
132
+
133
+
134
+ def revoke_api_key(uuid: str) -> APIKey:
135
+ """
136
+ Revoke an API key by UUID.
137
+
138
+ :param uuid: UUID of the key to revoke.
139
+ :returns: Updated APIKey object.
140
+ """
141
+ with api_client() as client:
142
+ response = client.post(f"/api_key/{uuid}/revoke")
143
+ response.raise_for_status()
144
+ return APIKey(**response.json())
@@ -92,7 +92,11 @@ from .protein_binder_design import (
92
92
  submit_protein_binder_design_workflow,
93
93
  )
94
94
  from .protein_cofolding import (
95
+ CofoldingModel,
95
96
  CofoldingResult,
97
+ ConstraintTarget,
98
+ ContactConstraint,
99
+ PocketConstraint,
96
100
  ProteinCofoldingResult,
97
101
  submit_protein_cofolding_workflow,
98
102
  )
@@ -2,7 +2,9 @@
2
2
 
3
3
  from dataclasses import dataclass
4
4
 
5
+ import httpx
5
6
  import stjames
7
+ from pydantic import ValidationError
6
8
  from stjames import ENGINE_METHODS, Engine, Method
7
9
 
8
10
  from ..folder import Folder
@@ -31,6 +33,22 @@ class ElectronicPropertiesResult(WorkflowResult):
31
33
 
32
34
  _stjames_class = stjames.ElectronicPropertiesWorkflow
33
35
 
36
+ def __post_init__(self) -> None:
37
+ # Computed results (dipole, charges, cubes, orbitals, ...) live only in
38
+ # the gzipped S3 blob, not in the workflow's object_data row. Fetch and
39
+ # parse it once on construction so accessors below see populated data.
40
+ super().__post_init__()
41
+ if not self.complete:
42
+ return
43
+ try:
44
+ with api_client() as client:
45
+ response = client.get(f"/orbitals/{self.workflow_uuid}/compressed_json")
46
+ response.raise_for_status()
47
+ populated = stjames.ElectronicPropertiesWorkflow.model_validate(response.json())
48
+ object.__setattr__(self, "_workflow", populated)
49
+ except (httpx.HTTPError, ValidationError):
50
+ pass
51
+
34
52
  def __repr__(self) -> str:
35
53
  return f"<ElectronicPropertiesResult dipole={self.dipole} D>"
36
54
 
@@ -9,8 +9,12 @@ from ..protein import Protein, retrieve_protein
9
9
  from ..utils import api_client
10
10
  from .base import Message, Workflow, WorkflowResult, parse_messages, register_result
11
11
 
12
- # Re-export cofolding model enum from stjames
13
12
  CofoldingModel = stjames.CofoldingModel
13
+ # `ConstraintTarget` aliases stjames's `Token` - one position in an input
14
+ # (a protein/nucleic-acid residue or a ligand atom) addressable by a constraint.
15
+ ConstraintTarget = stjames.workflows.protein_cofolding.Token
16
+ ContactConstraint = stjames.workflows.protein_cofolding.ContactConstraint
17
+ PocketConstraint = stjames.workflows.protein_cofolding.PocketConstraint
14
18
 
15
19
 
16
20
  @dataclass(frozen=True, slots=True)
@@ -244,6 +248,9 @@ def submit_protein_cofolding_workflow(
244
248
  ligand_binding_affinity_index: int | None = None,
245
249
  use_msa_server: bool = True,
246
250
  use_potentials: bool = False,
251
+ contact_constraints: list[ContactConstraint] | None = None,
252
+ pocket_constraints: list[PocketConstraint] | None = None,
253
+ num_samples: int | None = None,
247
254
  compute_strain: bool = False,
248
255
  do_pose_refinement: bool = False,
249
256
  name: str = "Protein-Ligand Co-Folding",
@@ -260,6 +267,10 @@ def submit_protein_cofolding_workflow(
260
267
  Predicts the 3D structure of protein-protein, protein-ligand, protein-DNA,
261
268
  protein-RNA, or other biomolecular complexes.
262
269
 
270
+ See `examples/protein_cofolding_with_constraints.py` for a worked example
271
+ of using `ConstraintTarget`, `ContactConstraint`, and `PocketConstraint`
272
+ (Boltz-2 only).
273
+
263
274
  :param initial_protein_sequences: Protein sequences to be cofolded.
264
275
  :param initial_dna_sequences: DNA sequences to be cofolded.
265
276
  :param initial_rna_sequences: RNA sequences to be cofolded.
@@ -267,7 +278,10 @@ def submit_protein_cofolding_workflow(
267
278
  :param ligand_binding_affinity_index: Index of the ligand for which to compute
268
279
  the binding affinity.
269
280
  :param use_msa_server: Whether to use the MSA server for the computation.
270
- :param use_potentials: Whether to use potentials for the computation.
281
+ :param use_potentials: Whether to use potentials (inference-time steering) with Boltz.
282
+ :param contact_constraints: Boltz contact constraints between two tokens.
283
+ :param pocket_constraints: Boltz pocket constraints between a binder and contact tokens.
284
+ :param num_samples: Number of diffusion samples to generate. If None, uses the model default.
271
285
  :param compute_strain: Whether to compute the strain of the pose
272
286
  (if `pose_refinement` is enabled).
273
287
  :param do_pose_refinement: Whether to optimize non-rotatable bonds in output poses.
@@ -302,6 +316,9 @@ def submit_protein_cofolding_workflow(
302
316
  workflow = stjames.ProteinCofoldingWorkflow(
303
317
  use_msa_server=use_msa_server,
304
318
  use_potentials=use_potentials,
319
+ contact_constraints=contact_constraints or [],
320
+ pocket_constraints=pocket_constraints or [],
321
+ num_samples=num_samples,
305
322
  model=model_str,
306
323
  ligand_binding_affinity_index=ligand_binding_affinity_index,
307
324
  initial_smiles_list=initial_smiles_list,
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: rowan-python
3
- Version: 3.0.7
3
+ Version: 3.0.8
4
4
  Summary: Rowan Python Library
5
5
  Project-URL: Homepage, https://github.com/rowansci/rowan-client
6
6
  Project-URL: Bug Tracker, https://github.com/rowansci/rowan-client/issues
@@ -11,7 +11,7 @@ Requires-Dist: httpx
11
11
  Requires-Dist: nest-asyncio
12
12
  Requires-Dist: rdkit
13
13
  Requires-Dist: setuptools
14
- Requires-Dist: stjames>=0.0.174
14
+ Requires-Dist: stjames<=0.0.182
15
15
  Description-Content-Type: text/markdown
16
16
 
17
17
  # Rowan Python Library
@@ -1,4 +1,5 @@
1
- rowan/__init__.py,sha256=HvGc27Y6l1_E_mcvabDhmDOOtMrJ6ME7Cx8LfVKqyVs,835
1
+ rowan/__init__.py,sha256=wiEroKH5I6VQ-tiseUcrNGJaS_0soFnYMX0EtpwyLIs,859
2
+ rowan/api_keys.py,sha256=TvG5l5MmQ3Qt8Z3Y7jCA_lcrwEHuI7xHy-KLbIdQ8_A,4793
2
3
  rowan/calculation.py,sha256=lZZ52DxPsuJWCTzFZXjhauHK6dV0KCUwzoxtmoxSY48,3442
3
4
  rowan/config.py,sha256=3cVKHUNzkIPnN2bvx7l5sia7Zc5poXS8lKOJlowXyLA,21088
4
5
  rowan/constants.py,sha256=emCH4m9OL2Hm5E-6mJGM_FgzrK_JrZT-FiKJ6pMNQ4Y,84
@@ -12,7 +13,7 @@ rowan/user.py,sha256=Tnwz1-u_92ACt1xATQegtMj3FcosFsAG4m-4YuAkiyg,5955
12
13
  rowan/utils.py,sha256=c1s6Ze-OqLtfvrD23OV60otskejmj-CD88nNf8_nFcw,3636
13
14
  rowan/rowan_rdkit/__init__.py,sha256=EATX2VRzywzKxqkpCUMTf7RNQLkWsfi5VcCNDW6EIiw,503
14
15
  rowan/rowan_rdkit/chem_utils.py,sha256=ZWdLziT59Qr5JzjvV789CAyRq0m5JIawsOP4RxUbQQA,35529
15
- rowan/workflows/__init__.py,sha256=UTkqcsqtANh9TtaYXwOSYmTOvzR9ucQ_xhDupAFHROI,4351
16
+ rowan/workflows/__init__.py,sha256=CM3GKDTafp6vl79dWqSOvkVfwH1Ty8zuc8yzi65ABxM,4438
16
17
  rowan/workflows/admet.py,sha256=0_wIwXXLfHF-3kgGx_1EM1ljjaYHLeEijJ-GbMYxpL8,2904
17
18
  rowan/workflows/analogue_docking.py,sha256=LJpbbaug0tZ9Cg-m0b7EgGH5hJ5894fHOC16Wefx7mE,9206
18
19
  rowan/workflows/base.py,sha256=eLGyzTc9NpXBGC-FRjsaUwKP6RluWrz0ipDZ4jDY_jc,30103
@@ -24,7 +25,7 @@ rowan/workflows/constants.py,sha256=el8jWE9gnGTLNWn5_n_V0H362vIRneOqgy7BOQ8CScg,
24
25
  rowan/workflows/descriptors.py,sha256=rGrNca6kA4SzX5BAOjP6rE91MOLTvCWSYKF_LW2Z0y4,2963
25
26
  rowan/workflows/docking.py,sha256=wmE7QJu1uDHBDynTT1XesXXAZtpB6xLjZUKsHOQyCcU,7386
26
27
  rowan/workflows/double_ended_ts_search.py,sha256=abBblMkshhbzq5UTwIf-ovNFxY8Ltp2O-bGu_plkI58,7806
27
- rowan/workflows/electronic_properties.py,sha256=2L0BUvhx0iT0k4gQf7hbceCpatznT2xui7rs6lpME7k,7564
28
+ rowan/workflows/electronic_properties.py,sha256=ia4mlmgnioEuLDrcgDuxAyyI6VqRkdxBqshjb9uiEI8,8385
28
29
  rowan/workflows/fukui.py,sha256=wLimH3QmorSpvkovRPlI91VuxHG4J91F2EcLYqg3eP0,5112
29
30
  rowan/workflows/hydrogen_bond_donor_acceptor_strength.py,sha256=WWiEK_GlumIEgTTOaqw-Y5gPDkRkaePWcIjTdIocrPc,4916
30
31
  rowan/workflows/interaction_energy_decomposition.py,sha256=Kjwkb-pviFSFIIZNjMb9i8nAkdKrbjFftmG_lkYtE40,6120
@@ -39,7 +40,7 @@ rowan/workflows/pka.py,sha256=TCFSE5HI5JLPslKdbUvJe4IxrVaLExrI_3PnDfXtxTw,8691
39
40
  rowan/workflows/pocket_detection.py,sha256=aGHY0puxekp4c4nsNYHcvKCe1fsetygL04BcSvNFvE8,3864
40
41
  rowan/workflows/pose_analysis_md.py,sha256=UvotLhWv0_VAkKteZboOutDry7l-Zt1K6_SBx3EXqgM,9530
41
42
  rowan/workflows/protein_binder_design.py,sha256=J-9NSbRLdHb6JQRhY_vq43HlHCDCiQqrkOZUCAF-2dk,8604
42
- rowan/workflows/protein_cofolding.py,sha256=1R29XjAVjWHyelGG-mylP2GIamZbLCQKaaaFsCAnYgI,13012
43
+ rowan/workflows/protein_cofolding.py,sha256=D2PVwL51H3traml1JDG-9jU3R0E29AdfR9Bk0tLX1Zw,14093
43
44
  rowan/workflows/protein_md.py,sha256=_n0IdmTQsunbP1geF-wUjXNMKNYV-ngmAPEMJlj0dkI,7021
44
45
  rowan/workflows/rbfe_graph.py,sha256=PLqzBRkxD7tPdBViYJZjgaCP8aA2UXKc9dD4odx5XUo,5788
45
46
  rowan/workflows/redox_potential.py,sha256=lg2Djev58oOmBmI4l3eIaGKafkNXMhwo17K2G7kQvjY,5319
@@ -50,7 +51,7 @@ rowan/workflows/solvent_dependent_conformers.py,sha256=ovvnhCE4xlkpdhccLHEq7oBJR
50
51
  rowan/workflows/spin_states.py,sha256=GZgBJPO6_ds9el4b7wbigIZ5213Z9DwXhokczJ5NDhs,7122
51
52
  rowan/workflows/strain.py,sha256=r0tlZoUlAslAiF7dPTpa9WlQFUAKyVjZ19zASjSS8Hs,6339
52
53
  rowan/workflows/tautomer_search.py,sha256=aqwXoj0ffWsb5gbvzfz_bpx5ifIfR_K07fbdWhU62Ko,5820
53
- rowan_python-3.0.7.dist-info/METADATA,sha256=-sirW_Q74BeqsjNRR4wAWYoV8ebpuNLzc6-8FrgOKRs,1600
54
- rowan_python-3.0.7.dist-info/WHEEL,sha256=QccIxa26bgl1E6uMy58deGWi-0aeIkkangHcxk2kWfw,87
55
- rowan_python-3.0.7.dist-info/licenses/LICENSE,sha256=i05z7xEhyrg6f8j0lR3XYjShnF-MJGFQ-DnpsZ8yiVI,1084
56
- rowan_python-3.0.7.dist-info/RECORD,,
54
+ rowan_python-3.0.8.dist-info/METADATA,sha256=Fm1PlPUx4eLxJqD6QWIvxuhPvwAOnrLWbYw3Lu_IYhU,1600
55
+ rowan_python-3.0.8.dist-info/WHEEL,sha256=QccIxa26bgl1E6uMy58deGWi-0aeIkkangHcxk2kWfw,87
56
+ rowan_python-3.0.8.dist-info/licenses/LICENSE,sha256=i05z7xEhyrg6f8j0lR3XYjShnF-MJGFQ-DnpsZ8yiVI,1084
57
+ rowan_python-3.0.8.dist-info/RECORD,,