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

Python interfacing for PDFmorph #148

Draft
wants to merge 55 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
55 commits
Select commit Hold shift + click to select a range
2bdfe5a
Add python interfacing
Sparks29032 Oct 31, 2024
0fd0a71
Update descriptors
Sparks29032 Oct 31, 2024
e357305
[pre-commit.ci] auto fixes from pre-commit hooks
pre-commit-ci[bot] Oct 31, 2024
2622188
Remove unused import
Sparks29032 Oct 31, 2024
0aadcbc
[pre-commit.ci] auto fixes from pre-commit hooks
pre-commit-ci[bot] Oct 31, 2024
511435a
Fix stash
Sparks29032 Oct 31, 2024
02d19e3
Return r, gr from python frunction
Sparks29032 Oct 31, 2024
bb81fe1
Add news
Sparks29032 Oct 31, 2024
8a89bd0
Remove multiple morphs
Sparks29032 Oct 31, 2024
2c04c84
Add silent flag
Sparks29032 Oct 31, 2024
54155cb
modify codecov.yml, remove .coveragerc, add news
bobleesj Nov 8, 2024
979c4d1
Add pyproject.toml with codespell added
bobleesj Nov 8, 2024
fea6a1e
fix typos suggested by codespell
bobleesj Nov 8, 2024
3f1192c
Add conda-forge release checklist to GitHub Issue template
bobleesj Nov 9, 2024
cc4037c
Rename to tests-on-pr from validate
bobleesj Nov 10, 2024
6d4b669
Rename to matrix-coverage instead of coverage
bobleesj Nov 10, 2024
428a3ae
Rename to check-news-item instead of validate
bobleesj Nov 10, 2024
8cad497
Fix tools diffpy.utils paths
Sparks29032 Nov 11, 2024
8228060
news
Sparks29032 Nov 11, 2024
f19064d
Add morph-shift capabilities
Sparks29032 Nov 12, 2024
92eb01b
Isolating shifting morphs
Sparks29032 Nov 12, 2024
5607e44
Add tests
Sparks29032 Nov 12, 2024
48c76fa
news
Sparks29032 Nov 12, 2024
8144b9b
Prevent duplication of morphshift
Sparks29032 Nov 12, 2024
fc11c5f
fix: use new `parsers` api
ycexiao Feb 8, 2025
e143dfe
docs: update news
ycexiao Feb 8, 2025
8f88485
docs: add `fix` news
ycexiao Feb 8, 2025
f3918cc
style: reduce the line width limit to 79
ycexiao Feb 9, 2025
f4cf88c
style: apply black to all files in the project
ycexiao Feb 9, 2025
ae106cf
style: apply manual editing to shorten the line width
ycexiao Feb 9, 2025
95f011e
style: use one line and less space to format long equations
ycexiao Feb 10, 2025
6474519
Rebase merge
Sparks29032 Apr 4, 2025
254610e
[pre-commit.ci] auto fixes from pre-commit hooks
pre-commit-ci[bot] Mar 19, 2025
370df61
Fix flake8
Sparks29032 Mar 19, 2025
76aa6f4
fix: add temperature field to tutorial data
ycexiao Mar 20, 2025
d4c5d2f
udpate news
ycexiao Mar 20, 2025
b9eb80a
docs: move manual to online doc
ycexiao Mar 20, 2025
ac25cce
adding admin username
sbillinge Mar 22, 2025
2787e0c
Add python interfacing
Sparks29032 Oct 31, 2024
06e3141
Update descriptors
Sparks29032 Oct 31, 2024
6a00568
Add back file
Sparks29032 Apr 4, 2025
67eaa3b
Remove file
Sparks29032 Apr 4, 2025
bd16cd7
File add
Sparks29032 Apr 4, 2025
9604404
Remove file
Sparks29032 Apr 4, 2025
0973bb8
Rename file
Sparks29032 Apr 4, 2025
f8d4eee
Rebasing
Sparks29032 Apr 4, 2025
82221f4
Merge branch 'main' into pdfmorphpy
Sparks29032 Apr 4, 2025
c90f139
[pre-commit.ci] auto fixes from pre-commit hooks
pre-commit-ci[bot] Apr 4, 2025
35f81d4
Fix flake8
Sparks29032 Apr 4, 2025
7591e43
Add table input option
Sparks29032 Apr 4, 2025
745d41f
[pre-commit.ci] auto fixes from pre-commit hooks
pre-commit-ci[bot] Apr 4, 2025
0acbd94
Update logic
Sparks29032 Apr 4, 2025
4b06314
[pre-commit.ci] auto fixes from pre-commit hooks
pre-commit-ci[bot] Apr 4, 2025
90407b4
Bug fix
Sparks29032 Apr 4, 2025
19bb348
[pre-commit.ci] auto fixes from pre-commit hooks
pre-commit-ci[bot] Apr 4, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions doc/source/api/diffpy.morph.rst
Original file line number Diff line number Diff line change
Expand Up @@ -20,10 +20,10 @@ Subpackages
Submodules
----------

