Skip to main content

Symmetry projectors and modulation signal loss estimators for screening molecular crystals for directional dark matter detection.

Project description

symmscreen

Symmetry projectors and modulation signal loss estimators for screening molecular crystals for directional dark matter detection.

Companion package to "The Role of Symmetries in Dark Matter Detector Design" (arXiv:..., Benjamin Lillard, Jack D. Shergold, and Juri Smirnov). The paper develops a symmetry-projection framework that predicts how much of a molecule's directional dark matter scattering signal survives crystallisation, using only the crystal's space group and the molecule's point-group symmetry and geometry. This package implements exactly that framework (paper Secs. 3-6): the crystal and molecule point-group projectors, the combined survival operator, and the three Lambda estimators for the fractional modulation-signal loss.

This package is intentionally lightweight, and does not compute electronic structure, form factors, scattering rates, or true modulation amplitudes. See SCarFFF and vsdm for software implementing these calculations.

Installation

pip install symmscreen

Quickstart

The fastest way to get the modulation loss estimate, $\Lambda_\mathrm{coord}$, for a molecular crystal with known coordinates is:

import symmscreen as ss

ss.lambda_coord("your_crystal.cif")

There's a matching one-line function for the quadrupole class of the crystal, and of the molecule itself, too: ss.crystal_quadrupole_class(cif_path), ss.molecule_quadrupole_class(cif_path).

If you want more than one metric for the same crystal, or want to inspect the underlying projector matrices, build a CombinedSurvival object directly instead. This caches the crystal/molecule projectors internally, so nothing is recomputed between calls. Since a CIF gives you the molecule's actual orientation, $\mathcal{T}$, the natural estimators here are the orientation-specific ones, lambda_L and lambda_coord:

from symmscreen import CombinedSurvival

survival = CombinedSurvival.from_cif("your_crystal.cif")

survival.crystal.quadrupole_class      # k in {0, 1, 2, 3, 5}, the quadrupole class of the crystal, Q_k.
survival.molecule.quadrupole_class     # k in {0, 1, 2, 3, 5}, the quadrupole class of the molecule's own symmetry H.
survival.crystal.matrix(l=2)           # Pi_2^(L), the l=2 projector for the crystal symmetry.
survival.molecule.matrix(l=2)          # Pi_2^(H), the l=2 projector for the internal molecule symmetry.
survival.matrix(l=2)                   # C^(2)(T), the combined survival operator.
survival.lambda_L()                    # Lambda^(L)(T), modulation signal loss due to crystallisation alone, for this crystal's specific orientation, T.
survival.lambda_coord()                # Lambda_coord, coordinate-aware estimator of total modulation signal loss.

Real, refined crystallographic coordinates are rarely exactly symmetric, so MoleculeProjector's symmetry-detection tolerance (passed to pymatgen's PointGroupAnalyzer, defaulting to pymatgen's own 0.3 Angstrom) matters. For example, caffeine's mirror plane needs tolerance >= 0.2 to be detected at all; below that its real geometry looks like $C_1$ (no symmetry) and quadrupole_class reports 5 instead of the expected 3. Pass mol_tolerance to override the default — lower it if you want stricter, closer-to-exact symmetry detection instead:

survival = CombinedSurvival.from_cif("your_crystal.cif", mol_tolerance=0.1)

Or work purely from symmetry labels, with no CIF or coordinates at all. Here the relative orientation, $\mathcal{T}$, is generally unknown, so the natural estimators are the analytic $\mathcal{T}$-averaged quantities, lambda_L_avg and lambda_ideal_avg. These have matching one-line functions that take the same symmetry labels rather than a CIF:

import symmscreen as ss

ss.lambda_L_avg(mol_pg_symbol="D6h", crys_pg_symbol="D2h")
ss.lambda_ideal_avg(mol_pg_symbol="D6h", crys_pg_symbol="D2h")   # ~ sqrt(2)/5, matching the benzene example in the paper.

Or build the combined survival operator and access them from there, as well as the matrices:

from symmscreen import CombinedSurvival

survival = CombinedSurvival.from_symmetry(mol_pg_symbol="D6h", crys_pg_symbol="D2h")
survival.matrix(l=2)          # C^(2), combined survival operator at l=2. Defaults to T=I, with T the relative orientation.
survival.lambda_L_avg()       # <Lambda^(L)>_T, loss due to crystallisation alone, averaged over T.
survival.lambda_ideal_avg()   # <Lambda>_T, ~sqrt(2)/5, matching the benzene example in the paper.

from_symmetry also takes an optional relative rotation T (the paper's $\mathcal{T}$), for when you want a specific embedding rather than the $\mathcal{T}$-averaged estimators above:

from symmscreen import CombinedSurvival
from scipy.spatial.transform import Rotation

T = Rotation.from_euler("z", 90, degrees=True).as_matrix()   # e.g. a 90 degree rotation about z.

survival = CombinedSurvival.from_symmetry(mol_pg_symbol="D6h", crys_pg_symbol="D2h", T=T)
survival.matrix(l=2)  # C^(2)(T), combined survival operator at l=2 for a fixed T.
survival.lambda_L()   # Lambda^(L)(T), orientation-specific crystallistion loss, rather than T-averaged.

$\mathcal{T}$ defaults to the identity if omitted.

Getting a CIF

symmscreen does not contain any example structures, as crystal structure databases are generally not freely redistributable. However, experimentally determined molecular crystal structures are available from:

Citation

If you use this package, please use the following citation:

Bibtex goes here.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

symmscreen-0.1.2.tar.gz (20.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

symmscreen-0.1.2-py3-none-any.whl (18.8 kB view details)

Uploaded Python 3

File details

Details for the file symmscreen-0.1.2.tar.gz.

File metadata

  • Download URL: symmscreen-0.1.2.tar.gz
  • Upload date:
  • Size: 20.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for symmscreen-0.1.2.tar.gz
Algorithm Hash digest
SHA256 9249b664a8e1c95b1f5ddb534e2cb140c6aacdd1c2467c74244ce46c621db6b7
MD5 196862b61ff831a5fe7018a8f47fb8de
BLAKE2b-256 e55c05ece53408f5f8be1c1cf9f285950dd245804abefb6971784df173ffe2a7

See more details on using hashes here.

Provenance

The following attestation bundles were made for symmscreen-0.1.2.tar.gz:

Publisher: publish.yml on jdshergold/symmscreen

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file symmscreen-0.1.2-py3-none-any.whl.

File metadata

  • Download URL: symmscreen-0.1.2-py3-none-any.whl
  • Upload date:
  • Size: 18.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for symmscreen-0.1.2-py3-none-any.whl
Algorithm Hash digest
SHA256 4f9cb951b15b86e6df1b741b2dbfbed5c4d88f9ce350a1cb7156a730a051c7b7
MD5 b4b4760a5363bc8ae4f61eb41a8d6b9c
BLAKE2b-256 286f80308f84e4bd07b48468d323ea21631e89203936c791b832936027b8daa9

See more details on using hashes here.

Provenance

The following attestation bundles were made for symmscreen-0.1.2-py3-none-any.whl:

Publisher: publish.yml on jdshergold/symmscreen

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page