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

Allow for scaling true energy in IRFs to assess energy-scale systematics #1261

Merged
merged 35 commits into from
Jul 8, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
b85e538
Update event_selection.py
deborahDOR May 27, 2024
14d53d1
Update lstchain_create_irf_files.py
deborahDOR May 27, 2024
ee2ff27
Update irf_dl3_tool_config.json
deborahDOR May 27, 2024
0ac1de3
Update event_selection.py
deborahDOR May 27, 2024
e0b8f9a
Update lstchain_create_irf_files.py
deborahDOR May 27, 2024
00dbe27
Update irf_dl3_tool_config.json
deborahDOR Jun 7, 2024
de4b751
Update lstchain_create_irf_files.py
deborahDOR Jun 7, 2024
e06e0ac
Update lstchain_create_irf_files.py
deborahDOR Jun 7, 2024
eb1c4ab
Update lstchain_create_irf_files.py
deborahDOR Jun 11, 2024
eac6f10
Update lstchain_create_irf_files.py
deborahDOR Jun 12, 2024
319258a
Update lstchain_create_irf_files.py
deborahDOR Jun 12, 2024
f28019d
Update test_tools.py
deborahDOR Jul 2, 2024
d9673e1
Update test_tools.py
deborahDOR Jul 2, 2024
c239f72
Create irf_dl3_tool_config_mod.json
deborahDOR Jul 2, 2024
3d0a153
Update test_tools.py
deborahDOR Jul 3, 2024
13c085a
Merge branch 'main' into energy_systematics_in_irfs
chaimain Jul 3, 2024
b8fb8c5
Update test_tools.py
deborahDOR Jul 3, 2024
7a92409
Delete docs/examples/irf_dl3_tool_config_mod.json
deborahDOR Jul 3, 2024
be83a58
Update lstchain_create_irf_files.py
deborahDOR Jul 3, 2024
f8614a2
Update test_tools.py
deborahDOR Jul 4, 2024
a41fa4f
Update lstchain_create_irf_files.py
deborahDOR Jul 4, 2024
dc361ce
Update lstchain_create_irf_files.py
deborahDOR Jul 4, 2024
9a0605a
Update test_tools.py
deborahDOR Jul 4, 2024
221aaf7
Update test_tools.py
deborahDOR Jul 4, 2024
264f39c
[skip ci] typo
morcuended Jul 4, 2024
ce611f5
Improve the exploration of Simtel config history to handle more LST M…
gabemery Jul 3, 2024
8b7d8f6
Fix test
gabemery Jul 3, 2024
b28556c
Convert to float
gabemery Jul 3, 2024
edfaf39
Remove try block without handled exception.
gabemery Jul 4, 2024
b0820e6
Add warning log message
gabemery Jul 4, 2024
c137ade
Move comment about NSB extraction from simtel file to the function do…
gabemery Jul 5, 2024
037f900
Run black and add the missing pytest marker
chaimain Jul 8, 2024
66a99d7
Update test_tools.py by running black and adding a missing pytest marker
chaimain Jul 8, 2024
980aabe
Update test_tools.py
chaimain Jul 8, 2024
e999002
Run pyflakes to clean up rebase issues
chaimain Jul 8, 2024
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
1 change: 1 addition & 0 deletions docs/examples/irf_dl3_tool_config.json
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
"true_energy_min": 0.005,
"true_energy_max": 500,
"true_energy_n_bins": 25,
"scale_true_energy": 1.0,
"reco_energy_min": 0.005,
"reco_energy_max": 500,
"reco_energy_n_bins": 25,
Expand Down
5 changes: 5 additions & 0 deletions lstchain/io/event_selection.py
Original file line number Diff line number Diff line change
Expand Up @@ -364,6 +364,11 @@ class DataBinning(Component):
default_value=25,
).tag(config=True)

scale_true_energy= Float(
help="Scaling value for True energy",
default_value=1.0,
).tag(config=True)

