Skip to content

Commit

Permalink
Merge branch 'develop' into badpix-monitor-step-args
Browse files Browse the repository at this point in the history
  • Loading branch information
mfixstsci authored Oct 19, 2023
2 parents 55b12f3 + b09eff7 commit 481aa25
Show file tree
Hide file tree
Showing 21 changed files with 550 additions and 130 deletions.
6 changes: 3 additions & 3 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -83,16 +83,16 @@ source activate base/root

**Note:** If you have added a step activating conda to your default terminal/shell (e.g. the `.bashrc`, `.zshrc`, or `.profile` file) then you don't need to do the above step.

Lastly, create the `jwql` environment via one of the `environment.yml` files (currently `environment_python_3_9.yml`, for python 3.9, and `environment_python_3.10.yml`, for python 3.10, are supported by `jwql`):
Lastly, create the `jwql` environment via one of the `environment.yml` files (currently `environment_python_3.9.yml`, for python 3.9, and `environment_python_3.10.yml`, for python 3.10, are supported by `jwql`):

```
conda env create -f environment_python_3_9.yml
conda env create -f environment_python_3.9.yml
```

or

```
conda env create -f environment_python_3_10.yml
conda env create -f environment_python_3.10.yml
```

### Configuration File
Expand Down
44 changes: 22 additions & 22 deletions environment_python_3.10.yml
Original file line number Diff line number Diff line change
Expand Up @@ -22,49 +22,49 @@ channels:
- defaults

dependencies:
- astropy=5.3.2
- astropy=5.3.4
- beautifulsoup4=4.12.2
- bokeh=2.4.3
- celery=5.3.1
- cryptography=41.0.3
- django=4.2.3
- celery=5.3.4
- cryptography=41.0.4
- django=4.2.6
- inflection=0.5.1
- ipython=8.14.0
- ipython=8.16.1
- jinja2=3.1.2
- jsonschema=4.19.0
- matplotlib=3.7.2
- nodejs=20.5.1
- jsonschema=4.19.1
- matplotlib=3.8.0
- nodejs=20.8.0
- numpy=1.25.2
- numpydoc=1.5.0
- pandas=2.0.3
- pandas=2.1.1
- pip=23.2.1
- postgresql=15.4
- psycopg2=2.9.6
- pytest=7.4.0
- psycopg2=2.9.7
- pytest=7.4.2
- pytest-cov=4.1.0
- pytest-mock=3.11.1
- python=3.10.12
- pyyaml=6.0
- redis
- ruff=0.0.285
- pyyaml=6.0.1
- redis=5.0.0
- ruff=0.0.292
- scipy=1.9.3
- setuptools=68.0.0
- sphinx=6.2.1
- sphinx_rtd_theme=1.2.2
- sqlalchemy=2.0.20
- setuptools=68.2.2
- sphinx=7.2.6
- sphinx_rtd_theme=1.3.0
- sqlalchemy=2.0.21
- twine=4.0.2
- wtforms=3.0.1

- pip:
- astroquery==0.4.6
- bandit==1.7.5
- jwst==1.11.4
- jwst==1.12.3
- pysiaf==0.20.0
- pysqlite3==0.5.1
- pysqlite3==0.5.2
- pyvo==1.4.2
- redis==5.0.0
- selenium==4.11.2
- stdatamodels==1.7.2
- selenium==4.13.0
- stdatamodels==1.8.3
- stsci_rtd_theme==1.0.0
- vine==5.0.0
- git+https://github.com/spacetelescope/jwst_reffiles
Expand Down
44 changes: 22 additions & 22 deletions environment_python_3.9.yml
Original file line number Diff line number Diff line change
Expand Up @@ -22,49 +22,49 @@ channels:
- defaults

