Matteo Scandolo | d2044a4 | 2017-08-07 16:08:28 -0700 | [diff] [blame] | 1 | |
| 2 | # Copyright 2017-present Open Networking Foundation |
| 3 | # |
| 4 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | # you may not use this file except in compliance with the License. |
| 6 | # You may obtain a copy of the License at |
| 7 | # |
| 8 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | # |
| 10 | # Unless required by applicable law or agreed to in writing, software |
| 11 | # distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | # See the License for the specific language governing permissions and |
| 14 | # limitations under the License. |
| 15 | |
| 16 | |
Matteo Scandolo | 67654fa | 2017-06-09 09:33:17 -0700 | [diff] [blame] | 17 | import unittest |
| 18 | import os |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 19 | from xosgenx.generator import XOSProcessor |
Matteo Scandolo | 67654fa | 2017-06-09 09:33:17 -0700 | [diff] [blame] | 20 | from helpers import FakeArgs |
Sapan Bhatia | d022aeb | 2017-06-07 15:49:55 +0200 | [diff] [blame] | 21 | import yaml |
| 22 | |
Matteo Scandolo | 67654fa | 2017-06-09 09:33:17 -0700 | [diff] [blame] | 23 | PROTO_EXPECTED_OUTPUT = """ |
| 24 | message VRouterPort { |
| 25 | option bases = "XOSBase"; |
| 26 | optional string name = 1 [ null = "True", max_length = "20", blank = "True", help_text = ""port friendly name"", modifier = "optional", db_index = "False" ]; |
| 27 | required string openflow_id = 2 [ null = "False", max_length = "21", blank = "False", help_text = ""port identifier in ONOS"", modifier = "required", db_index = "False" ]; |
| 28 | required int32 vrouter_device = 3 [ null = "False", blank = "False", model = "VRouterDevice", modifier = "required", type = "link", port = "ports", link_type = "manytoone", db_index = "True" ]; |
| 29 | required int32 vrouter_service = 4 [ null = "False", blank = "False", model = "VRouterService", modifier = "required", type = "link", port = "device_ports", link_type = "manytoone", db_index = "True" ]; |
Sapan Bhatia | d022aeb | 2017-06-07 15:49:55 +0200 | [diff] [blame] | 30 | } |
| 31 | """ |
Matteo Scandolo | 67654fa | 2017-06-09 09:33:17 -0700 | [diff] [blame] | 32 | VROUTER_XPROTO = os.path.abspath(os.path.dirname(os.path.realpath(__file__)) + "/xproto/vrouterport.xproto") |
Sapan Bhatia | d022aeb | 2017-06-07 15:49:55 +0200 | [diff] [blame] | 33 | |
Matteo Scandolo | 67654fa | 2017-06-09 09:33:17 -0700 | [diff] [blame] | 34 | # Generate other formats from xproto |
| 35 | class XProtoTranslatorTest(unittest.TestCase): |
| 36 | def _test_proto_generator(self): |
| 37 | args = FakeArgs() |
| 38 | args.files = [VROUTER_XPROTO] |
| 39 | args.target = 'proto.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 40 | output = XOSProcessor.process(args) |
Matteo Scandolo | 67654fa | 2017-06-09 09:33:17 -0700 | [diff] [blame] | 41 | self.assertEqual(output, PROTO_EXPECTED_OUTPUT) |
| 42 | |
Sapan Bhatia | d022aeb | 2017-06-07 15:49:55 +0200 | [diff] [blame] | 43 | def test_yaml_generator(self): |
| 44 | xproto = \ |
| 45 | """ |
Matteo Scandolo | 67654fa | 2017-06-09 09:33:17 -0700 | [diff] [blame] | 46 | option app_label = "test"; |
Sapan Bhatia | d022aeb | 2017-06-07 15:49:55 +0200 | [diff] [blame] | 47 | |
| 48 | message Port (PlCoreBase,ParameterMixin){ |
| 49 | required manytoone network->Network:links = 1 [db_index = True, null = False, blank = False]; |
| 50 | optional manytoone instance->Instance:ports = 2 [db_index = True, null = True, blank = True]; |
| 51 | optional string ip = 3 [max_length = 39, content_type = "ip", blank = True, help_text = "Instance ip address", null = True, db_index = False]; |
| 52 | optional string port_id = 4 [help_text = "Neutron port id", max_length = 256, null = True, db_index = False, blank = True]; |
| 53 | optional string mac = 5 [help_text = "MAC address associated with this port", max_length = 256, null = True, db_index = False, blank = True]; |
| 54 | required bool xos_created = 6 [default = False, null = False, db_index = False, blank = True]; |
| 55 | } |
| 56 | |
| 57 | message Instance (PlCoreBase){ |
| 58 | optional string instance_id = 1 [max_length = 200, content_type = "stripped", blank = True, help_text = "Nova instance id", null = True, db_index = False]; |
| 59 | optional string instance_uuid = 2 [max_length = 200, content_type = "stripped", blank = True, help_text = "Nova instance uuid", null = True, db_index = False]; |
| 60 | required string name = 3 [max_length = 200, content_type = "stripped", blank = False, help_text = "Instance name", null = False, db_index = False]; |
| 61 | optional string instance_name = 4 [max_length = 200, content_type = "stripped", blank = True, help_text = "OpenStack generated name", null = True, db_index = False]; |
| 62 | optional string ip = 5 [max_length = 39, content_type = "ip", blank = True, help_text = "Instance ip address", null = True, db_index = False]; |
| 63 | required manytoone image->Image:instances = 6 [db_index = True, null = False, blank = False]; |
| 64 | optional manytoone creator->User:instances = 7 [db_index = True, null = True, blank = True]; |
| 65 | required manytoone slice->Slice:instances = 8 [db_index = True, null = False, blank = False]; |
| 66 | required manytoone deployment->Deployment:instance_deployment = 9 [db_index = True, null = False, blank = False]; |
| 67 | required manytoone node->Node:instances = 10 [db_index = True, null = False, blank = False]; |
| 68 | required int32 numberCores = 11 [help_text = "Number of cores for instance", default = 0, null = False, db_index = False, blank = False]; |
| 69 | required manytoone flavor->Flavor:instance = 12 [help_text = "Flavor of this instance", default = "get_default_flavor()", null = False, db_index = True, blank = False]; |
| 70 | optional string userData = 13 [help_text = "user_data passed to instance during creation", null = True, db_index = False, blank = True]; |
| 71 | required string isolation = 14 [default = "vm", choices = "(('vm', 'Virtual Machine'), ('container', 'Container'), ('container_vm', 'Container In VM'))", max_length = 30, blank = False, null = False, db_index = False]; |
| 72 | optional string volumes = 15 [help_text = "Comma-separated list of directories to expose to parent context", null = True, db_index = False, blank = True]; |
| 73 | optional manytoone parent->Instance:instance = 16 [help_text = "Parent Instance for containers nested inside of VMs", null = True, db_index = True, blank = True]; |
| 74 | required manytomany tags->Tag = 17 [db_index = False, null = False, blank = True]; |
| 75 | } |
| 76 | |
| 77 | message Network (PlCoreBase,ParameterMixin) { |
| 78 | required string name = 1 [db_index = False, max_length = 32, null = False, blank = False]; |
| 79 | required manytoone template->NetworkTemplate:network = 2 [db_index = True, null = False, blank = False]; |
| 80 | required string subnet = 3 [db_index = False, max_length = 32, null = False, blank = True]; |
| 81 | required string start_ip = 4 [db_index = False, max_length = 32, null = False, blank = True]; |
| 82 | required string end_ip = 5 [db_index = False, max_length = 32, null = False, blank = True]; |
| 83 | optional string ports = 6 [db_index = False, max_length = 1024, null = True, blank = True]; |
| 84 | optional string labels = 7 [db_index = False, max_length = 1024, null = True, blank = True]; |
| 85 | required manytoone owner->Slice:ownedNetworks = 8 [help_text = "Slice that owns control of this Network", null = False, db_index = True, blank = False]; |
| 86 | required int32 guaranteed_bandwidth = 9 [default = 0, null = False, db_index = False, blank = False]; |
| 87 | required bool permit_all_slices = 10 [default = False, null = False, db_index = False, blank = True]; |
| 88 | optional string topology_parameters = 11 [db_index = False, null = True, blank = True]; |
| 89 | optional string controller_url = 12 [db_index = False, max_length = 1024, null = True, blank = True]; |
| 90 | optional string controller_parameters = 13 [db_index = False, null = True, blank = True]; |
| 91 | optional string network_id = 14 [help_text = "Quantum network", max_length = 256, null = True, db_index = False, blank = True]; |
| 92 | optional string router_id = 15 [help_text = "Quantum router id", max_length = 256, null = True, db_index = False, blank = True]; |
| 93 | optional string subnet_id = 16 [help_text = "Quantum subnet id", max_length = 256, null = True, db_index = False, blank = True]; |
| 94 | required bool autoconnect = 17 [help_text = "This network can be autoconnected to the slice that owns it", default = True, null = False, db_index = False, blank = True]; |
| 95 | required manytomany permitted_slices->Slice/Network_permitted_slices:availableNetworks = 18 [db_index = False, null = False, blank = True]; |
| 96 | required manytomany slices->Slice/NetworkSlice:networks = 19 [db_index = False, null = False, blank = True]; |
| 97 | required manytomany instances->Instance/Port:networks = 20 [db_index = False, null = False, blank = True]; |
| 98 | } |
| 99 | |
| 100 | message Slice (PlCoreBase){ |
| 101 | required string name = 1 [max_length = 80, content_type = "stripped", blank = False, help_text = "The Name of the Slice", null = False, db_index = False]; |
| 102 | required bool enabled = 2 [help_text = "Status for this Slice", default = True, null = False, db_index = False, blank = True]; |
| 103 | required bool omf_friendly = 3 [default = False, null = False, db_index = False, blank = True]; |
| 104 | required string description = 4 [help_text = "High level description of the slice and expected activities", max_length = 1024, null = False, db_index = False, blank = True]; |
| 105 | required string slice_url = 5 [db_index = False, max_length = 512, null = False, content_type = "url", blank = True]; |
| 106 | required manytoone site->Site:slices = 6 [help_text = "The Site this Slice belongs to", null = False, db_index = True, blank = False]; |
| 107 | required int32 max_instances = 7 [default = 10, null = False, db_index = False, blank = False]; |
| 108 | optional manytoone service->Service:slices = 8 [db_index = True, null = True, blank = True]; |
| 109 | optional string network = 9 [blank = True, max_length = 256, null = True, db_index = False, choices = "((None, 'Default'), ('host', 'Host'), ('bridged', 'Bridged'), ('noauto', 'No Automatic Networks'))"]; |
| 110 | optional string exposed_ports = 10 [db_index = False, max_length = 256, null = True, blank = True]; |
| 111 | optional manytoone serviceClass->ServiceClass:slices = 11 [db_index = True, null = True, blank = True]; |
| 112 | optional manytoone creator->User:slices = 12 [db_index = True, null = True, blank = True]; |
| 113 | optional manytoone default_flavor->Flavor:slices = 13 [db_index = True, null = True, blank = True]; |
| 114 | optional manytoone default_image->Image:slices = 14 [db_index = True, null = True, blank = True]; |
| 115 | optional manytoone default_node->Node:slices = 15 [db_index = True, null = True, blank = True]; |
| 116 | optional string mount_data_sets = 16 [default = "GenBank", max_length = 256, content_type = "stripped", blank = True, null = True, db_index = False]; |
| 117 | required string default_isolation = 17 [default = "vm", choices = "(('vm', 'Virtual Machine'), ('container', 'Container'), ('container_vm', 'Container In VM'))", max_length = 30, blank = False, null = False, db_index = False]; |
| 118 | required manytomany tags->Tag = 18 [db_index = False, null = False, blank = True]; |
| 119 | } |
| 120 | """ |
| 121 | |
Matteo Scandolo | 67654fa | 2017-06-09 09:33:17 -0700 | [diff] [blame] | 122 | args = FakeArgs() |
| 123 | args.inputs = xproto |
| 124 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 125 | output = XOSProcessor.process(args) |
Sapan Bhatia | d022aeb | 2017-06-07 15:49:55 +0200 | [diff] [blame] | 126 | |
Sapan Bhatia | d022aeb | 2017-06-07 15:49:55 +0200 | [diff] [blame] | 127 | yaml_ir = yaml.load(output) |
| 128 | self.assertEqual(len(yaml_ir['items']), 4) |
| 129 | |
Matteo Scandolo | 5341859 | 2017-07-26 15:51:29 -0700 | [diff] [blame] | 130 | def test_gui_hidden_models(self): |
| 131 | xproto = \ |
| 132 | """ |
| 133 | option app_label = "test"; |
| 134 | |
| 135 | message Foo { |
Matteo Scandolo | e425f9d | 2017-08-15 15:56:19 -0700 | [diff] [blame] | 136 | option gui_hidden = True; |
Matteo Scandolo | 5341859 | 2017-07-26 15:51:29 -0700 | [diff] [blame] | 137 | required string name = 1 [ null = "False", blank="False"]; |
| 138 | } |
| 139 | |
| 140 | message Bar { |
| 141 | option gui_hidden = "False"; |
| 142 | required string name = 1 [ null = "False", blank="False"]; |
| 143 | } |
| 144 | """ |
| 145 | args = FakeArgs() |
| 146 | args.inputs = xproto |
| 147 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 148 | output = XOSProcessor.process(args) |
Matteo Scandolo | 5341859 | 2017-07-26 15:51:29 -0700 | [diff] [blame] | 149 | yaml_ir = yaml.load(output) |
| 150 | self.assertEqual(len(yaml_ir['items']), 1) |
| 151 | self.assertIn('Bar', output) |
| 152 | self.assertNotIn('Foo', output) |
| 153 | |
| 154 | def test_gui_hidden_model_fields(self): |
| 155 | xproto = \ |
| 156 | """ |
| 157 | option app_label = "test"; |
| 158 | |
| 159 | message Foo { |
| 160 | required string name = 1 [ null = "False", blank="False"]; |
| 161 | required string secret = 1 [ null = "False", blank="False", gui_hidden = "True"]; |
| 162 | } |
| 163 | """ |
| 164 | args = FakeArgs() |
| 165 | args.inputs = xproto |
| 166 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 167 | output = XOSProcessor.process(args) |
Matteo Scandolo | 5341859 | 2017-07-26 15:51:29 -0700 | [diff] [blame] | 168 | yaml_ir = yaml.load(output) |
| 169 | self.assertEqual(len(yaml_ir['items']), 1) |
| 170 | self.assertIn('name', output) |
| 171 | self.assertNotIn('secret', output) |
Matteo Scandolo | 292cc2a | 2017-07-31 19:02:12 -0700 | [diff] [blame] | 172 | |
| 173 | def test_static_options(self): |
| 174 | xproto = \ |
| 175 | """ |
| 176 | option app_label = "test"; |
| 177 | |
| 178 | message Foo { |
| 179 | required string name = 1 [ null = "False", blank="False"]; |
| 180 | required string isolation = 14 [default = "vm", choices = "(('vm', 'Virtual Machine'), ('container', 'Container'), ('container_vm', 'Container In VM'))", max_length = 30, blank = False, null = False, db_index = False]; |
| 181 | } |
| 182 | """ |
| 183 | |
| 184 | args = FakeArgs() |
| 185 | args.inputs = xproto |
| 186 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 187 | output = XOSProcessor.process(args) |
Matteo Scandolo | 292cc2a | 2017-07-31 19:02:12 -0700 | [diff] [blame] | 188 | self.assertIn("options:", output) |
| 189 | self.assertIn(" {'id': 'container_vm', 'label': 'Container In VM'}", output) |
| 190 | |
| 191 | def test_not_static_options(self): |
| 192 | xproto = \ |
| 193 | """ |
| 194 | option app_label = "test"; |
| 195 | |
| 196 | message Foo { |
| 197 | required string name = 1 [ null = "False", blank="False"]; |
| 198 | } |
| 199 | """ |
| 200 | |
| 201 | args = FakeArgs() |
| 202 | args.inputs = xproto |
| 203 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 204 | output = XOSProcessor.process(args) |
Matteo Scandolo | 292cc2a | 2017-07-31 19:02:12 -0700 | [diff] [blame] | 205 | self.assertNotIn("options:", output) |
| 206 | |
| 207 | def test_default_value_in_modeldef(self): |
| 208 | xproto = \ |
| 209 | """ |
| 210 | option app_label = "test"; |
| 211 | |
| 212 | message Foo { |
| 213 | required string name = 1 [ null = "False", blank="False", default = "bar"]; |
| 214 | required string falsetrue = 1 [ null = "False", blank="False", default = False]; |
| 215 | required string truefalse = 1 [ null = "False", blank="False", default = True]; |
| 216 | required string some = 1 [ null = "False", blank="False", default = None]; |
| 217 | required string zero = 1 [ null = "False", blank="False", default = 0]; |
| 218 | } |
| 219 | """ |
| 220 | |
| 221 | args = FakeArgs() |
| 222 | args.inputs = xproto |
| 223 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 224 | output = XOSProcessor.process(args) |
Matteo Scandolo | 292cc2a | 2017-07-31 19:02:12 -0700 | [diff] [blame] | 225 | self.assertIn('default: "bar"', output) |
| 226 | self.assertIn('default: "false"', output) |
| 227 | self.assertIn('default: "true"', output) |
| 228 | self.assertIn('default: "null"', output) |
| 229 | self.assertIn('default: "0"', output) |
| 230 | |
| 231 | def test_not_default_value_in_modeldef(self): |
| 232 | xproto = \ |
| 233 | """ |
| 234 | option app_label = "test"; |
| 235 | |
| 236 | message Foo { |
| 237 | required string name = 1 [ null = "False", blank="False"]; |
| 238 | } |
| 239 | """ |
| 240 | |
| 241 | args = FakeArgs() |
| 242 | args.inputs = xproto |
| 243 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 244 | output = XOSProcessor.process(args) |
Matteo Scandolo | 292cc2a | 2017-07-31 19:02:12 -0700 | [diff] [blame] | 245 | self.assertNotIn('default:', output) |
| 246 | |
Matteo Scandolo | 1f826a4 | 2017-08-02 12:02:02 -0700 | [diff] [blame] | 247 | def test_one_to_many_in_modeldef(self): |
| 248 | xproto = \ |
| 249 | """ |
| 250 | option app_label = "test"; |
| 251 | |
| 252 | message ServiceDependency { |
| 253 | required manytoone provider_service->Service:provided_dependencies = 1; |
| 254 | required manytoone subscriber_service->Service:subscribed_dependencies = 2; |
| 255 | } |
| 256 | |
| 257 | message Service { |
| 258 | required string name = 1; |
| 259 | } |
| 260 | """ |
| 261 | |
| 262 | args = FakeArgs() |
| 263 | args.inputs = xproto |
| 264 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 265 | output = XOSProcessor.process(args) |
Matteo Scandolo | 1f826a4 | 2017-08-02 12:02:02 -0700 | [diff] [blame] | 266 | # Service deps model |
| 267 | self.assertIn('{model: Service, type: manytoone, on_field: provider_service}', output) |
| 268 | self.assertIn('{model: Service, type: manytoone, on_field: provider_service}', output) |
| 269 | |
| 270 | # Service model |
| 271 | self.assertIn('{model: ServiceDependency, type: onetomany, on_field: provider_service}', output) |
| 272 | self.assertIn('{model: ServiceDependency, type: onetomany, on_field: provider_service}', output) |
| 273 | |
Matteo Scandolo | e425f9d | 2017-08-15 15:56:19 -0700 | [diff] [blame] | 274 | def test_model_description(self): |
| 275 | xproto = \ |
| 276 | """ |
| 277 | option app_label = "test"; |
| 278 | |
| 279 | message Foo { |
| 280 | option description="This is the Foo model"; |
| 281 | required string name = 1 [ null = "False", blank="False"]; |
| 282 | required string isolation = 14 [default = "vm", choices = "(('vm', 'Virtual Machine'), ('container', 'Container'), ('container_vm', 'Container In VM'))", max_length = 30, blank = False, null = False, db_index = False]; |
| 283 | } |
| 284 | |
| 285 | message Bar { |
| 286 | required string name = 1; |
| 287 | } |
| 288 | """ |
| 289 | |
| 290 | args = FakeArgs() |
| 291 | args.inputs = xproto |
| 292 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 293 | output = XOSProcessor.process(args) |
Matteo Scandolo | e425f9d | 2017-08-15 15:56:19 -0700 | [diff] [blame] | 294 | self.assertIn('description: "This is the Foo model"', output) |
| 295 | |
| 296 | def test_model_verbose_name(self): |
| 297 | xproto = \ |
| 298 | """ |
| 299 | option app_label = "test"; |
| 300 | |
| 301 | message Foo { |
| 302 | option verbose_name="Verbose Foo Name"; |
| 303 | required string name = 1 [ null = "False", blank="False"]; |
| 304 | required string isolation = 14 [default = "vm", choices = "(('vm', 'Virtual Machine'), ('container', 'Container'), ('container_vm', 'Container In VM'))", max_length = 30, blank = False, null = False, db_index = False]; |
| 305 | } |
| 306 | |
| 307 | message Bar { |
| 308 | required string name = 1; |
| 309 | } |
| 310 | """ |
| 311 | |
| 312 | args = FakeArgs() |
| 313 | args.inputs = xproto |
| 314 | args.target = 'modeldefs.xtarget' |
Sapan Bhatia | bfb233a | 2018-02-09 14:53:09 -0800 | [diff] [blame] | 315 | output = XOSProcessor.process(args) |
Matteo Scandolo | e425f9d | 2017-08-15 15:56:19 -0700 | [diff] [blame] | 316 | self.assertIn('verbose_name: "Verbose Foo Name"', output) |
| 317 | |
Matteo Scandolo | 23cf15f | 2018-03-06 18:12:36 -0800 | [diff] [blame] | 318 | def test_feedback_field(self): |
| 319 | xproto = \ |
| 320 | """ |
| 321 | option app_label = "test"; |
| 322 | |
| 323 | message ParentFoo { |
| 324 | required string parent_name = 1 [null = False, blank = False, feedback_state = True]; |
| 325 | } |
| 326 | |
| 327 | message Foo (ParentFoo) { |
| 328 | required string name = 1 [null = False, blank = False, feedback_state = True]; |
| 329 | } |
| 330 | """ |
| 331 | |
| 332 | args = FakeArgs() |
| 333 | args.inputs = xproto |
| 334 | args.target = 'modeldefs.xtarget' |
| 335 | output = XOSProcessor.process(args) |
| 336 | |
| 337 | read_only = filter(lambda s: 'read_only: True' in s, output.splitlines()) |
| 338 | self.assertEqual(len(read_only), 3) # readonly is 1 for ParentFoo and 2 for Foo |
| 339 | |
Sapan Bhatia | d022aeb | 2017-06-07 15:49:55 +0200 | [diff] [blame] | 340 | if __name__ == '__main__': |
| 341 | unittest.main() |
| 342 | |
| 343 | |