reco_energy_min = Float(
help="Minimum value for Reco Energy bins in TeV units",
default_value=0.005,
Expand Down
58 changes: 26 additions & 32 deletions lstchain/io/io.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import logging
import os
import re
import warnings
from multiprocessing import Pool
from contextlib import ExitStack
Expand All @@ -22,9 +21,8 @@
from ctapipe.instrument import SubarrayDescription
from ctapipe.io import HDF5TableReader, HDF5TableWriter

from eventio import Histograms, EventIOFile
from eventio.search_utils import yield_toplevel_of_type, yield_all_subobjects
from eventio.simtel.objects import History, HistoryConfig
from eventio import Histograms, SimTelFile
from eventio.search_utils import yield_toplevel_of_type

from pyirf.simulations import SimulatedEventsInfo

Expand Down Expand Up @@ -59,7 +57,6 @@
'global_metadata',
'merge_dl2_runs',
'merging_check',
'parse_cfg_bytestring',
'read_data_dl2_to_QTable',
'read_dl2_params',
'read_mc_dl2_to_QTable',
Expand Down Expand Up @@ -1264,37 +1261,34 @@ def remove_duplicated_events(data):
data.remove_rows(remove_row_list)


def parse_cfg_bytestring(bytestring):
"""
Parse configuration as read by eventio
:param bytes bytestring: A ``Bytes`` object with configuration data for one parameter
:return: Tuple in form ``('parameter_name', 'value')``
def extract_simulation_nsb(filename):
"""
line_decoded = bytestring.decode('utf-8').rstrip()
if 'ECHO' in line_decoded or '#' in line_decoded:
return None
line_list = line_decoded.split('%', 1)[0] # drop comment
res = re.sub(' +', ' ', line_list).strip().split(' ', 1) # remove extra whitespaces and split
return res[0].upper(), res[1]
Get current run NSB from configuration in simtel file.

WARNING : In current MC, correct NSB are logged after 'STORE_PHOTOELECTRONS' entries
In any new production, behaviour needs to be verified.
New version of simtel will allow to use better metadata.

def extract_simulation_nsb(filename):
"""
Get current run NSB from configuration in simtel file
:param str filename: Input file name
:return array of `float` by tel_id: NSB rate
"""
nsb = []
with EventIOFile(filename) as f:
for o in yield_all_subobjects(f, [History, HistoryConfig]):
if hasattr(o, 'parse'):
try:
cfg_element = parse_cfg_bytestring(o.parse()[1])
if cfg_element is not None:
if cfg_element[0] == 'NIGHTSKY_BACKGROUND':
nsb.append(float(cfg_element[1].strip('all:')))
except Exception as e:
print('Unexpected end of %s,\n caught exception %s', filename, e)
:return dict of `float` by tel_id: NSB rate
"""
nsb = {}
next_nsb = False
tel_id = 1
with SimTelFile(filename) as f:
for _, line in f.history:
line = line.decode('utf-8').strip().split(' ')
if next_nsb and line[0] == 'NIGHTSKY_BACKGROUND':
nsb[tel_id] = float(line[1].strip('all:'))
tel_id = tel_id+1
if line[0] == 'STORE_PHOTOELECTRONS':
next_nsb = True
else:
next_nsb = False
log.warning('Original MC night sky background extracted from the config history in the simtel file.\n'
'This is done for existing LST MC such as the one created using: '
'https://github.com/cta-observatory/lst-sim-config/tree/sim-tel_LSTProd2_MAGICST0316'
'\nExtracted values are: ' + str(np.asarray(nsb)) + 'GHz. Check that it corresponds to expectations.')
return nsb


Expand Down
3 changes: 1 addition & 2 deletions lstchain/io/tests/test_io.py
Original file line number Diff line number Diff line change
Expand Up @@ -156,8 +156,7 @@ def test_extract_simulation_nsb(mc_gamma_testfile):
from lstchain.io.io import extract_simulation_nsb

nsb = extract_simulation_nsb(mc_gamma_testfile)
assert np.isclose(nsb[0], 0.246, rtol=0.1)
assert np.isclose(nsb[1], 0.217, rtol=0.1)
assert np.isclose(nsb[1], 0.246, rtol=0.1)


def test_remove_duplicated_events():
Expand Down
4 changes: 1 addition & 3 deletions lstchain/reco/r0_to_dl1.py
Original file line number Diff line number Diff line change
Expand Up @@ -433,10 +433,8 @@ def r0_to_dl1(
charge_spe_cumulative_pdf = interp1d(spe_integral, spe[0], kind='cubic',
bounds_error=False, fill_value=0.,
assume_sorted=True)
allowed_tel = np.zeros(len(nsb_original), dtype=bool)
allowed_tel[np.array(config['source_config']['LSTEventSource']['allowed_tels'])] = True
logger.info('Tuning NSB on MC waveform from '
+ str(np.asarray(nsb_original)[allowed_tel])
+ str(np.asarray(nsb_original))
+ 'GHz to {0:d}%'.format(int(nsb_tuning_ratio * 100 + 100.5))
+ ' for telescopes ids ' + str(config['source_config']['LSTEventSource']['allowed_tels']))
nsb_tuning_args = [nsb_tuning_ratio, nsb_original, pulse_template, charge_spe_cumulative_pdf]
Expand Down
28 changes: 26 additions & 2 deletions lstchain/tools/lstchain_create_irf_files.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,15 @@
If you want to generate source-dependent IRFs, source-dep flag should be activated.
The global alpha cut used to generate IRFs is stored as AL_CUT in the HDU header.

Modified IRFs with true energy scaled by a given factor can be created to evaluate
the systematic uncertainty in the light collection efficiency. This can be done by
setting a value different from one for the "scale_true_energy" argument present in
the DataBinning Component of the configuration file of the IRF creation Tool.
(The true energy of the MC events will be scaled before filling the IRFs histograms
when pyirf commands are used. The effects expected are a non-diagonal energy dispersion
matrix and a different spectrum).


"""

from astropy import table
Expand Down Expand Up @@ -136,7 +145,12 @@ class IRFFITSWriter(Tool):
--global-alpha-cut 10
--source-dep

"""
To build modified IRFs by specifying a scaling factor applying to the true energy (without using a config file):
> lstchain_create_irf_files
-g /path/to/DL2_MC_gamma_file.h5
-o /path/to/irf.fits.gz
--scale-true-energy 1.15
"""

input_gamma_dl2 = traits.Path(
help="Input MC gamma DL2 file",
Expand Down Expand Up @@ -221,6 +235,7 @@ class IRFFITSWriter(Tool):
"global-alpha-cut": "DL3Cuts.global_alpha_cut",
"allowed-tels": "DL3Cuts.allowed_tels",
"overwrite": "IRFFITSWriter.overwrite",
"scale-true-energy": "DataBinning.scale_true_energy"
}

flags = {
Expand Down Expand Up @@ -317,6 +332,12 @@ def start(self):
p["geomag_params"],
) = read_mc_dl2_to_QTable(p["file"])


if self.data_bin.scale_true_energy != 1.0:
p["events"]["true_energy"] *= self.data_bin.scale_true_energy
p["simulation_info"].energy_min *= self.data_bin.scale_true_energy
p["simulation_info"].energy_max *= self.data_bin.scale_true_energy

p["mc_type"] = check_mc_type(p["file"])

self.log.debug(
Expand Down Expand Up @@ -527,7 +548,10 @@ def start(self):
geomag_params["GEOMAG_DELTA"].to_value(u.deg),
"deg",
)

extra_headers["ETRUE_SCALE"]= (
self.data_bin.scale_true_energy
)

if self.point_like:
self.log.info("Generating point_like IRF HDUs")
else:
Expand Down
128 changes: 121 additions & 7 deletions lstchain/tools/tests/test_tools.py
Original file line number Diff line number Diff line change
Expand Up @@ -381,11 +381,16 @@ def test_index_dl3_files(temp_dir_observed_files):
assert 2008 in data.obs_table["OBS_ID"]

for hdu_name in [
'EVENTS', 'GTI', 'POINTING',
'EFFECTIVE AREA', 'ENERGY DISPERSION',
'BACKGROUND', 'PSF'
"EVENTS",
"GTI",
"POINTING",
"EFFECTIVE AREA",
"ENERGY DISPERSION",
"BACKGROUND",
"PSF",
]:
assert hdu_name in data.hdu_table['HDU_NAME']
assert hdu_name in data.hdu_table["HDU_NAME"]


@pytest.mark.private_data
def test_index_srcdep_dl3_files(temp_dir_observed_srcdep_files):
Expand All @@ -411,7 +416,116 @@ def test_index_srcdep_dl3_files(temp_dir_observed_srcdep_files):
assert 2008 in data.obs_table["OBS_ID"]

for hdu_name in [
'EVENTS', 'GTI', 'POINTING',
'EFFECTIVE AREA', 'ENERGY DISPERSION'
"EVENTS",
"GTI",
"POINTING",
"EFFECTIVE AREA",
"ENERGY DISPERSION",
]:
assert hdu_name in data.hdu_table['HDU_NAME']
assert hdu_name in data.hdu_table["HDU_NAME"]


@pytest.mark.private_data
def test_add_scale_true_energy_in_irfs(temp_dir_observed_files, simulated_dl2_file):
"""
Checking the validity of modified IRFs after scaling the True Energy by a factor.
"""

import astropy.units as u
from gammapy.irf import EffectiveAreaTable2D, EnergyDispersion2D
from lstchain.tools.lstchain_create_irf_files import IRFFITSWriter

irf_file = temp_dir_observed_files / "fe_irf.fits.gz"
irf_file_mod = temp_dir_observed_files / "mod_irf.fits.gz"
config_file = os.path.join(os.getcwd(), "docs/examples/irf_dl3_tool_config.json")

assert (
run_tool(
IRFFITSWriter(),
argv=[
f"--input-gamma-dl2={simulated_dl2_file}",
f"--input-proton-dl2={simulated_dl2_file}",
f"--input-electron-dl2={simulated_dl2_file}",
f"--output-irf-file={irf_file}",
f"--config={config_file}",
"--overwrite",
"--DataBinning.true_energy_n_bins=2",
"--DataBinning.reco_energy_n_bins=2",
"--DataBinning.true_energy_min: 0.2",
"--DataBinning.true_energy_max: 0.3",
"--DL3Cuts.min_event_p_en_bin=2",
],
cwd=temp_dir_observed_files,
)
== 0
)
assert (
run_tool(
IRFFITSWriter(),
argv=[
f"--input-gamma-dl2={simulated_dl2_file}",
f"--input-proton-dl2={simulated_dl2_file}",
f"--input-electron-dl2={simulated_dl2_file}",
f"--output-irf-file={irf_file_mod}",
f"--config={config_file}",
"--overwrite",
"--DataBinning.true_energy_n_bins=2",
"--DataBinning.reco_energy_n_bins=2",
"--DataBinning.true_energy_min: 0.2",
"--DataBinning.true_energy_max: 0.3",
"--DL3Cuts.min_event_p_en_bin=2",
"--DataBinning.scale_true_energy=1.15",
],
cwd=temp_dir_observed_files,
)
== 0
)

aeff_hdu = EffectiveAreaTable2D.read(irf_file, hdu="EFFECTIVE AREA")
aeff_mod_hdu = EffectiveAreaTable2D.read(irf_file_mod, hdu="EFFECTIVE AREA")

edisp_hdu = EnergyDispersion2D.read(irf_file, hdu="ENERGY DISPERSION")
edisp_mod_hdu = EnergyDispersion2D.read(irf_file_mod, hdu="ENERGY DISPERSION")

assert aeff_mod_hdu.data.shape == aeff_hdu.data.shape
assert edisp_mod_hdu.data.shape == edisp_hdu.data.shape

edisp = EnergyDispersion2D.read(irf_file)
edisp_mod = EnergyDispersion2D.read(irf_file_mod)

e_migra = edisp.axes["migra"].center
e_migra_mod = edisp_mod.axes["migra"].center

e_true_list = [0.2, 2, 20]
e_migra_prob = []
e_migra_prob_mod = []

for i in e_true_list:
e_true = i * u.TeV
e_migra_prob.append(
edisp.evaluate(
offset=0.4 * u.deg,
energy_true=e_true,
migra=e_migra,
)
)
e_migra_prob_mod.append(
edisp_mod.evaluate(
offset=0.4 * u.deg,
energy_true=e_true,
migra=e_migra_mod,
)
)

# Check that the maximum of the density probability of the migration has shifted
order_max = []
order_max_mod = []
for idx, _ in enumerate(e_true_list):
for j in range(len(e_migra)):
if e_migra_prob[idx][j] > e_migra_prob[idx][j - 1]:
order_max.append(j)
if e_migra_prob_mod[idx][j] > e_migra_prob_mod[idx][j - 1]:
order_max_mod.append(j)

for i in range(len(order_max)):
assert order_max[i] != order_max_mod[i]