dependencies:
- astropy=5.3.2
- astropy=5.3.3
- beautifulsoup4=4.12.2
- bokeh=2.4.3
- celery=5.3.1
- cryptography=41.0.3
- django=4.2.3
- celery=5.3.4
- cryptography=41.0.4
- django=4.2.5
- inflection=0.5.1
- ipython=8.14.0
- ipython=8.16.1
- jinja2=3.1.2
- jsonschema=4.19.0
- matplotlib=3.7.2
- nodejs=20.5.1
- jsonschema=4.19.1
- matplotlib=3.8.0
- nodejs=20.8.0
- numpy=1.25.2
- numpydoc=1.5.0
- pandas=2.0.3
- pandas=2.1.1
- pip=23.2.1
- postgresql=15.4
- psycopg2=2.9.6
- pytest=7.4.0
- psycopg2=2.9.7
- pytest=7.4.2
- pytest-cov=4.1.0
- pytest-mock=3.11.1
- python=3.9.17
- pyyaml=6.0
- redis
- ruff=0.0.285
- pyyaml=6.0.1
- redis=5.0.0
- ruff=0.0.292
- scipy=1.9.3
- setuptools=68.0.0
- sphinx=6.2.1
- sphinx_rtd_theme=1.2.2
- sqlalchemy=2.0.20
- setuptools=68.2.2
- sphinx=7.2.6
- sphinx_rtd_theme=1.3.0
- sqlalchemy=2.0.21
- twine=4.0.2
- wtforms=3.0.1

- pip:
- astroquery==0.4.6
- bandit==1.7.5
- jwst==1.11.4
- jwst==1.12.3
- pysiaf==0.20.0
- pysqlite3==0.5.1
- pysqlite3==0.5.2
- pyvo==1.4.2
- redis==5.0.0
- selenium==4.11.2
- stdatamodels==1.7.2
- selenium==4.13.0
- stdatamodels==1.8.3
- stsci_rtd_theme==1.0.0
- vine==5.0.0
- git+https://github.com/spacetelescope/jwst_reffiles
Expand Down
25 changes: 8 additions & 17 deletions jwql/instrument_monitors/common_monitors/dark_monitor.py
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,6 @@
from astropy.modeling import models
from astropy.stats import sigma_clipped_stats
from astropy.time import Time
from bokeh.io import export_png
from bokeh.models import ColorBar, ColumnDataSource, HoverTool, Legend
from bokeh.models import LinearColorMapper
from bokeh.plotting import figure
Expand All @@ -106,7 +105,7 @@
from jwql.utils.constants import JWST_INSTRUMENT_NAMES_MIXEDCASE, JWST_DATAPRODUCTS, RAPID_READPATTERNS
from jwql.utils.logging_functions import log_info, log_fail
from jwql.utils.permissions import set_permissions
from jwql.utils.utils import copy_files, ensure_dir_exists, get_config, filesystem_path
from jwql.utils.utils import copy_files, ensure_dir_exists, get_config, filesystem_path, save_png

THRESHOLDS_FILE = os.path.join(os.path.split(__file__)[0], 'dark_monitor_file_thresholds.txt')

