EGYADMIN commited on
Commit
bc4fbcf
·
verified ·
1 Parent(s): 8ee6404

Create supply_chain/local_content_calculator.py

Browse files
supply_chain/local_content_calculator.py ADDED
@@ -0,0 +1,554 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # الحصول على معلومات العمالة
2
+ labor = project_data.get("labor", [])
3
+
4
+ # الحصول على معلومات الخدمات
5
+ services = project_data.get("services", [])
6
+
7
+ # حساب نسبة المحتوى المحلي للمواد
8
+ materials_local_content = self._calculate_materials_local_content(materials, sector)
9
+
10
+ # حساب نسبة المحتوى المحلي للعمالة
11
+ labor_local_content = self._calculate_labor_local_content(labor)
12
+
13
+ # حساب نسبة المحتوى المحلي للخدمات
14
+ services_local_content = self._calculate_services_local_content(services, sector)
15
+
16
+ # حساب نسبة المحتوى المحلي الإجمالية
17
+ total_local_content = self._calculate_total_local_content(
18
+ materials_local_content,
19
+ labor_local_content,
20
+ services_local_content,
21
+ sector
22
+ )
23
+
24
+ # تحديد النسبة المستهدفة للمحتوى المحلي
25
+ target_local_content = self._get_target_local_content(sector)
26
+
27
+ # تقييم الامتثال
28
+ compliance_status = "مستوفي" if total_local_content >= target_local_content else "غير مستوفي"
29
+
30
+ # إعداد النتائج
31
+ results = {
32
+ "total_local_content": total_local_content,
33
+ "target_local_content": target_local_content,
34
+ "compliance_status": compliance_status,
35
+ "materials_local_content": materials_local_content,
36
+ "labor_local_content": labor_local_content,
37
+ "services_local_content": services_local_content,
38
+ "details": {
39
+ "materials": self._get_materials_details(materials),
40
+ "labor": self._get_labor_details(labor),
41
+ "services": self._get_services_details(services)
42
+ },
43
+ "recommendations": self._generate_recommendations(
44
+ total_local_content,
45
+ target_local_content,
46
+ materials_local_content,
47
+ labor_local_content,
48
+ services_local_content,
49
+ sector
50
+ )
51
+ }
52
+
53
+ logger.info(f"اكتمل حساب نسب المحتوى المحلي: إجمالي {total_local_content:.1f}%، مستهدف {target_local_content:.1f}%")
54
+ return results
55
+
56
+ except Exception as e:
57
+ logger.error(f"فشل في حساب نسب المحتوى المحلي: {str(e)}")
58
+ return {
59
+ "total_local_content": 0,
60
+ "target_local_content": 0,
61
+ "compliance_status": "غير مستوفي",
62
+ "materials_local_content": 0,
63
+ "labor_local_content": 0,
64
+ "services_local_content": 0,
65
+ "details": {},
66
+ "recommendations": [],
67
+ "error": str(e)
68
+ }
69
+
70
+ def _calculate_materials_local_content(self, materials: List[Dict[str, Any]], sector: str) -> float:
71
+ """
72
+ حساب نسبة المحتوى المحلي للمواد
73
+
74
+ المعاملات:
75
+ ----------
76
+ materials : List[Dict[str, Any]]
77
+ قائمة المواد
78
+ sector : str
79
+ القطاع
80
+
81
+ المخرجات:
82
+ --------
83
+ float
84
+ نسبة المحتوى المحلي للمواد
85
+ """
86
+ if not materials:
87
+ return 0.0
88
+
89
+ total_cost = 0.0
90
+ local_cost = 0.0
91
+
92
+ for material in materials:
93
+ cost = material.get("cost", 0)
94
+ local_percentage = material.get("local_percentage", 0)
95
+
96
+ total_cost += cost
97
+ local_cost += cost * (local_percentage / 100)
98
+
99
+ if total_cost == 0:
100
+ return 0.0
101
+
102
+ # تطبيق المعاملات حسب القطاع
103
+ sector_weight = self.calculation_rules.get("sector_weights", {}).get(sector, {}).get("materials", 1.0)
104
+
105
+ local_content = (local_cost / total_cost) * 100 * sector_weight
106
+ return min(100.0, local_content)
107
+
108
+ def _calculate_labor_local_content(self, labor: List[Dict[str, Any]]) -> float:
109
+ """
110
+ حساب نسبة المحتوى المحلي للعمالة
111
+
112
+ المعاملات:
113
+ ----------
114
+ labor : List[Dict[str, Any]]
115
+ قائمة العمالة
116
+
117
+ المخرجات:
118
+ --------
119
+ float
120
+ نسبة المحتوى المحلي للعمالة
121
+ """
122
+ if not labor:
123
+ return 0.0
124
+
125
+ total_cost = 0.0
126
+ local_cost = 0.0
127
+
128
+ for worker in labor:
129
+ cost = worker.get("cost", 0)
130
+ is_saudi = worker.get("is_saudi", False)
131
+
132
+ total_cost += cost
133
+ if is_saudi:
134
+ local_cost += cost
135
+
136
+ if total_cost == 0:
137
+ return 0.0
138
+
139
+ local_content = (local_cost / total_cost) * 100
140
+ return min(100.0, local_content)
141
+
142
+ def _calculate_services_local_content(self, services: List[Dict[str, Any]], sector: str) -> float:
143
+ """
144
+ حساب نسبة المحتوى المحلي للخدمات
145
+
146
+ المعاملات:
147
+ ----------
148
+ services : List[Dict[str, Any]]
149
+ قائمة الخدمات
150
+ sector : str
151
+ القطاع
152
+
153
+ المخرجات:
154
+ --------
155
+ float
156
+ نسبة المحتوى المحلي للخدمات
157
+ """
158
+ if not services:
159
+ return 0.0
160
+
161
+ total_cost = 0.0
162
+ local_cost = 0.0
163
+
164
+ for service in services:
165
+ cost = service.get("cost", 0)
166
+ local_percentage = service.get("local_percentage", 0)
167
+
168
+ total_cost += cost
169
+ local_cost += cost * (local_percentage / 100)
170
+
171
+ if total_cost == 0:
172
+ return 0.0
173
+
174
+ # تطبيق المعاملات حسب القطاع
175
+ sector_weight = self.calculation_rules.get("sector_weights", {}).get(sector, {}).get("services", 1.0)
176
+
177
+ local_content = (local_cost / total_cost) * 100 * sector_weight
178
+ return min(100.0, local_content)
179
+
180
+ def _calculate_total_local_content(self, materials_local_content: float,
181
+ labor_local_content: float,
182
+ services_local_content: float,
183
+ sector: str) -> float:
184
+ """
185
+ حساب نسبة المحتوى المحلي الإجمالية
186
+
187
+ المعاملات:
188
+ ----------
189
+ materials_local_content : float
190
+ نسبة المحتوى المحلي للمواد
191
+ labor_local_content : float
192
+ نسبة المحتوى المحلي للعمالة
193
+ services_local_content : float
194
+ نسبة المحتوى المحلي للخدمات
195
+ sector : str
196
+ القطاع
197
+
198
+ المخرجات:
199
+ --------
200
+ float
201
+ نسبة المحتوى المحلي الإجمالية
202
+ """
203
+ # الحصول على أوزان القطاع
204
+ sector_weights = self.calculation_rules.get("sector_weights", {}).get(sector, {})
205
+
206
+ materials_weight = sector_weights.get("materials_weight", 0.5)
207
+ labor_weight = sector_weights.get("labor_weight", 0.3)
208
+ services_weight = sector_weights.get("services_weight", 0.2)
209
+
210
+ # حساب المتوسط المرجح
211
+ total_local_content = (
212
+ materials_local_content * materials_weight +
213
+ labor_local_content * labor_weight +
214
+ services_local_content * services_weight
215
+ )
216
+
217
+ return min(100.0, total_local_content)
218
+
219
+ def _get_target_local_content(self, sector: str) -> float:
220
+ """
221
+ تحديد النسبة المستهدفة للمحتوى المحلي
222
+
223
+ المعاملات:
224
+ ----------
225
+ sector : str
226
+ القطاع
227
+
228
+ المخرجات:
229
+ --------
230
+ float
231
+ النسبة المستهدفة للمحتوى المحلي
232
+ """
233
+ # الحصول على النسبة المستهدفة حسب القطاع
234
+ sector_targets = self.calculation_rules.get("sector_targets", {})
235
+
236
+ target = sector_targets.get(sector, 0.0)
237
+ if target > 0:
238
+ return target
239
+
240
+ # إذا لم يكن هناك هدف محدد للقطاع، استخدم الهدف الافتراضي
241
+ return self.calculation_rules.get("default_target", 30.0)
242
+
243
+ def _get_materials_details(self, materials: List[Dict[str, Any]]) -> List[Dict[str, Any]]:
244
+ """
245
+ الحصول على تفاصيل المواد
246
+
247
+ المعاملات:
248
+ ----------
249
+ materials : List[Dict[str, Any]]
250
+ قائمة المواد
251
+
252
+ المخرجات:
253
+ --------
254
+ List[Dict[str, Any]]
255
+ تفاصيل المواد
256
+ """
257
+ details = []
258
+
259
+ for material in materials:
260
+ details.append({
261
+ "name": material.get("name", ""),
262
+ "cost": material.get("cost", 0),
263
+ "local_percentage": material.get("local_percentage", 0),
264
+ "local_amount": material.get("cost", 0) * material.get("local_percentage", 0) / 100,
265
+ "manufacturer": material.get("manufacturer", ""),
266
+ "country_of_origin": material.get("country_of_origin", "")
267
+ })
268
+
269
+ return details
270
+
271
+ def _get_labor_details(self, labor: List[Dict[str, Any]]) -> List[Dict[str, Any]]:
272
+ """
273
+ الحصول على تفاصيل العمالة
274
+
275
+ المعاملات:
276
+ ----------
277
+ labor : List[Dict[str, Any]]
278
+ قائمة العمالة
279
+
280
+ المخرجات:
281
+ --------
282
+ List[Dict[str, Any]]
283
+ تفاصيل العمالة
284
+ """
285
+ details = []
286
+
287
+ for worker in labor:
288
+ details.append({
289
+ "position": worker.get("position", ""),
290
+ "cost": worker.get("cost", 0),
291
+ "is_saudi": worker.get("is_saudi", False),
292
+ "local_amount": worker.get("cost", 0) if worker.get("is_saudi", False) else 0,
293
+ "qualification": worker.get("qualification", ""),
294
+ "experience": worker.get("experience", "")
295
+ })
296
+
297
+ return details
298
+
299
+ def _get_services_details(self, services: List[Dict[str, Any]]) -> List[Dict[str, Any]]:
300
+ """
301
+ الحصول على تفاصيل الخدمات
302
+
303
+ المعاملات:
304
+ ----------
305
+ services : List[Dict[str, Any]]
306
+ قائمة الخدمات
307
+
308
+ المخرجات:
309
+ --------
310
+ List[Dict[str, Any]]
311
+ تفاصيل الخدمات
312
+ """
313
+ details = []
314
+
315
+ for service in services:
316
+ details.append({
317
+ "name": service.get("name", ""),
318
+ "cost": service.get("cost", 0),
319
+ "local_percentage": service.get("local_percentage", 0),
320
+ "local_amount": service.get("cost", 0) * service.get("local_percentage", 0) / 100,
321
+ "provider": service.get("provider", ""),
322
+ "description": service.get("description", "")
323
+ })
324
+
325
+ return details
326
+
327
+ def _generate_recommendations(self, total_local_content: float,
328
+ target_local_content: float,
329
+ materials_local_content: float,
330
+ labor_local_content: float,
331
+ services_local_content: float,
332
+ sector: str) -> List[str]:
333
+ """
334
+ إنشاء توصيات لتحسين نسبة المحتوى المحلي
335
+
336
+ المعاملات:
337
+ ----------
338
+ total_local_content : float
339
+ نسبة المحتوى المحلي الإجمالية
340
+ target_local_content : float
341
+ النسبة المستهدفة للمحتوى المحلي
342
+ materials_local_content : float
343
+ نسبة المحتوى المحلي للمواد
344
+ labor_local_content : float
345
+ نسبة المحتوى المحلي للعمالة
346
+ services_local_content : float
347
+ نسبة المحتوى المحلي للخدمات
348
+ sector : str
349
+ القطاع
350
+
351
+ المخرجات:
352
+ --------
353
+ List[str]
354
+ قائمة التوصيات
355
+ """
356
+ recommendations = []
357
+
358
+ # التحقق مما إذا كانت النسبة الإجمالية تلبي الهدف
359
+ if total_local_content < target_local_content:
360
+ gap = target_local_content - total_local_content
361
+ recommendations.append(f"زيادة نسبة المحتوى المحلي الإجمالية بمقدار {gap:.1f}% لتلبية الهدف المطلوب.")
362
+
363
+ # تحديد أي المكونات تحتاج إلى تحسين
364
+ sector_weights = self.calculation_rules.get("sector_weights", {}).get(sector, {})
365
+ materials_weight = sector_weights.get("materials_weight", 0.5)
366
+ labor_weight = sector_weights.get("labor_weight", 0.3)
367
+ services_weight = sector_weights.get("services_weight", 0.2)
368
+
369
+ # تقييم المكونات
370
+ component_gaps = [
371
+ ("المواد", materials_local_content, materials_weight),
372
+ ("العمالة", labor_local_content, labor_weight),
373
+ ("الخدمات", services_local_content, services_weight)
374
+ ]
375
+
376
+ # ترتيب المكونات حسب الفجوة المرجحة (أكبر فجوة في المساهمة)
377
+ component_gaps.sort(key=lambda x: (100 - x[1]) * x[2], reverse=True)
378
+
379
+ # إضافة توصيات محددة للمكونات ذات الأولوية
380
+ for component, content, weight in component_gaps:
381
+ if component == "المواد" and content < 70:
382
+ recommendations.append(f"زيادة نسبة المحتوى المحلي للمواد من {content:.1f}% إلى ما لا يقل عن 70% من خلال:")
383
+ recommendations.append(" - البحث عن موردين محليين للمواد الأساسية")
384
+ recommendations.append(" - استبدال المواد المستوردة ببدائل محلية الصنع")
385
+ recommendations.append(" - التعاون مع المصنعين المحليين لتطوير المنتجات المطلوبة")
386
+
387
+ elif component == "العمالة" and content < 60:
388
+ recommendations.append(f"زيادة نسبة المحتوى المحلي للعمالة من {content:.1f}% إلى ما لا يقل عن 60% من خلال:")
389
+ recommendations.append(" - توظيف المزيد من الكوادر السعودية")
390
+ recommendations.append(" - تدريب وتأهيل الكوادر الوطنية للوظائف الفنية")
391
+ recommendations.append(" - الاستفادة من برامج دعم التوظيف المقدمة من هدف وصندوق تنمية الموارد البشرية")
392
+
393
+ elif component == "الخدمات" and content < 50:
394
+ recommendations.append(f"زيادة نسبة المحتوى المحلي للخدمات من {content:.1f}% إلى ما لا يقل عن 50% من خلال:")
395
+ recommendations.append(" - التعاقد مع شركات خدمات محلية")
396
+ recommendations.append(" - الاستعانة بمكاتب استشارية سعودية")
397
+ recommendations.append(" - تطوير القدرات المحلية في مجالات الخدمات المتخصصة")
398
+
399
+ else:
400
+ recommendations.append(f"نسبة المحتوى المحلي الحالية ({total_local_content:.1f}%) تلبي الهدف المطلوب ({target_local_content:.1f}%).")
401
+ recommendations.append("للحفاظ على هذا المستوى وتحسينه:")
402
+ recommendations.append(" - توثيق مصادر المواد والخدمات المحلية بشكل دقيق")
403
+ recommendations.append(" - مراجعة خطة المحتوى المحلي بانتظام")
404
+ recommendations.append(" - بناء علاقات طويلة الأمد مع الموردين المحليين")
405
+
406
+ # توصيات عامة
407
+ recommendations.append("\nتوصيات عامة لتعزيز المحتوى المحلي:")
408
+ recommendations.append(" - الاستفادة من برامج ومبادرات هيئة المحتوى المحلي وتنمية القطاع الخاص")
409
+ recommendations.append(" - المشاركة في المعارض والفعاليات المحلية للتعرف على الموردين المحليين")
410
+ recommendations.append(" - الاستثمار في نقل التقنية وتوطينها")
411
+ recommendations.append(" - تطوير قاعدة بيانات للموردين المحليين وتحديثها بانتظام")
412
+
413
+ return recommendations
414
+
415
+ def _load_calculation_rules(self) -> Dict[str, Any]:
416
+ """
417
+ تحميل قواعد حساب المحتوى المحلي
418
+
419
+ المخرجات:
420
+ --------
421
+ Dict[str, Any]
422
+ قواعد حساب المحتوى المحلي
423
+ """
424
+ try:
425
+ file_path = 'data/templates/local_content_calculation_rules.json'
426
+ if os.path.exists(file_path):
427
+ with open(file_path, 'r', encoding='utf-8') as f:
428
+ return json.load(f)
429
+ else:
430
+ logger.warning(f"ملف قواعد حساب المحتوى المحلي غير موجود: {file_path}")
431
+ # إنشاء قواعد افتراضية
432
+ return self._create_default_calculation_rules()
433
+ except Exception as e:
434
+ logger.error(f"فشل في تحميل قواعد حساب المحتوى المحلي: {str(e)}")
435
+ return self._create_default_calculation_rules()
436
+
437
+ def _create_default_calculation_rules(self) -> Dict[str, Any]:
438
+ """
439
+ إنشاء قواعد حساب المحتوى المحلي الافتراضية
440
+
441
+ المخرجات:
442
+ --------
443
+ Dict[str, Any]
444
+ قواعد حساب المحتوى المحلي الافتراضية
445
+ """
446
+ return {
447
+ "default_target": 30.0,
448
+ "sector_targets": {
449
+ "oil_and_gas": 40.0,
450
+ "petrochemicals": 35.0,
451
+ "energy": 35.0,
452
+ "water": 25.0,
453
+ "construction": 20.0,
454
+ "infrastructure": 25.0,
455
+ "transportation": 30.0,
456
+ "telecommunications": 20.0,
457
+ "healthcare": 25.0,
458
+ "education": 35.0,
459
+ "tourism": 30.0,
460
+ "military": 50.0,
461
+ "industrial": 35.0,
462
+ "commercial": 25.0,
463
+ "residential": 20.0,
464
+ "general": 30.0
465
+ },
466
+ "sector_weights": {
467
+ "oil_and_gas": {
468
+ "materials_weight": 0.6,
469
+ "labor_weight": 0.25,
470
+ "services_weight": 0.15,
471
+ "materials": 1.1,
472
+ "services": 0.9
473
+ },
474
+ "construction": {
475
+ "materials_weight": 0.5,
476
+ "labor_weight": 0.35,
477
+ "services_weight": 0.15,
478
+ "materials": 1.0,
479
+ "services": 1.0
480
+ },
481
+ "infrastructure": {
482
+ "materials_weight": 0.55,
483
+ "labor_weight": 0.3,
484
+ "services_weight": 0.15,
485
+ "materials": 1.0,
486
+ "services": 1.0
487
+ },
488
+ "telecommunications": {
489
+ "materials_weight": 0.4,
490
+ "labor_weight": 0.3,
491
+ "services_weight": 0.3,
492
+ "materials": 0.9,
493
+ "services": 1.1
494
+ },
495
+ "healthcare": {
496
+ "materials_weight": 0.4,
497
+ "labor_weight": 0.4,
498
+ "services_weight": 0.2,
499
+ "materials": 0.9,
500
+ """
501
+ حاسبة نسب المحتوى المحلي
502
+ تقوم بحساب نسب المحتوى المحلي وفقًا للوائح هيئة المحتوى المحلي وتنمية القطاع الخاص
503
+ """
504
+
505
+ import logging
506
+ import json
507
+ import os
508
+ from typing import Dict, List, Any, Tuple, Optional, Union
509
+
510
+ logger = logging.getLogger(__name__)
511
+
512
+ class LocalContentCalculator:
513
+ """
514
+ حاسبة نسب المحتوى المحلي
515
+ """
516
+
517
+ def __init__(self, config=None):
518
+ """
519
+ تهيئة حاسبة المحتوى المحلي
520
+
521
+ المعاملات:
522
+ ----------
523
+ config : Dict, optional
524
+ إعدادات الحاسبة
525
+ """
526
+ self.config = config or {}
527
+
528
+ # تحميل قواعد حساب المحتوى المحلي
529
+ self.calculation_rules = self._load_calculation_rules()
530
+
531
+ logger.info("تم تهيئة حاسبة المحتوى المحلي")
532
+
533
+ def calculate(self, project_data: Dict[str, Any]) -> Dict[str, Any]:
534
+ """
535
+ حساب نسب المحتوى المحلي للمشروع
536
+
537
+ المعاملات:
538
+ ----------
539
+ project_data : Dict[str, Any]
540
+ بيانات المشروع
541
+
542
+ المخرجات:
543
+ --------
544
+ Dict[str, Any]
545
+ نتائج حساب المحتوى المحلي
546
+ """
547
+ try:
548
+ logger.info("بدء حساب نسب المحتوى المحلي")
549
+
550
+ # الحصول على معلومات القطاع
551
+ sector = project_data.get("sector", "general")
552
+
553
+ # الحصول على معلومات المواد
554
+ materials = project_data.get("materials", [])