mangono-addon-export_json 18.0.1.0.4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,821 @@
1
+ import json
2
+ import re
3
+ from unittest.mock import MagicMock, patch
4
+
5
+ from odoo.tests import common, tagged
6
+
7
+ from odoo.addons.export_json.controller import main as controller_module
8
+ from odoo.addons.export_json.controller.main import JsonExportFormat
9
+
10
+
11
+ @tagged("export_json")
12
+ class TestExportJson(common.HttpCase):
13
+ def setUp(self):
14
+ self.partner = self.env["res.partner"].create(
15
+ {
16
+ "name": "TEST",
17
+ "email": "test@test.fr",
18
+ "type": "contact",
19
+ "child_ids": [
20
+ (
21
+ 0,
22
+ None,
23
+ {
24
+ "name": "test child 1",
25
+ "email": "test_child_1@test.fr",
26
+ "type": "delivery",
27
+ },
28
+ ),
29
+ (
30
+ 0,
31
+ None,
32
+ {
33
+ "name": "test child 2",
34
+ "email": "test_child_2@test.fr",
35
+ "type": "invoice",
36
+ },
37
+ ),
38
+ ],
39
+ }
40
+ )
41
+
42
+ self.data_dict = {
43
+ "model": "res.partner",
44
+ "fields": [
45
+ {"name": "id", "label": "External ID"},
46
+ {"name": "name", "label": "Name"},
47
+ {"name": "email", "label": "Email"},
48
+ {"name": "type", "label": "Type"},
49
+ {"name": "child_ids/name", "label": "Name"},
50
+ {"name": "child_ids/email", "label": "Email"},
51
+ {"name": "child_ids/type", "label": "Type"},
52
+ ],
53
+ "ids": [self.partner.id],
54
+ "domain": [["type", "in", ("contact", "delivery")]],
55
+ "groupby": [],
56
+ "context": {
57
+ "lang": "en_US",
58
+ "tz": "Europe/Paris",
59
+ "uid": 2,
60
+ "allowed_company_ids": [1],
61
+ },
62
+ "import_compat": True,
63
+ }
64
+ self.basic_expect_dict = {
65
+ "id": self.partner.id,
66
+ "name": "TEST",
67
+ "email": "test@test.fr",
68
+ "type": "contact",
69
+ "child_ids": [
70
+ {
71
+ "name": "test child 1",
72
+ "email": "test_child_1@test.fr",
73
+ "type": "delivery",
74
+ },
75
+ {
76
+ "name": "test child 2",
77
+ "email": "test_child_2@test.fr",
78
+ "type": "invoice",
79
+ },
80
+ ],
81
+ }
82
+
83
+ self.expected_list_dict_merge = [
84
+ {
85
+ "amount_total": 300.0,
86
+ "amount_untaxed": 250.0,
87
+ "company_id": "My Company",
88
+ "currency_id": "EUR",
89
+ "invoice_status": "invoice_status",
90
+ "invoice_status_FR": "Rien à facturer",
91
+ "invoice_status_US": "Nothing to Bill",
92
+ "order_line": [
93
+ {
94
+ "company_id": "My Company",
95
+ "currency_id": "EUR",
96
+ "display_type": "Section",
97
+ "name": "Exmple de section",
98
+ "state": "order_line/state",
99
+ "state_FR": "Bon de commande fournisseur",
100
+ "state_US": "Purchase Order",
101
+ "tax_calculation_rounding_method": "order_line/tax_calculation_rounding_method",
102
+ "tax_calculation_rounding_method_FR": "Arrondir à la ligne",
103
+ "tax_calculation_rounding_method_US": "Round per Line",
104
+ "taxes_id": [],
105
+ },
106
+ {
107
+ "company_id": "My Company",
108
+ "currency_id": "EUR",
109
+ "name": "Produit test 2",
110
+ "price_subtotal": 75.0,
111
+ "price_unit": 75.0,
112
+ "product_id": "order_line/product_id",
113
+ "product_id_FR": "Produit test 2",
114
+ "product_id_US": "ENGLISH Produit test 2",
115
+ "product_type": "order_line/product_type",
116
+ "product_type_FR": "Biens",
117
+ "product_type_US": "Goods",
118
+ "product_uom": "order_line/product_uom",
119
+ "product_uom_FR": "Unité(s)",
120
+ "product_uom_US": "Units",
121
+ "product_uom_category_id": "order_line/product_uom_category_id",
122
+ "product_uom_category_id_FR": "Unité",
123
+ "product_uom_category_id_US": "Unit",
124
+ "state": "order_line/state",
125
+ "state_FR": "Bon de commande fournisseur",
126
+ "state_US": "Purchase Order",
127
+ "tax_calculation_rounding_method": "order_line/tax_calculation_rounding_method",
128
+ "tax_calculation_rounding_method_FR": "Arrondir à la ligne",
129
+ "tax_calculation_rounding_method_US": "Round per Line",
130
+ "taxes_id": ["20%"],
131
+ },
132
+ ],
133
+ "payment_term_id": "payment_term_id",
134
+ "payment_term_id_FR": "21 jours ",
135
+ "payment_term_id_US": "21 Days",
136
+ "tax_country_id": "tax_country_id",
137
+ "tax_country_id_FR": "États-Unis",
138
+ "tax_country_id_US": "United States",
139
+ }
140
+ ]
141
+
142
+ self.data_str = str(self.data_dict)
143
+ self.data = json.dumps(self.data_dict)
144
+
145
+ self.json_export_format = JsonExportFormat()
146
+ self.mock_request = MagicMock()
147
+ self.mock_request.env = self.env
148
+ super().setUp()
149
+
150
+ def helper_get_csrf_token(self):
151
+ resp = self.url_open("/web/login")
152
+ html = resp.content.decode()
153
+ match = re.search(r'csrf_token:\s*"([^"]+)"', html)
154
+ return match and match.group(1) or False
155
+
156
+ def test_generate_export_json(self):
157
+ # Perform the json export
158
+ fields = [field["name"] for field in self.data_dict["fields"]]
159
+ model = self.env[self.data_dict["model"]]
160
+
161
+ with patch.object(controller_module, "request", self.mock_request):
162
+ result_json = self.json_export_format.perform_json_export(
163
+ self.data_dict["domain"], fields, self.data_dict["ids"], model
164
+ )
165
+
166
+ dict_result_json = json.loads(result_json)
167
+ self.assertDictEqual(
168
+ self.basic_expect_dict,
169
+ dict_result_json[0],
170
+ "Check if json export is equal to expected data",
171
+ )
172
+
173
+ def test_perform_json_export_with_lang(self):
174
+ self.env["res.lang"]._activate_lang("fr_FR")
175
+
176
+ self.data_dict["fields"].append({"name": "title/name", "label": "Title"})
177
+ self.data_dict["fields"].append({"name": "country_id/name", "label": "Country"})
178
+ fields = [field["name"] for field in self.data_dict["fields"]]
179
+ model = self.env[self.data_dict["model"]]
180
+
181
+ # Ajouts de traductions
182
+ title = (
183
+ self.env["res.partner.title"]
184
+ .with_context(lang="en_US")
185
+ .create(
186
+ {
187
+ "name": "Doctor",
188
+ }
189
+ )
190
+ )
191
+ title.with_context(lang="fr_FR").name = "Docteur"
192
+ self.partner.title = title
193
+
194
+ self.env.ref("base.es").with_context(lang="fr_FR").name = "Espagne"
195
+ self.partner.country_id = self.env.ref("base.es")
196
+
197
+ with patch.object(controller_module, "request", self.mock_request):
198
+ result_json = self.json_export_format.perform_json_export(
199
+ self.data_dict["domain"],
200
+ fields,
201
+ self.data_dict["ids"],
202
+ model,
203
+ langs_code=["fr_FR", "en_US"],
204
+ )
205
+
206
+ dict_result_json = json.loads(result_json)
207
+ expected_dict = {
208
+ "id": self.partner.id,
209
+ "name": "TEST",
210
+ "email": "test@test.fr",
211
+ "type": "contact",
212
+ "title": {
213
+ "name": "title/name",
214
+ "name_FR": "Docteur",
215
+ "name_US": "Doctor",
216
+ },
217
+ "country_id": {
218
+ "name": "country_id/name",
219
+ "name_FR": "Espagne",
220
+ "name_US": "Spain",
221
+ },
222
+ "child_ids": [
223
+ {
224
+ "name": "test child 1",
225
+ "email": "test_child_1@test.fr",
226
+ "type": "delivery",
227
+ },
228
+ {
229
+ "name": "test child 2",
230
+ "email": "test_child_2@test.fr",
231
+ "type": "invoice",
232
+ },
233
+ ],
234
+ }
235
+ self.assertDictEqual(expected_dict, dict_result_json[0], "Le json doit contenir les traductions")
236
+
237
+ def test_convert_simple_list(self):
238
+ """La fonction _convert_simple_list prend en entrée une liste de clée, correspondant
239
+ à des noms de champs, et retourne une liste de dict{"name": nom_de_champ}"""
240
+ res_partner_fields_names = list(self.partner._fields.keys())
241
+
242
+ res = self.json_export_format._convert_simple_list(res_partner_fields_names)
243
+
244
+ self.assertEqual(
245
+ len(res_partner_fields_names),
246
+ len(res),
247
+ "Tous les noms de champs du modèle res.partner doivent être retourné sous la forme {'name': nom_de_champ}",
248
+ )
249
+
250
+ # Si on ajoute un dict dans les noms de champ du res_partner, le dict doit être retourné tel quel, lors
251
+ # du passage de la fonction _convert_simple_list
252
+
253
+ mock_dict = {
254
+ "random_key": "random_value",
255
+ "random_key1": "random_value1",
256
+ "random_key2": "random_value2",
257
+ }
258
+ res_partner_fields_names.append(mock_dict)
259
+
260
+ res = self.json_export_format._convert_simple_list(res_partner_fields_names)
261
+
262
+ self.assertEqual(len(res_partner_fields_names), len(res))
263
+ self.assertIn(
264
+ mock_dict,
265
+ res,
266
+ "Le dict ajouté dans les champs de res.partner est bien retourner telquel dans la list",
267
+ )
268
+
269
+ def test_get_country_code(self):
270
+ """La fonction prend un paramètre un string de lang, elle doit retourner la deuxième partie du code
271
+ (séparé par le premier '_' du string)"""
272
+
273
+ for test_val, expected in [
274
+ ("fr_FR", "FR"),
275
+ ("_fr_FR", "fr"),
276
+ ("FR_", ""),
277
+ ("", ""),
278
+ ]:
279
+ self.assertEqual(self.json_export_format.get_country_code(test_val), expected)
280
+
281
+ def test_add_unique_vals(self):
282
+ """La fonction prend en paramètre un dict, et retourne un set de valeur unique.
283
+ La fonction stringify les list et les dicts, si dans le dict d'entrée, des valeurs sont
284
+ des dicts ou des lists."""
285
+
286
+ mock_dict_same_value = {
287
+ "random_key": "random_value",
288
+ "random_key1": "random_value",
289
+ "random_key2": "random_value",
290
+ }
291
+ res = self.json_export_format.add_unique_vals(mock_dict_same_value)
292
+
293
+ self.assertEqual(
294
+ len(res),
295
+ 1,
296
+ "Le dict d'entrée contenait 3 clés valeurs, mais les valeurs liées aux clés étaient identiques.",
297
+ )
298
+ self.assertIn("random_value", res)
299
+
300
+ mock_dict_different_value = {
301
+ "random_key": "random_value",
302
+ "random_key1": "random_value1",
303
+ "random_key2": "random_value",
304
+ }
305
+ res = self.json_export_format.add_unique_vals(mock_dict_different_value)
306
+ self.assertEqual(len(res), 2)
307
+ for expected in ["random_value", "random_value1"]:
308
+ self.assertIn(expected, res)
309
+
310
+ mock_dict_with_dict_list = {
311
+ "random_key": {"an_other": "one"},
312
+ "random_key1": [10, 20, 40],
313
+ "random_key2": [10, 20, 40],
314
+ }
315
+ res = self.json_export_format.add_unique_vals(mock_dict_with_dict_list)
316
+
317
+ self.assertEqual(len(res), 2)
318
+ for expected in ["{'an_other': 'one'}", "[10, 20, 40]"]:
319
+ self.assertIn(expected, res)
320
+
321
+ self.assertEqual(0, len(self.json_export_format.add_unique_vals({})))
322
+
323
+ def test_retrieve_all_keys(self):
324
+ test_dict = {
325
+ "fr_FR": {
326
+ "champ_1": "valeur",
327
+ "champ_2": "valeur",
328
+ "champ_3": "valeur",
329
+ "champ_4": "valeur",
330
+ },
331
+ "zn_CN": {
332
+ "champ_5": "valeur",
333
+ "champ_6": "valeur",
334
+ "champ_3": "valeur",
335
+ "champ_4": "valeur",
336
+ },
337
+ }
338
+
339
+ res = self.json_export_format.retrieve_all_keys(test_dict)
340
+ for expected in [
341
+ "champ_1",
342
+ "champ_2",
343
+ "champ_3",
344
+ "champ_4",
345
+ "champ_5",
346
+ "champ_6",
347
+ ]:
348
+ self.assertIn(expected, res)
349
+
350
+ other_dict = {
351
+ "champ_1": "valeur",
352
+ "champ_2": "valeur",
353
+ "champ_3": "valeur",
354
+ }
355
+ with self.assertRaises(AssertionError):
356
+ self.json_export_format.retrieve_all_keys(other_dict)
357
+
358
+ def test_process_keys_without_parent_key(self):
359
+ test_dict = {
360
+ "fr_FR": {
361
+ "champ_1": "valeur",
362
+ "champ_2": "valeur",
363
+ "champ_3": "valeur",
364
+ "champ_4": "valeur",
365
+ },
366
+ "zn_CN": {
367
+ "champ_5": "valeur",
368
+ "champ_6": "valeur",
369
+ "champ_3": "valeur",
370
+ "champ_4": "valeur",
371
+ },
372
+ }
373
+
374
+ merged_dict = {}
375
+ all_keys = self.json_export_format.retrieve_all_keys(test_dict)
376
+
377
+ self.json_export_format.process_keys(all_keys, test_dict, merged_dict)
378
+
379
+ for expected in [
380
+ "champ_1",
381
+ "champ_2",
382
+ "champ_3",
383
+ "champ_4",
384
+ "champ_5",
385
+ "champ_6",
386
+ ]:
387
+ self.assertIn(
388
+ expected,
389
+ merged_dict,
390
+ "Chaque champ doit être inclus dans le dictionnaire final, qui fusionne"
391
+ "l'ensemble des champs des dictionnaires d'entrées, par langue.",
392
+ )
393
+
394
+ self.json_export_format.process_keys(all_keys, test_dict, merged_dict)
395
+
396
+ # Si pour un même nom de champ donné, la valeur est différente entre des langues, les deux clés concaténées
397
+ # du suffixe du code la langue,
398
+ # doivent être présent dans le dict final.
399
+ test_dict_2 = {
400
+ "fr_FR": {
401
+ "champ_1": "valeur",
402
+ "champ_2": "valeur",
403
+ "champ_3": "VALEUR EN FRANÇAIS",
404
+ "champ_4": "valeur",
405
+ },
406
+ "zn_CN": {
407
+ "champ_5": "valeur",
408
+ "champ_6": "valeur",
409
+ "champ_3": "VALEUR PAS EN FRANÇAIS",
410
+ "champ_4": "valeur",
411
+ },
412
+ }
413
+ self.json_export_format.process_keys(all_keys, test_dict_2, merged_dict)
414
+ for expected in [
415
+ "champ_1",
416
+ "champ_2",
417
+ "champ_3",
418
+ "champ_4",
419
+ "champ_5",
420
+ "champ_6",
421
+ "champ_3_FR",
422
+ "champ_3_CN",
423
+ ]:
424
+ self.assertIn(
425
+ expected,
426
+ merged_dict,
427
+ "Chaque champ doit être inclus dans le dictionnaire final, qui fusionne"
428
+ "l'ensemble des champs des dictionnaires d'entrées, par langue.",
429
+ )
430
+ self.assertEqual(merged_dict.get("champ_3_FR"), "VALEUR EN FRANÇAIS")
431
+ self.assertEqual(merged_dict.get("champ_3_CN"), "VALEUR PAS EN FRANÇAIS")
432
+
433
+ def test_process_keys_with_parent_key(self):
434
+ test_dict = {
435
+ "fr_FR": {
436
+ "champ_1": "valeur",
437
+ "champ_2": "valeur",
438
+ "champ_3": {
439
+ "cle_dict_imbrique_1": "valeur1",
440
+ "cle_dict_imbrique_2": "valeur2",
441
+ "cle_dict_imbrique_3": [
442
+ "element_liste_imbriqueFR",
443
+ "element_autre_listeFR",
444
+ ],
445
+ },
446
+ },
447
+ "zn_CN": {
448
+ "champ_1": "valeur",
449
+ "champ_2": "valeur",
450
+ "champ_3": {
451
+ "cle_dict_imbrique_1": "valeur1489",
452
+ "cle_dict_imbrique_2": "valeur24996",
453
+ "cle_dict_imbrique_3": [
454
+ "element_liste_imbrique",
455
+ "element_autre_liste",
456
+ ],
457
+ },
458
+ },
459
+ }
460
+ merged_dict = {}
461
+ all_keys = self.json_export_format.retrieve_all_keys(test_dict)
462
+
463
+ self.json_export_format.process_keys(all_keys, test_dict, merged_dict, "champ_3")
464
+
465
+ # Le dict attendu doit contenir toutes les valeurs de chaque langues, même si les valeurs sont eux mêmes dans
466
+ # des dict ou dans des listes. Un regroupement doit être effectué autour du champ_3 :
467
+ # le dict du champ_3 doit contenir la clé original de chaque dict (cle_dict_imbrique_1, cle_dict_imbrique_2,
468
+ # cle_dict_imbrique_3)
469
+ # avec pour valeurs le chemin absolu de la valeur (champ_3/cle_dict_imbrique_1, champ_3/cle_dict_imbrique_2,
470
+ # champ_3/cle_dict_imbrique_3)
471
+
472
+ expected_dict = {
473
+ "champ_1": "valeur",
474
+ "champ_2": "valeur",
475
+ "champ_3": {
476
+ "cle_dict_imbrique_1": "champ_3/cle_dict_imbrique_1",
477
+ "cle_dict_imbrique_1_CN": "valeur1489",
478
+ "cle_dict_imbrique_1_FR": "valeur1",
479
+ "cle_dict_imbrique_2": "champ_3/cle_dict_imbrique_2",
480
+ "cle_dict_imbrique_2_CN": "valeur24996",
481
+ "cle_dict_imbrique_2_FR": "valeur2",
482
+ "cle_dict_imbrique_3": "champ_3/cle_dict_imbrique_3",
483
+ "cle_dict_imbrique_3_CN": [
484
+ "element_liste_imbrique",
485
+ "element_autre_liste",
486
+ ],
487
+ "cle_dict_imbrique_3_FR": [
488
+ "element_liste_imbriqueFR",
489
+ "element_autre_listeFR",
490
+ ],
491
+ },
492
+ }
493
+ self.assertDictEqual(expected_dict, merged_dict)
494
+
495
+ def test_process_keys_nested_dict(self):
496
+ test_dict = {
497
+ "fr_FR": {
498
+ "champ_1": [
499
+ {
500
+ "cle_dict_imbrique_1": "valeur1",
501
+ "cle_dict_imbrique_2": "valeur2",
502
+ "cle_dict_imbrique_3": [
503
+ "element_liste_imbriqueFR",
504
+ "element_autre_listeFR",
505
+ ],
506
+ },
507
+ {
508
+ "cle_dict_imbrique_1": "valeur44",
509
+ "cle_dict_imbrique_2": "valeur4858",
510
+ "cle_dict_imbrique_3": [
511
+ "element_liste_imbriqueFR",
512
+ "element_autre_liste",
513
+ ],
514
+ },
515
+ ],
516
+ "champ_2": "valeur",
517
+ },
518
+ "zn_CN": {
519
+ "champ_1": [
520
+ {
521
+ "cle_dict_imbrique_1": "valeur1",
522
+ "cle_dict_imbrique_2": "valeur2",
523
+ "cle_dict_imbrique_3": [
524
+ "element_liste_imbriqueFR",
525
+ "element_autre_listeFR",
526
+ ],
527
+ },
528
+ {
529
+ "cle_dict_imbrique_1": "valeur44",
530
+ "cle_dict_imbrique_2": "valeur4858",
531
+ "cle_dict_imbrique_3": [
532
+ "element_liste_imbrique",
533
+ "element_autre_liste",
534
+ ],
535
+ },
536
+ ],
537
+ },
538
+ }
539
+ merged_dict = {}
540
+ all_keys = self.json_export_format.retrieve_all_keys(test_dict)
541
+ self.json_export_format.process_keys(all_keys, test_dict, merged_dict)
542
+
543
+ expected_dict = {
544
+ "champ_1": [
545
+ {
546
+ "cle_dict_imbrique_1": "valeur1",
547
+ "cle_dict_imbrique_2": "valeur2",
548
+ "cle_dict_imbrique_3": [
549
+ "element_liste_imbriqueFR",
550
+ "element_autre_listeFR",
551
+ ],
552
+ },
553
+ {
554
+ "cle_dict_imbrique_1": "valeur44",
555
+ "cle_dict_imbrique_2": "valeur4858",
556
+ "cle_dict_imbrique_3": "champ_1/cle_dict_imbrique_3",
557
+ "cle_dict_imbrique_3_CN": [
558
+ "element_liste_imbrique",
559
+ "element_autre_liste",
560
+ ],
561
+ "cle_dict_imbrique_3_FR": [
562
+ "element_liste_imbriqueFR",
563
+ "element_autre_liste",
564
+ ],
565
+ },
566
+ ],
567
+ "champ_2": "valeur",
568
+ }
569
+ self.assertDictEqual(expected_dict, merged_dict)
570
+
571
+ def test_merge_list_of_dicts_without_lang(self):
572
+ self.assertEqual(
573
+ [],
574
+ self.json_export_format._merge_list_of_dicts({}),
575
+ "Sans langue, cette fonction ne doit rien faire et retourner une liste vide",
576
+ )
577
+ self.assertEqual(
578
+ [],
579
+ self.json_export_format.merge_multilingual_dicts({}),
580
+ "Même chose, sans langue, pas de dict",
581
+ )
582
+
583
+ def test_merge_multilingual_dicts(self):
584
+ test_dict = {
585
+ "en_US": [
586
+ {
587
+ "amount_total": 300.0,
588
+ "amount_untaxed": 250.0,
589
+ "company_id": "My Company",
590
+ "currency_id": "EUR",
591
+ "invoice_status": "Nothing to Bill",
592
+ "order_line": [
593
+ {
594
+ "company_id": "My Company",
595
+ "currency_id": "EUR",
596
+ "display_type": "Section",
597
+ "name": "Exmple de section",
598
+ "state": "Purchase Order",
599
+ "tax_calculation_rounding_method": "Round per Line",
600
+ "taxes_id": [],
601
+ },
602
+ {
603
+ "company_id": "My Company",
604
+ "currency_id": "EUR",
605
+ "name": "Produit test 2",
606
+ "price_subtotal": 75.0,
607
+ "price_unit": 75.0,
608
+ "product_type": "Goods",
609
+ "product_id": "ENGLISH Produit test 2",
610
+ "product_uom": "Units",
611
+ "product_uom_category_id": "Unit",
612
+ "state": "Purchase Order",
613
+ "tax_calculation_rounding_method": "Round per Line",
614
+ "taxes_id": ["20%"],
615
+ },
616
+ ],
617
+ "payment_term_id": "21 Days",
618
+ "tax_country_id": "United States",
619
+ }
620
+ ],
621
+ "fr_FR": [
622
+ {
623
+ "amount_total": 300.0,
624
+ "amount_untaxed": 250.0,
625
+ "company_id": "My Company",
626
+ "currency_id": "EUR",
627
+ "invoice_status": "Rien à facturer",
628
+ "order_line": [
629
+ {
630
+ "company_id": "My Company",
631
+ "currency_id": "EUR",
632
+ "display_type": "Section",
633
+ "name": "Exmple de section",
634
+ "state": "Bon de commande fournisseur",
635
+ "tax_calculation_rounding_method": "Arrondir à la ligne",
636
+ "taxes_id": [],
637
+ },
638
+ {
639
+ "company_id": "My Company",
640
+ "currency_id": "EUR",
641
+ "name": "Produit test 2",
642
+ "price_subtotal": 75.0,
643
+ "price_unit": 75.0,
644
+ "product_id": "Produit test 2",
645
+ "product_type": "Biens",
646
+ "product_uom": "Unité(s)",
647
+ "product_uom_category_id": "Unité",
648
+ "state": "Bon de commande fournisseur",
649
+ "tax_calculation_rounding_method": "Arrondir à la ligne",
650
+ "taxes_id": ["20%"],
651
+ },
652
+ ],
653
+ "payment_term_id": "21 jours ",
654
+ "tax_country_id": "États-Unis",
655
+ }
656
+ ],
657
+ }
658
+ self.assertDictEqual(
659
+ self.expected_list_dict_merge[0],
660
+ self.json_export_format.merge_multilingual_dicts(test_dict)[0],
661
+ )
662
+
663
+ def test_format_data(self):
664
+ fields = self.expected_list_dict_merge
665
+ res_json = self.json_export_format.format_data(fields)
666
+ self.assertTrue(json.loads(res_json), "Le résultat de la fonction doit être au format .json")
667
+
668
+ fields[0].update({"mock_bytes_key": b"mock_bytes_value"})
669
+ res_json = self.json_export_format.format_data(fields)
670
+
671
+ list_dict_from_json = json.loads(res_json)
672
+ bytes_key = list_dict_from_json[0].get("mock_bytes_key")
673
+ self.assertEqual(
674
+ "data:image/png;base64,mock_bytes_value",
675
+ bytes_key,
676
+ "La présence d'une clé contenant une valeur en bytes, dans les champs d'entrée, doit obligatoirement"
677
+ "être modifié en sorti dans le json, pour y inclure un préfixe.",
678
+ )
679
+
680
+ def test_define_parser(self):
681
+ res_partner_fields_names = list(self.partner._fields.keys())
682
+ set_fields_names = set(res_partner_fields_names)
683
+
684
+ # Ajouts de faux nom de champs qui sont concaténés.
685
+ res_partner_fields_names.append("company_id/name")
686
+ res_partner_fields_names.append("company_id/street1")
687
+ res_partner_fields_names.append("company_id/siret")
688
+
689
+ res_partner_fields_names.append("country_id/currency_id/name")
690
+ res_partner_fields_names.append("country_id/currency_id/iso_numeric")
691
+ converted_fields_name = self.json_export_format._convert_simple_list(res_partner_fields_names)
692
+
693
+ parser = self.json_export_format.define_parser(converted_fields_name)
694
+
695
+ self.assertTrue(
696
+ set_fields_names.issubset(parser),
697
+ "Tous les champs du modèle res.partner doivent être présent dans le parser.",
698
+ )
699
+ expected_tuple_1 = ("company_id", ["name", "street1", "siret"])
700
+ expected_tuple_2 = ("country_id", [("currency_id", ["name", "iso_numeric"])])
701
+ for expected_tuple in [expected_tuple_1, expected_tuple_2]:
702
+ self.assertIn(
703
+ expected_tuple,
704
+ parser,
705
+ "Un tuple nom_de_champ et liste de nom de champ (qui peut lui même contenir un tuple avec "
706
+ "la même combinaison), doit être présent, pour les champs de type M2O, avec un accès à un "
707
+ "des champs du M2O",
708
+ )
709
+
710
+ def test_add_list_or_dict_on_merge_item(self):
711
+ # Faire la version avec un parent_key qui est un dict
712
+ test_dict = {
713
+ "fr_FR": {
714
+ "champ_2": "valeur",
715
+ "champ_3": {
716
+ "cle_dict_imbrique_1": "valeur1",
717
+ "cle_dict_imbrique_2": "valeur2",
718
+ "cle_dict_imbrique_3": [
719
+ "element_liste_imbriqueFR",
720
+ "element_autre_listeFR",
721
+ ],
722
+ },
723
+ },
724
+ "zn_CN": {
725
+ "champ_2": "valeur",
726
+ "champ_3": {
727
+ "cle_dict_imbrique_1": "valeur1489",
728
+ "cle_dict_imbrique_2": "valeur24996",
729
+ "cle_dict_imbrique_3": [
730
+ "element_liste_imbrique",
731
+ "element_autre_liste",
732
+ ],
733
+ },
734
+ },
735
+ }
736
+ merged_dict = {}
737
+ all_keys = self.json_export_format.retrieve_all_keys(test_dict)
738
+
739
+ # On simule le fait que le champ_2 soit le parent key de tout le dict d'entrée "test_dict".
740
+ # La clé doit apparaître dans l'arborescence de chaque champ imbriquée.
741
+ self.json_export_format.process_keys(all_keys, test_dict, merged_dict, "champ_2")
742
+
743
+ expected_dict = {
744
+ "champ_2": "valeur",
745
+ "champ_3": {
746
+ "cle_dict_imbrique_2": "champ_2/champ_3/cle_dict_imbrique_2",
747
+ "cle_dict_imbrique_2_FR": "valeur2",
748
+ "cle_dict_imbrique_2_CN": "valeur24996",
749
+ "cle_dict_imbrique_3": "champ_2/champ_3/cle_dict_imbrique_3",
750
+ "cle_dict_imbrique_3_FR": [
751
+ "element_liste_imbriqueFR",
752
+ "element_autre_listeFR",
753
+ ],
754
+ "cle_dict_imbrique_3_CN": [
755
+ "element_liste_imbrique",
756
+ "element_autre_liste",
757
+ ],
758
+ "cle_dict_imbrique_1": "champ_2/champ_3/cle_dict_imbrique_1",
759
+ "cle_dict_imbrique_1_FR": "valeur1",
760
+ "cle_dict_imbrique_1_CN": "valeur1489",
761
+ },
762
+ }
763
+ self.assertDictEqual(expected_dict, merged_dict)
764
+
765
+ def test_index_json_export_route(self):
766
+ # Fonction appelé par le controller qui gère l'export au format json (bouton "Action" > "Exporter")
767
+ self.authenticate("admin", "admin")
768
+ csrf = self.helper_get_csrf_token()
769
+
770
+ url = "/web/export/json"
771
+ payload = {
772
+ "csrf_token": csrf,
773
+ "data": json.dumps(self.data_dict),
774
+ }
775
+
776
+ response = self.url_open(url, payload)
777
+ self.assertEqual(response.status_code, 200)
778
+ self.assertDictEqual(self.basic_expect_dict, response.json()[0])
779
+
780
+ def test_formats_export_formats(self):
781
+ # On vérifie que la pop-up d'export ajoute bien la coche json dans les options de format de sortie.
782
+ self.authenticate("admin", "admin")
783
+ url = "/web/export/formats"
784
+ payload = {"jsonrpc": "2.0", "method": "call", "params": {}, "id": 1}
785
+
786
+ response = self.url_open(
787
+ url,
788
+ data=json.dumps(payload),
789
+ headers={"Content-Type": "application/json"},
790
+ )
791
+ result_format = response.json().get("result")
792
+ self.assertIn(
793
+ {"label": "JSON", "tag": "json"},
794
+ result_format,
795
+ "Le nouveau format json doit être disponible enexport",
796
+ )
797
+
798
+ def test_json_export_on_none_ordinary_table(self):
799
+ fields = [field["name"] for field in self.data_dict["fields"]]
800
+ model = self.env[self.data_dict["model"]]
801
+ with patch("odoo.models.BaseModel._is_an_ordinary_table", return_value=False):
802
+ with patch.object(controller_module, "request", self.mock_request):
803
+ result_json = self.json_export_format.perform_json_export(
804
+ self.data_dict["domain"], fields, self.data_dict["ids"], model
805
+ )
806
+
807
+ res_dict = json.loads(result_json)
808
+ self.basic_expect_dict.pop("id")
809
+ self.assertDictEqual(
810
+ res_dict[0],
811
+ self.basic_expect_dict,
812
+ "Seul l'id ne doit pas être récupérer, "
813
+ "si la table associé au modèle ne correspond pas à une table classique."
814
+ "Toutes les autres informations doivent être récupérés normalement.",
815
+ )
816
+
817
+ def test_property_extension(self):
818
+ self.assertEqual(self.json_export_format.extension, ".json")
819
+
820
+ def test_property_content_type(self):
821
+ self.assertEqual(self.json_export_format.content_type, "text/json;charset=utf8")