Coverage for tests / test_interface.py: 32%

138 statements  

« prev     ^ index     » next       coverage.py v7.14.0, created at 2026-05-20 08:10 +0000

1# This file is part of rucio_register 

2# 

3# Developed for the LSST Data Management System. 

4# This product includes software developed by the LSST Project 

5# (https://www.lsst.org). 

6# See the COPYRIGHT file at the top-level directory of this distribution 

7# for details of code ownership. 

8# 

9# This program is free software: you can redistribute it and/or modify 

10# it under the terms of the GNU General Public License as published by 

11# the Free Software Foundation, either version 3 of the License, or 

12# (at your option) any later version. 

13# 

14# This program is distributed in the hope that it will be useful, 

15# but WITHOUT ANY WARRANTY; without even the implied warranty of 

16# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 

17# GNU General Public License for more details. 

18# 

19# You should have received a copy of the GNU General Public License 

20# along with this program. If not, see <https://www.gnu.org/licenses/>. 

21 

22 

23import os 

24import shutil 

25import tempfile 

26from unittest.mock import MagicMock, patch 

27 

28from rucio.client.didclient import DIDClient 

29from rucio.client.replicaclient import ReplicaClient 

30from rucio.common.exception import ( 

31 DataIdentifierAlreadyExists, 

32 DataIdentifierNotFound, 

33 FileAlreadyExists, 

34 RucioException, 

35) 

36 

37import lsst.utils.tests 

38from lsst.daf.butler import Butler, DatasetRef, DimensionUniverse 

39 

40# from lsst.daf.butler.registry import DatasetTypeError, MissingCollectionError 

41from lsst.resources import ResourceInfo, ResourcePath 

42from lsst.resources.file import FileResourcePath 

43from lsst.rucio.register.data_type import DataType 

44from lsst.rucio.register.rucio_interface import RucioInterface 

45 

46 

47class InterfaceTestCase(lsst.utils.tests.TestCase): 

48 maxDiff = None 

49 

50 def setUp(self): 

51 self.butler_repo = tempfile.mkdtemp(dir="/tmp") 

52 test_dir = os.path.abspath(os.path.dirname(__file__)) 

53 

54 self.dataset_ref_file = os.path.join(test_dir, "data", "dataset_ref.json") 

55 

56 Butler.makeRepo(self.butler_repo) 

57 

58 data_name = "visitSummary_HSC_y_HSC-Y_318_HSC_runs_RC2_w_2023_32_DM-40356_20230814T170253Z.fits" 

59 json_name = "visitSummary_HSC_y_HSC-Y_318_HSC_runs_RC2_w_2023_32_DM-40356_20230814T170253Z.json" 

60 

61 self.data_file = os.path.join(test_dir, "data", data_name) 

62 self.json_file = os.path.join(test_dir, "data", json_name) 

63 

64 self.butler = Butler(self.butler_repo, writeable=True) 

65 self.butler.getURI = MagicMock(return_value=ResourcePath(f"file://{self.data_file}")) 

66 

67 self.rse_root = tempfile.mkdtemp(dir="/tmp") 

68 

69 # patch __init__ methods 

70 self.rc_init = patch.object(ReplicaClient, "__init__", return_value=None) 

71 self.dc_init = patch.object(DIDClient, "__init__", return_value=None) 

72 self.rc_add_replicas = patch.object(ReplicaClient, "add_replicas", return_value=None) 

73 self.dc_attach_dids = patch.object(DIDClient, "attach_dids", return_value=None) 

74 self.rand = patch("random.randint", return_value=1) 

75 

76 self.mock_rc_init = self.rc_init.start() 

77 self.mock_dc_init = self.dc_init.start() 

78 self.mock_rc_add_replicas = self.rc_add_replicas.start() 

79 self.mock_dc_attach_dids = self.dc_attach_dids.start() 

80 self.mock_rand = self.rand.start() 

81 

82 rucio_rse = "DRR1" 

83 scope = "test" 

84 dtn_url = "root://xrd1:1094//rucio" 

85 self.ri = RucioInterface(self.butler, rucio_rse, scope, self.rse_root, dtn_url, DataType.DATA_PRODUCT) 

86 

87 def testChecksumsCase(self): 

88 fake_info = MagicMock(spec=ResourceInfo) 

89 fake_info.checksums = {"adler32": "abcd1234"} 

90 fake_info.size = 1234 

91 fake_info.is_file = True 

92 fake_info.last_modified = None 

93 fake_info.uri = f"file://{self.data_file}" 

94 

95 patcher = patch.object(FileResourcePath, "get_info", return_value=fake_info) 

96 patcher.start() 

97 self.addCleanup(patcher.stop) 

98 

99 with open(self.dataset_ref_file) as f: 

100 json_ref = f.readline() 

101 

102 ref = DatasetRef.from_json(json_ref, DimensionUniverse()) 

103 

104 self.butler.registry.registerDatasetType(ref.datasetType) 

105 cnt = self.ri.register_as_replicas("mydataset", [ref]) 