Expand Down Expand Up @@ -276,14 +275,14 @@ def create_mean_slope_figure(self, image, num_files, hotxy=None, deadxy=None, no

# Create figure
start_time = Time(float(self.query_start), format='mjd').tt.datetime.strftime("%m/%d/%Y")
end_time = Time(float(self.query_end), format='mjd').tt.datetime.strftime("%m/%d/%Y")
end_time = Time(float(self.query_end), format='mjd').tt.datetime.strftime("%m/%d/%Y")

self.plot = figure(title=f'{self.aperture}: {num_files} files. {start_time} to {end_time}', tools='')
# tools='pan,box_zoom,reset,wheel_zoom,save')
self.plot.x_range.range_padding = self.plot.y_range.range_padding = 0

# Create the color mapper that will be used to scale the image
mapper = LinearColorMapper(palette='Viridis256', low=(img_med-5*img_dev) ,high=(img_med+5*img_dev))
mapper = LinearColorMapper(palette='Viridis256', low=(img_med - (5 * img_dev)), high=(img_med + (5 * img_dev)))

# Plot image and add color bar
imgplot = self.plot.image(image=[image], x=0, y=0, dw=nx, dh=ny,
Expand All @@ -292,13 +291,6 @@ def create_mean_slope_figure(self, image, num_files, hotxy=None, deadxy=None, no
color_bar = ColorBar(color_mapper=mapper, width=8, title='DN/sec')
self.plot.add_layout(color_bar, 'right')

# Add hover tool for all pixel values
#hover_tool = HoverTool(tooltips=[('(x, y):', '($x{int}, $y{int})'),
# ('value:', '@image')
# ],
# renderers=[imgplot])
#self.plot.tools.append(hover_tool)

if (('FULL' in self.aperture) or ('_CEN' in self.aperture)):

if hotxy is not None:
Expand Down Expand Up @@ -338,22 +330,21 @@ def create_mean_slope_figure(self, image, num_files, hotxy=None, deadxy=None, no
base_start = Time(float(base_parts[3]), format='mjd').tt.datetime
base_end = Time(float(base_parts[5]), format='mjd').tt.datetime
base_start_time = base_start.strftime("%m/%d/%Y")
base_end_time = base_end.strftime("%m/%d/%Y")
base_end_time = base_end.strftime("%m/%d/%Y")
legend_title = f'Compared to dark from {base_start_time} to {base_end_time}'
else:
legend_title = 'Compared to previous mean dark'
legend = Legend(items=[hot_legend, dead_legend, noisy_legend],
location="center",
orientation='vertical',
title = legend_title)
title=legend_title)

self.plot.add_layout(legend, 'below')

# Save the plot in a png
export_png(self.plot, filename=output_filename)
save_png(self.plot, filename=output_filename)
set_permissions(output_filename)


def get_metadata(self, filename):
"""Collect basic metadata from a fits file
Expand Down Expand Up @@ -650,15 +641,15 @@ def overplot_bad_pix(self, pix_type, coords, values):

sources[pix_type] = ColumnDataSource(data=dict(pixels_x=coords[0],
pixels_y=coords[1]
)
)
)

# Overplot the bad pixel locations
badpixplots[pix_type] = self.plot.circle(x=f'pixels_x', y=f'pixels_y',
source=sources[pix_type], color=colors[pix_type])

# Add to the legend
if numpix > 0:
if numpix > 0:
if numpix <= DARK_MONITOR_MAX_BADPOINTS_TO_PLOT:
text = f"{numpix} pix {adjective[pix_type]} than baseline"
else:
Expand Down
17 changes: 16 additions & 1 deletion jwql/tests/test_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,11 @@
from pathlib import Path
import pytest

from jwql.utils.utils import copy_files, get_config, filename_parser, filesystem_path, _validate_config
from bokeh.models import LinearColorMapper
from bokeh.plotting import figure
import numpy as np

from jwql.utils.utils import copy_files, get_config, filename_parser, filesystem_path, save_png, _validate_config


# Determine if tests are being run on Github Actions
Expand Down Expand Up @@ -479,6 +483,17 @@ def test_filesystem_path():
assert check == location


def test_save_png():
"""Test that we can create a png file"""
plot = figure(title='test', tools='')
image = np.zeros((200, 200))
image[100:105, 100:105] = 1
ny, nx = image.shape
mapper = LinearColorMapper(palette='Viridis256', low=0, high=1.1)
imgplot = plot.image(image=[image], x=0, y=0, dw=nx, dh=ny, color_mapper=mapper, level="image")
save_png(plot, filename='test.png')


@pytest.mark.skipif(ON_GITHUB_ACTIONS, reason='Requires access to central storage.')
def test_validate_config():
"""Test that the config validator works."""
Expand Down
41 changes: 31 additions & 10 deletions jwql/utils/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -45,14 +45,15 @@
from bokeh.plotting import figure
import numpy as np
from PIL import Image
from selenium import webdriver

from jwql.utils import permissions
from jwql.utils.constants import FILE_AC_CAR_ID_LEN, FILE_AC_O_ID_LEN, FILE_ACT_LEN, \
FILE_DATETIME_LEN, FILE_EPOCH_LEN, FILE_GUIDESTAR_ATTMPT_LEN_MIN, \
FILE_GUIDESTAR_ATTMPT_LEN_MAX, FILE_OBS_LEN, FILE_PARALLEL_SEQ_ID_LEN, \
FILE_PROG_ID_LEN, FILE_SEG_LEN, FILE_SOURCE_ID_LEN, FILE_SUFFIX_TYPES, \
FILE_TARG_ID_LEN, FILE_VISIT_GRP_LEN, FILE_VISIT_LEN, FILETYPE_WO_STANDARD_SUFFIX, \
JWST_INSTRUMENT_NAMES_SHORTHAND
FILE_DATETIME_LEN, FILE_EPOCH_LEN, FILE_GUIDESTAR_ATTMPT_LEN_MIN, \
FILE_GUIDESTAR_ATTMPT_LEN_MAX, FILE_OBS_LEN, FILE_PARALLEL_SEQ_ID_LEN, \
FILE_PROG_ID_LEN, FILE_SEG_LEN, FILE_SOURCE_ID_LEN, FILE_SUFFIX_TYPES, \
FILE_TARG_ID_LEN, FILE_VISIT_GRP_LEN, FILE_VISIT_LEN, FILETYPE_WO_STANDARD_SUFFIX, \
JWST_INSTRUMENT_NAMES_SHORTHAND

__location__ = os.path.dirname(os.path.dirname(os.path.dirname(__file__)))

Expand Down Expand Up @@ -157,8 +158,7 @@ def create_png_from_fits(filename, outdir):
plot.ygrid.visible = False

# Create the color mapper that will be used to scale the image
#mapper = LogColorMapper(palette='Viridis256', low=(img_med-5*img_dev) ,high=(img_med+5*img_dev))
mapper = LogColorMapper(palette='Greys256', low=(img_med-5*img_dev) ,high=(img_med+5*img_dev))
mapper = LogColorMapper(palette='Greys256', low=(img_med - (5 * img_dev)), high=(img_med + (5 * img_dev)))

# Plot image
imgplot = plot.image(image=[image], x=0, y=0, dw=nx, dh=ny,
Expand All @@ -169,8 +169,8 @@ def create_png_from_fits(filename, outdir):
plot.yaxis.visible = False

# Save the plot in a png
output_filename = os.path.join(outdir, os.path.basename(filename).replace('fits','png'))
export_png(plot, filename=output_filename)
output_filename = os.path.join(outdir, os.path.basename(filename).replace('fits', 'png'))
save_png(plot, filename=output_filename)
permissions.set_permissions(output_filename)
return output_filename
else:
Expand Down Expand Up @@ -749,13 +749,34 @@ def read_png(filename):

# Copy the RGBA image into view, flipping it so it comes right-side up
# with a lower-left origin
view[:,:,:] = np.flipud(np.asarray(rgba_img))
view[:, :, :] = np.flipud(np.asarray(rgba_img))
else:
view = None
# Return the 2D version
return img


def save_png(fig, filename=''):
"""Starting with selenium version 4.10.0, our testing has shown that on the JWQL
servers, we need to specify an instance of a web driver when exporting a Bokeh
figure as a png. This is a wrapper function that creates the web driver instance
and calls Bokeh's export_png function.
Parameters
----------
fig : bokeh.plotting.figure
Bokeh figure to be saved as a png
filename : str
Filename to use for the png file
"""
options = webdriver.FirefoxOptions()
options.add_argument('-headless')
driver = webdriver.Firefox(options=options)
export_png(fig, filename=filename, webdriver=driver)
driver.quit()


def grouper(iterable, chunksize):
"""
Take a list of items (iterable), and group it into chunks of chunksize, with the
Expand Down
Loading

0 comments on commit 481aa25

Please sign in to comment.