diffpy.morph.pdfplot module
diffpy.morph.plot module
^^^^^^^^^^^^^^^^^^^^^^^^^^^

.. automodule:: diffpy.morph.pdfplot
.. automodule:: diffpy.morph.plot
:members:
:undoc-members:
:show-inheritance:
Expand Down
24 changes: 24 additions & 0 deletions news/morphpy.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
**Added:**

* Python interfacing to call PDFmorph
* Returns dictionary of morph metrics (dict) and the r, gr pair for plotting or further manipulation

**Changed:**

* <news item>

**Deprecated:**

* <news item>

**Removed:**

* <news item>

**Fixed:**

* <news item>

**Security:**

* <news item>
63 changes: 38 additions & 25 deletions src/diffpy/morph/morphapp.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@
import diffpy.morph.morph_helpers as helpers
import diffpy.morph.morph_io as io
import diffpy.morph.morphs as morphs
import diffpy.morph.pdfplot as pdfplot
import diffpy.morph.plot as plot
import diffpy.morph.refine as refine
import diffpy.morph.tools as tools
from diffpy.morph import __save_morph_as__
Expand Down Expand Up @@ -403,22 +403,31 @@ def custom_error(self, msg):
return parser


def single_morph(parser, opts, pargs, stdout_flag=True):
def single_morph(parser, opts, pargs, stdout_flag=True, python_wrap=False):
if len(pargs) < 2:
parser.error("You must supply FILE1 and FILE2.")
elif len(pargs) > 2:
elif len(pargs) > 2 and not python_wrap:
parser.error(
"Too many arguments. Make sure you only supply FILE1 and FILE2."
)
elif not (len(pargs) == 2 or len(pargs) == 6) and python_wrap:
parser.error("Python wrapper error.")

# Get the PDFs
x_morph, y_morph = getPDFFromFile(pargs[0])
x_target, y_target = getPDFFromFile(pargs[1])
# If we get from python, we may wrap, which has input size 4
if len(pargs) == 6 and python_wrap:
x_morph = pargs[2]
y_morph = pargs[3]
x_target = pargs[4]
y_target = pargs[5]
else:
x_morph, y_morph = getPDFFromFile(pargs[0])
x_target, y_target = getPDFFromFile(pargs[1])

if y_morph is None:
parser.error(f"No data table found in file: {pargs[0]}.")
parser.error(f"No data table found in: {pargs[0]}.")
if y_target is None:
parser.error(f"No data table found in file: {pargs[1]}.")
parser.error(f"No data table found in: {pargs[1]}.")

# Get configuration values
scale_in = "None"
Expand Down Expand Up @@ -603,7 +612,7 @@ def single_morph(parser, opts, pargs, stdout_flag=True):
maglim = opts.maglim
mag = opts.mag
l_width = opts.lwidth
pdfplot.comparePDFs(
plot.comparePDFs(
pairlist,
labels,
rmin=pmin,
Expand All @@ -614,10 +623,14 @@ def single_morph(parser, opts, pargs, stdout_flag=True):
l_width=l_width,
)

return morph_results
# Return different things depending on whether it is python interfaced
if python_wrap:
return morph_results, chain.x_morph_out, chain.y_morph_out
else:
return morph_results


def multiple_targets(parser, opts, pargs, stdout_flag=True):
def multiple_targets(parser, opts, pargs, stdout_flag=True, python_wrap=False):
# Custom error messages since usage is distinct when --multiple tag is
# applied
if len(pargs) < 2:
Expand Down Expand Up @@ -782,13 +795,9 @@ def multiple_targets(parser, opts, pargs, stdout_flag=True):
else:
try:
if field_list is not None:
pdfplot.plot_param(
field_list, param_list, param_name, field
)
plot.plot_param(field_list, param_list, param_name, field)
else:
pdfplot.plot_param(
target_file_names, param_list, param_name
)
plot.plot_param(target_file_names, param_list, param_name)
# Can occur for non-refined plotting parameters
# i.e. --smear is not selected as an option, but smear is the
# plotting parameter
Expand All @@ -798,10 +807,14 @@ def multiple_targets(parser, opts, pargs, stdout_flag=True):
"morph and target pair. No plot shown."
)