106 self.assertEqual(cnt, 1) 

107 

108 rb = self.ri._make_dataset_ref_bundle("mydataset", ref) 

109 self.assertEqual(rb.dataset_id, "mydataset") 

110 

111 did = rb.did.model_dump() 

112 

113 self.assertEqual(did["adler32"], "abcd1234") 

114 

115 def testInterfaceTestCase(self): 

116 dtn_url = "root://xrd1:1094//rucio" 

117 

118 json_ref = None 

119 with open(self.dataset_ref_file) as f: 

120 json_ref = f.readline() 

121 

122 ref = DatasetRef.from_json(json_ref, DimensionUniverse()) 

123 

124 self.butler.registry.registerDatasetType(ref.datasetType) 

125 cnt = self.ri.register_as_replicas("mydataset", [ref]) 

126 self.assertEqual(cnt, 1) 

127 

128 rb = self.ri._make_dataset_ref_bundle("mydataset", ref) 

129 self.assertEqual(rb.dataset_id, "mydataset") 

130 

131 did = rb.did.model_dump() 

132 self.assertEqual(did["pfn"], f"{dtn_url}{self.data_file}") 

133 self.assertEqual(did["bytes"], 1365120) 

134 self.assertEqual(did["adler32"], "480be4de") 

135 self.assertEqual(did["name"], self.data_file) 

136 self.assertEqual(did["scope"], "test") 

137 

138 meta = did["meta"] 

139 self.assertEqual(meta["rubin_butler"], DataType.DATA_PRODUCT) 

140 

141 def common(self): 

142 json_ref = None 

143 with open(self.dataset_ref_file) as f: 

144 json_ref = f.readline() 

145 

146 ref = DatasetRef.from_json(json_ref, DimensionUniverse()) 

147 

148 self.butler.registry.registerDatasetType(ref.datasetType) 

149 self.ri.register_as_replicas("mydataset", [ref]) 

150 

151 @patch.object(ReplicaClient, "add_replicas", side_effect=RucioException("failed")) 

152 def testException1TestCase(self, MC1): 

153 self.ri.register_to_dataset = MagicMock(name="register_to_dataset") 

154 with self.assertRaises(Exception): 

155 self.common() 

156 

157 @patch.object(DIDClient, "add_files_to_dataset", side_effect=RucioException("failed")) 

158 def testException2TestCase(self, MC1): 

159 with self.assertRaises(Exception): 

160 self.common() 

161 

162 @patch.object(DIDClient, "add_files_to_dataset", side_effect=FileAlreadyExists("failed")) 

163 def testException3TestCase(self, MC1): 

164 self.common() 

165 

166 @patch.object(DIDClient, "add_dataset", return_value=None) 

167 @patch.object(DIDClient, "add_files_to_dataset", side_effect=DataIdentifierNotFound("failed")) 

168 def testException4TestCase(self, MC1, MC2): 

169 with self.assertRaises(Exception): 

170 self.common() 

171 

172 @patch.object(DIDClient, "add_files_to_dataset", side_effect=RucioException("failed")) 

173 def testException5TestCase(self, MC1): 

174 with self.assertRaises(Exception): 

175 self.common() 

176 

177 @patch.object(DIDClient, "add_dataset", side_effect=DataIdentifierAlreadyExists("failed")) 

178 @patch.object(DIDClient, "add_files_to_dataset", side_effect=DataIdentifierNotFound("failed")) 

179 def testException6TestCase(self, MC1, MC2): 

180 with self.assertRaises(Exception): 

181 self.common() 

182 

183 @patch.object(DIDClient, "add_files_to_dataset", side_effect=RucioException("failed")) 

184 def testException7TestCase(self, MC1): 

185 with self.assertRaises(Exception): 

186 self.common() 

187 

188 @patch.object(DIDClient, "add_dataset", side_effect=RucioException("failed")) 

189 @patch.object(DIDClient, "add_files_to_dataset", side_effect=DataIdentifierNotFound("failed")) 

190 def testException8TestCase(self, MC1, MC2): 

191 with self.assertRaises(Exception): 

192 self.common() 

193 

194 @patch.object(DIDClient, "add_files_to_dataset", side_effect=RucioException("failed")) 

195 def testException9Case(self, MC1): 

196 rucio_rse = "DRR1" 

197 scope = "test" 

198 dtn_url = "root://xrd1:1094//rucio" 

199 

200 ri = RucioInterface(self.butler, rucio_rse, scope, self.rse_root, dtn_url, DataType.DATA_PRODUCT) 

201 with self.assertRaises(Exception): 

202 ri._add_file_to_dataset_with_retries(None, None) 

203 

204 def tearDown(self): 

205 patch.stopall() 

206 shutil.rmtree(self.butler_repo, ignore_errors=True) 

207 shutil.rmtree(self.rse_root, ignore_errors=True) 

208 

209 

210class MemoryTester(lsst.utils.tests.MemoryTestCase): 

211 pass 

212 

213 

214def setup_module(module): 

215 lsst.utils.tests.init()