return morph_results
if python_wrap:
# FIXME: Return dictionary of morphs
return morph_results
else:
return morph_results


def multiple_morphs(parser, opts, pargs, stdout_flag=True):
def multiple_morphs(parser, opts, pargs, stdout_flag=True, python_wrap=False):
# Custom error messages since usage is distinct when --multiple tag is
# applied
if len(pargs) < 2:
Expand Down Expand Up @@ -967,13 +980,9 @@ def multiple_morphs(parser, opts, pargs, stdout_flag=True):
else:
try:
if field_list is not None:
pdfplot.plot_param(
field_list, param_list, param_name, field
)
plot.plot_param(field_list, param_list, param_name, field)
else:
pdfplot.plot_param(
morph_file_names, param_list, param_name
)
plot.plot_param(morph_file_names, param_list, param_name)
# Can occur for non-refined plotting parameters
# i.e. --smear is not selected as an option, but smear is the
# plotting parameter
Expand All @@ -983,7 +992,11 @@ def multiple_morphs(parser, opts, pargs, stdout_flag=True):
"morph and target pair. No plot shown."
)

return morph_results
if python_wrap:
# FIXME: Return dictionary of morphs
return morph_results
else:
return morph_results


def getPDFFromFile(fn):
Expand Down
151 changes: 151 additions & 0 deletions src/diffpy/morph/morphpy.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,151 @@
#!/usr/bin/env python

import numpy as np

from diffpy.morph.morphapp import (
create_option_parser,
multiple_morphs,
multiple_targets,
single_morph,
)


def morph(morph_file, target_file, **kwargs):
"""Run diffpy.morph at Python level.

Parameters
----------
morph_file: str
Path-like object to the file to be morphed.
target_file: str
Path-like object to the target file.
kwargs: dict
See the diffpy.morph manual for options.

Returns
-------
dict:
Summary of morphs.
"""

parser = create_option_parser()

inputs = []
for key, value in kwargs.items():
inputs.append(f"--{key}")
inputs.append(f"{value}")
(opts, pargs) = parser.parse_args(inputs)
pargs = [morph_file, target_file]

return single_morph(
parser, opts, pargs, stdout_flag=False, python_wrap=True
)


def morphpy(
morph_table, target_table, morph_header=None, target_header=None, **kwargs
):
"""Run diffpy.morph at Python level.

Parameters
----------
morph_table: numpy.array
Two-column array of (r, gr) for morphed function.
target_table: numpy.array
Two-column array of (r, gr) for target function.
morph_header: dict
Any relevant parameters (e.g. wavelength, composition, temperature)
for the morphed function.
target_header: dict
Any relevant parameters for the target ction.
kwargs: dict
See the diffpy.morph manual for options.

Returns
-------
dict:
Summary of morphs.
"""

parser = create_option_parser()

inputs = []
for key, value in kwargs.items():
inputs.append(f"--{key}")
inputs.append(f"{value}")
(opts, pargs) = parser.parse_args(inputs)

morph_table = np.array(morph_table)
target_table = np.array(target_table)

x_morph = morph_table[:, 0]
y_morph = morph_table[:, 1]
x_target = target_table[:, 0]
y_target = target_table[:, 1]

pargs = ["Morph", "Target", x_morph, y_morph, x_target, y_target]
print(pargs)

return single_morph(
parser, opts, pargs, stdout_flag=False, python_wrap=True
)


def morph_multiple_targets(file, dir, **kwargs):
"""Run diffpy.morph with multiple targets at Python level.

Parameters
----------
file1: str
Path-like object to the file to be morphed.
file2: str
Path-like object to the target file.
kwargs: dict
See the diffpy.morph manual for options.

Returns
-------
dict:
Summary of morphs.
"""

parser = create_option_parser()

inputs = []
for key, value in kwargs.items():
inputs.append(f"--{key}")
inputs.append(f"{value}")
(opts, pargs) = parser.parse_args(inputs)
pargs = [file, dir]

return multiple_targets(parser, opts, pargs)


def morph_multiple_morphs(dir, file, **kwargs):
"""Run diffpy.morph with multiple files morphed at Python level.

Parameters
----------
file1: str
Path-like object to the file to be morphed.
file2: str
Path-like object to the target file.
kwargs: dict
See the diffpy.morph manual for options.

Returns
-------
dict:
Summary of morphs.
"""

parser = create_option_parser()

inputs = []
for key, value in kwargs.items():
inputs.append(f"--{key}")
inputs.append(f"{value}")
(opts, pargs) = parser.parse_args(inputs)
pargs = [dir, file]

return multiple_morphs(parser, opts, pargs)
Loading
Loading