edu-rdm-integration 0.8.6__py3-none-any.whl → 0.9.0__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.
@@ -36,7 +36,7 @@ class ReformatLogsMixin:
36
36
  operation = LOG_OPERATION_MAP[log.operation]
37
37
 
38
38
  if operation in EntityLogOperation.values:
39
- fields = log.data
39
+ fields = log.transformed_data
40
40
  else:
41
41
  fields = {}
42
42
 
@@ -131,9 +131,9 @@ class BaseCollectingExportedDataFunctionCacheStorage(WebEduFunctionCacheStorage)
131
131
  operation = LOG_OPERATION_MAP[log.operation]
132
132
 
133
133
  if operation in (EntityLogOperation.CREATE, EntityLogOperation.DELETE):
134
- fields = log.data
134
+ fields = log.transformed_data
135
135
  elif operation == EntityLogOperation.UPDATE:
136
- fields = log.changes
136
+ fields = log.transformed_changes
137
137
  else:
138
138
  fields = {}
139
139
 
@@ -44,6 +44,9 @@ from django.utils.datastructures import (
44
44
  from educommon import (
45
45
  logger,
46
46
  )
47
+ from educommon.async_task.models import (
48
+ RunningTask,
49
+ )
47
50
  from educommon.utils.date import (
48
51
  get_today_max_datetime,
49
52
  )
@@ -126,12 +129,11 @@ class BaseExportEntitiesData(BaseOperationData):
126
129
  #TODO Вынужденная мера, т.к. при запуске команды не производится проверка готовности конфигов приложений.
127
130
  # Нужно переработать механизм конфигурирования клиента загрузчика.
128
131
  """
132
+ import uploader_client
129
133
  from django.core.cache import (
130
134
  DEFAULT_CACHE_ALIAS,
131
135
  caches,
132
136
  )
133
-
134
- import uploader_client
135
137
  from uploader_client.contrib.rdm.interfaces.configurations import (
136
138
  RegionalDataMartUploaderConfig,
137
139
  )
@@ -290,7 +292,7 @@ class BaseExportLatestEntitiesData(BaseExportEntitiesData):
290
292
  """Добавляет в описание асинхронной задачи список выгруженных сущностей."""
291
293
  if exported_entities and self.task_id:
292
294
  self.async_task.objects.filter(
293
- task_id=self.task_id,
295
+ pk=self.task_id,
294
296
  ).update(
295
297
  description=Concat(
296
298
  'description',
@@ -406,3 +408,11 @@ class BaseExportLatestEntitiesData(BaseExportEntitiesData):
406
408
 
407
409
  class ExportEntitiesData(BaseExportEntitiesData):
408
410
  """Экспорт сущностей РВД за указанных период."""
411
+
412
+
413
+ class ExportLatestEntitiesData(BaseExportLatestEntitiesData):
414
+ """Класс выгрузки сущностей с момента последней успешной выгрузки."""
415
+
416
+ def _get_async_task(self) -> Model:
417
+ """Возвращает модель асинхронной задачи."""
418
+ return RunningTask
@@ -0,0 +1,137 @@
1
+ from concurrent.futures import (
2
+ ThreadPoolExecutor,
3
+ )
4
+ from json import (
5
+ JSONDecodeError,
6
+ )
7
+ from typing import (
8
+ TYPE_CHECKING,
9
+ Any,
10
+ Dict,
11
+ Optional,
12
+ Tuple,
13
+ )
14
+
15
+ from django.db import (
16
+ transaction,
17
+ )
18
+ from django.db.models import (
19
+ QuerySet,
20
+ )
21
+ from uploader_client.adapters import (
22
+ adapter,
23
+ )
24
+
25
+ from educommon import (
26
+ logger,
27
+ )
28
+
29
+ from edu_rdm_integration.enums import (
30
+ FileUploadStatusEnum,
31
+ )
32
+ from edu_rdm_integration.export_data.base.requests import (
33
+ RegionalDataMartStatusRequest,
34
+ )
35
+ from edu_rdm_integration.models import (
36
+ DataMartRequestStatus,
37
+ ExportingDataSubStageUploaderClientLog,
38
+ UploadStatusRequestLog,
39
+ )
40
+
41
+
42
+ if TYPE_CHECKING:
43
+ from uploader_client.models import (
44
+ Entry,
45
+ )
46
+
47
+
48
+ class UploadStatusHelper:
49
+ """Хелпер проверки статуса загрузки данных в витрину."""
50
+
51
+ def __init__(self, in_progress_uploads: QuerySet) -> None:
52
+ self._in_progress_uploads = in_progress_uploads
53
+
54
+ def run(self, thread_count: int = 1) -> None:
55
+ """Запускает проверки статусов."""
56
+ if thread_count > 1:
57
+ with ThreadPoolExecutor(max_workers=thread_count) as pool:
58
+ pool.map(self._process_upload, self._in_progress_uploads)
59
+ else:
60
+ for upload in self._in_progress_uploads:
61
+ self._process_upload(upload)
62
+
63
+ @classmethod
64
+ def send_upload_status_request(cls, request_id: str) -> Tuple[Optional[Dict[str, Any]], 'Entry']:
65
+ """Формирует и отправляет запрос для получения статуса загрузки данных в витрину."""
66
+ request = RegionalDataMartStatusRequest(
67
+ request_id=request_id,
68
+ method='GET',
69
+ parameters={},
70
+ headers={
71
+ 'Content-Type': 'application/json',
72
+ },
73
+ )
74
+
75
+ result = adapter.send(request)
76
+
77
+ response = None
78
+
79
+ if result.error:
80
+ logger.error(
81
+ f'Ошибка при получении статуса загрузки данных в витрину. Идентификатор загрузки: {request_id}. '
82
+ f'Ошибка: {result.error}, запрос: {result.log.request}, ответ: {result.log.response}',
83
+ )
84
+ else:
85
+ logger.info(
86
+ f'Получен ответ со статусом {result.response.status_code} и содержимым {result.response.text}. '
87
+ f'Идентификатор загрузки: {request_id}',
88
+ )
89
+ try:
90
+ response = result.response.json()
91
+ except JSONDecodeError:
92
+ logger.error(
93
+ f'Не удалось получить данные из ответа запроса статуса загрузки данных в витрину. '
94
+ f'Идентификатор загрузки: {request_id}, ответ: {result.response.text}',
95
+ )
96
+
97
+ return response, result.log
98
+
99
+ @classmethod
100
+ def update_upload_status(
101
+ cls,
102
+ upload: ExportingDataSubStageUploaderClientLog,
103
+ response: Optional[Dict[str, Any]],
104
+ log_entry: 'Entry',
105
+ ) -> None:
106
+ """Обновляет статус загрузки данных в витрину."""
107
+ request_status = None
108
+
109
+ if isinstance(response, dict):
110
+ request_status = DataMartRequestStatus.get_values_to_enum_data().get(response.get('code'))
111
+
112
+ if not request_status:
113
+ logger.error(
114
+ 'Не удалось определить статус загрузки данных в витрину. Идентификатор загрузки: '
115
+ f'{upload.request_id}, данные ответа: {response}',
116
+ )
117
+
118
+ with transaction.atomic():
119
+ UploadStatusRequestLog.objects.create(
120
+ upload=upload,
121
+ entry=log_entry,
122
+ request_status_id=getattr(request_status, 'key', None),
123
+ )
124
+
125
+ if request_status in {DataMartRequestStatus.FAILED_PROCESSING, DataMartRequestStatus.REQUEST_ID_NOT_FOUND}:
126
+ upload.file_upload_status = FileUploadStatusEnum.ERROR
127
+
128
+ elif request_status == DataMartRequestStatus.SUCCESSFULLY_PROCESSED:
129
+ upload.file_upload_status = FileUploadStatusEnum.FINISHED
130
+
131
+ if upload.file_upload_status != FileUploadStatusEnum.IN_PROGRESS:
132
+ upload.save()
133
+
134
+ def _process_upload(self, upload: ExportingDataSubStageUploaderClientLog) -> None:
135
+ """Обрабатывает запись загрузки данных в витрину."""
136
+ response, log_entry = self.send_upload_status_request(upload.request_id)
137
+ self.update_upload_status(upload, response, log_entry)
@@ -0,0 +1,97 @@
1
+ """
2
+ Команда для проверки состояния отправленных в витрину данных.
3
+
4
+ Необязательные параметры:
5
+ --period_started_at - дата и время начала периода загрузки данных в витрину
6
+ --period_ended_at - дата и время конца периода загрузки данных в витрину
7
+ --thread_count - количество потоков для обработки
8
+
9
+ Пример использования:
10
+ python manage.py check_upload_status --period_started_at="01.11.2023 10:00:00" --thread_count=4
11
+ """
12
+ from datetime import (
13
+ date,
14
+ datetime,
15
+ time,
16
+ )
17
+ from typing import (
18
+ TYPE_CHECKING,
19
+ Any,
20
+ Dict,
21
+ Tuple,
22
+ )
23
+
24
+ from django.core.management.base import (
25
+ BaseCommand,
26
+ )
27
+
28
+ from edu_rdm_integration.consts import (
29
+ DATETIME_FORMAT,
30
+ )
31
+ from edu_rdm_integration.enums import (
32
+ FileUploadStatusEnum,
33
+ )
34
+ from edu_rdm_integration.helpers import UploadStatusHelper
35
+ from edu_rdm_integration.models import (
36
+ ExportingDataSubStageUploaderClientLog,
37
+ )
38
+
39
+
40
+ if TYPE_CHECKING:
41
+ from django.core.management.base import (
42
+ CommandParser,
43
+ )
44
+
45
+
46
+ class Command(BaseCommand):
47
+ """Команда для проверки состояния отправленных в витрину данных."""
48
+
49
+ help = 'Команда для проверки состояния отправленных в витрину данных' # noqa: A003
50
+
51
+ def add_arguments(self, parser: 'CommandParser') -> None:
52
+ """Добавляет аргументы парсера."""
53
+ parser.add_argument(
54
+ '--period_started_at',
55
+ action='store',
56
+ dest='period_started_at',
57
+ type=lambda started_at: datetime.strptime(started_at, DATETIME_FORMAT),
58
+ default=datetime.combine(date.today(), time.min),
59
+ help=(
60
+ 'Дата и время начала периода загрузки данных в витрину. Значение предоставляется в формате '
61
+ '"дд.мм.гггг чч:мм:сс". По умолчанию, сегодняшний день, время 00:00:00.'
62
+ ),
63
+ )
64
+
65
+ parser.add_argument(
66
+ '--period_ended_at',
67
+ action='store',
68
+ dest='period_ended_at',
69
+ type=lambda ended_at: datetime.strptime(ended_at, DATETIME_FORMAT),
70
+ default=datetime.combine(date.today(), time.max),
71
+ help=(
72
+ 'Дата и время конца периода загрузки данных в витрину. Значение предоставляется в формате '
73
+ '"дд.мм.гггг чч:мм:сс". По умолчанию, сегодняшний день, время 23:59:59.'
74
+ ),
75
+ )
76
+
77
+ parser.add_argument(
78
+ '--thread_count',
79
+ default=1,
80
+ type=int,
81
+ help='Количество потоков для обработки. По умолчанию, 1.',
82
+ )
83
+
84
+ def handle(self, *args: Tuple[Any], **kwargs: Dict[str, Any]) -> None:
85
+ """Обработчик команды."""
86
+ thread_count = kwargs['thread_count']
87
+ if thread_count < 1:
88
+ raise ValueError(f'Количество потоков {thread_count} должно быть больше 0.')
89
+
90
+ in_progress_attachment_uploads = ExportingDataSubStageUploaderClientLog.objects.filter(
91
+ created__gte=kwargs['period_started_at'],
92
+ created__lte=kwargs['period_ended_at'],
93
+ is_emulation=False,
94
+ file_upload_status=FileUploadStatusEnum.IN_PROGRESS,
95
+ )
96
+
97
+ UploadStatusHelper(in_progress_attachment_uploads).run(thread_count=thread_count)
@@ -0,0 +1,20 @@
1
+ from edu_rdm_integration.collect_data.collect import (
2
+ BaseCollectLatestModelsData,
3
+ BaseCollectModelsData,
4
+ )
5
+ from edu_rdm_integration.management.general import (
6
+ BaseCollectModelDataCommand,
7
+ )
8
+
9
+
10
+ class Command(BaseCollectModelDataCommand):
11
+ """
12
+ Команда для сбора на основе логов за период с последней сборки до указанной даты.
13
+ """
14
+
15
+ def _prepare_collect_models_data_class(self, *args, **kwargs) -> BaseCollectModelsData:
16
+ return BaseCollectLatestModelsData(
17
+ models=kwargs.get('models'),
18
+ logs_period_started_at=kwargs.get('logs_period_started_at'),
19
+ logs_period_ended_at=kwargs.get('logs_period_ended_at'),
20
+ )
@@ -0,0 +1,26 @@
1
+ from edu_rdm_integration.export_data.export import (
2
+ BaseExportEntitiesData,
3
+ ExportLatestEntitiesData,
4
+ )
5
+ from edu_rdm_integration.management.general import (
6
+ BaseExportEntityDataCommand,
7
+ )
8
+
9
+
10
+ class Command(BaseExportEntityDataCommand):
11
+ """
12
+ Команда для экспорта данных за период с последней сборки до указанной даты.
13
+ """
14
+
15
+ # flake8: noqa: A003
16
+ help = 'Команда для запуска функции экспорта данных для интеграции с "Региональная витрина данных"'
17
+
18
+ def _prepare_export_entities_data_class(self, *args, **kwargs) -> BaseExportEntitiesData:
19
+ """Возвращает объект класса экспорта данных сущностей РВД."""
20
+ return ExportLatestEntitiesData(
21
+ entities=kwargs.get('entities'),
22
+ period_started_at=kwargs.get('period_started_at'),
23
+ period_ended_at=kwargs.get('period_ended_at'),
24
+ task_id=kwargs.get('task_id'),
25
+ update_modified=kwargs.get('update_modified'),
26
+ )
@@ -0,0 +1,94 @@
1
+ from celery.schedules import (
2
+ crontab,
3
+ )
4
+ from django.conf import (
5
+ settings,
6
+ )
7
+
8
+ from educommon.async_task.models import (
9
+ AsyncTaskType,
10
+ )
11
+ from educommon.async_task.tasks import (
12
+ PeriodicAsyncTask,
13
+ )
14
+
15
+ from edu_rdm_integration.collect_data.helpers import (
16
+ set_failed_status_suspended_collecting_data_stages,
17
+ )
18
+ from edu_rdm_integration.consts import (
19
+ TASK_QUEUE_NAME,
20
+ )
21
+ from edu_rdm_integration.enums import (
22
+ FileUploadStatusEnum,
23
+ )
24
+ from edu_rdm_integration.export_data.helpers import (
25
+ set_failed_status_suspended_exporting_data_stages,
26
+ )
27
+ from edu_rdm_integration.helpers import (
28
+ UploadStatusHelper,
29
+ )
30
+ from edu_rdm_integration.models import (
31
+ ExportingDataSubStageUploaderClientLog,
32
+ )
33
+
34
+
35
+ class RDMCheckUploadStatus(PeriodicAsyncTask):
36
+ """Периодическая задача для сбора статусов по загрузке файла в витрину."""
37
+
38
+ queue = TASK_QUEUE_NAME
39
+ routing_key = TASK_QUEUE_NAME
40
+ description = 'Сбор статусов загрузки данных в витрину "Региональная витрина данных"'
41
+ task_type = AsyncTaskType.UNKNOWN
42
+ run_every = crontab(
43
+ minute=settings.RDM_UPLOAD_STATUS_TASK_MINUTE,
44
+ hour=settings.RDM_UPLOAD_STATUS_TASK_HOUR,
45
+ day_of_week=settings.RDM_UPLOAD_STATUS_TASK_DAY_OF_WEEK,
46
+ )
47
+
48
+ def process(self, *args, **kwargs):
49
+ """Выполнение."""
50
+ super().process(*args, **kwargs)
51
+
52
+ # Получаем незавершенные загрузки данных в витрину
53
+ in_progress_uploads = ExportingDataSubStageUploaderClientLog.objects.filter(
54
+ file_upload_status=FileUploadStatusEnum.IN_PROGRESS,
55
+ is_emulation=False,
56
+ )
57
+
58
+ UploadStatusHelper(in_progress_uploads).run()
59
+
60
+
61
+ class CheckSuspendedExportedStagePeriodicTask(PeriodicAsyncTask):
62
+ """Периодическая задача поиска зависших этапов/подэтапов экспорта."""
63
+
64
+ queue = TASK_QUEUE_NAME
65
+ routing_key = TASK_QUEUE_NAME
66
+ description = 'Поиск зависших этапов/подэтапов экспорта в "Региональная витрина данных"'
67
+ task_type = AsyncTaskType.SYSTEM
68
+ run_every = crontab(
69
+ minute=settings.RDM_CHECK_SUSPEND_TASK_MINUTE,
70
+ hour=settings.RDM_CHECK_SUSPEND_TASK_HOUR,
71
+ day_of_week=settings.RDM_CHECK_SUSPEND_TASK_DAY_OF_WEEK,
72
+ )
73
+
74
+ def process(self, *args, **kwargs):
75
+ """Выполнение задачи."""
76
+ super().process(*args, **kwargs)
77
+
78
+ change_status_collecting_result = set_failed_status_suspended_collecting_data_stages()
79
+ change_status_exporting_result = set_failed_status_suspended_exporting_data_stages()
80
+
81
+ task_result = {
82
+ 'Прервано сборок': (
83
+ f'Этапов {change_status_collecting_result["change_stage_count"]}'
84
+ f' и подэтапов {change_status_collecting_result["change_sub_stage_count"]}'
85
+ ),
86
+ 'Прервано выгрузок': (
87
+ f'Этапов {change_status_exporting_result["change_stage_count"]}'
88
+ f' и подэтапов {change_status_exporting_result["change_sub_stage_count"]}'
89
+ ),
90
+ }
91
+
92
+ self.set_progress(
93
+ values=task_result
94
+ )
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: edu-rdm-integration
3
- Version: 0.8.6
3
+ Version: 0.9.0
4
4
  Summary: Интеграция с Региональной витриной данных
5
5
  Home-page:
6
6
  Download-URL:
@@ -28,7 +28,8 @@ Requires-Dist: setuptools <69,>=47.3.1
28
28
  Requires-Dist: wheel <0.42,>=0.37.1
29
29
  Requires-Dist: transliterate <2
30
30
  Requires-Dist: Django <2.3,>=1.11
31
- Requires-Dist: educommon <4,>=3.6.0
31
+ Requires-Dist: celery ==4.4.7
32
+ Requires-Dist: educommon <4,>=3.7.0
32
33
  Requires-Dist: function-tools <1,>=0.8.0
33
34
  Requires-Dist: m3-db-utils <1,>=0.3.10
34
35
  Requires-Dist: uploader-client <1,>=0.2.1
@@ -289,6 +290,28 @@ Requires-Dist: uploader-client <1,>=0.2.1
289
290
 
290
291
  ### Удалено
291
292
 
293
+ ## [0.9.0] - 2023-12-29
294
+
295
+ Из ЭШ перенесены периодические задачи по сбору статусов загрузки файлов в витрину, а также
296
+ по поиску зависших этапов/подэтапов экспорта.
297
+
298
+ Также из ЭШ перенесены менедж-команды:
299
+ - check_upload_status - проверка состояния отправленных данных в витрину,
300
+ - collect_lastest_models_data - сбор на основе логов за период с последней сборки до указанной даты,
301
+ - export_latest_entities_data - экспорт данных за период с последней сборки до указанной даты.
302
+
303
+ Типы получаемых из log_change.fields полей соответствуют типам полей из логируемых моделей.
304
+
305
+ ### Добавлено
306
+
307
+ - [EDUSCHL-21013](https://jira.bars.group/browse/EDUSCHL-21013)
308
+ MINOR Перенесена часть асинхронных РВД задач из ЭШ, а также часть менедж-команд
309
+
310
+ ### Изменено
311
+
312
+ - [EDUSCHL-20793](https://jira.bars.group/browse/EDUSCHL-20793)
313
+ MINOR Типы получаемых из log_change.fields полей должны соответствовать типам полей из логируемых моделей
314
+
292
315
 
293
316
  ## [0.8.6] - 2023-12-18
294
317
 
@@ -5,10 +5,12 @@ edu_rdm_integration/base.py,sha256=_G0qPTAXe6bXfgDHNiZMSsYt3sMuUhLKnHuQCWSFttU,1
5
5
  edu_rdm_integration/consts.py,sha256=FFwcMHNsfjP_s9LfkccLAHjJMEMp7ppPmrRlJcgV88k,1104
6
6
  edu_rdm_integration/entities.py,sha256=mAjsYlcIbemo4xT5CSCr4payZubiBHB7Rb3Ow1CVsy0,14552
7
7
  edu_rdm_integration/enums.py,sha256=6Gv_hpYrC6v75ZtBA_xBrHqvza9NbJKhMa1TdTHkzys,4048
8
+ edu_rdm_integration/helpers.py,sha256=-UjORR2Pj_zW9gcAz-Dqs2Qi-s8TIptKVUBJVUBjRus,4958
8
9
  edu_rdm_integration/mapping.py,sha256=bwa2fJCbV4YjQcAgRrgT3hgM6dJhr_uBtQgx3L3F2Ck,473
9
10
  edu_rdm_integration/models.py,sha256=NgUR_1uPzKSaj013T1ofPCmA8rhf4DBAwl-liY1oWns,28630
10
11
  edu_rdm_integration/signals.py,sha256=3eRlpkDcFCF6TN80-QM8yBYLcyozzcmoPjz6r4_ApWg,73
11
12
  edu_rdm_integration/storages.py,sha256=o5WqUG7SnkeuMt-z8spUi-IraivST-7KHzfY-M3v7FA,6807
13
+ edu_rdm_integration/tasks.py,sha256=XzRhcfv6u2LSANvWTNvocdw0UJZDKLn3MzenViYrAVE,3421
12
14
  edu_rdm_integration/utils.py,sha256=vjme0N6tEXnHt6SaqjavZshjwc-mVv4X3Pz37a5YgTw,7092
13
15
  edu_rdm_integration/adapters/__init__.py,sha256=cU0swn4Ny5ZQz5buWRcWsT1mpWuUFJaUlHf2l7TtEBo,83
14
16
  edu_rdm_integration/adapters/apps.py,sha256=TyJTkSPs2qAHJ11fqbwLGk3Ea7ujtqWwbxqmvYNQxG8,363
@@ -37,7 +39,7 @@ edu_rdm_integration/collect_data/base/caches.py,sha256=3BaJxYBk9fi0aiAVzym-Jz8aN
37
39
  edu_rdm_integration/collect_data/base/functions.py,sha256=HT23EyiD-H50p4NLx2_LtioktTHHFVLRmAgWdbuHErw,2379
38
40
  edu_rdm_integration/collect_data/base/helpers.py,sha256=ONbPEELI_9ImpqCLrdIL3kb9aIQ2xQcF6XMa2C5KlDw,1017
39
41
  edu_rdm_integration/collect_data/base/managers.py,sha256=s9eUpVlsTw58mFuDgSBO_EKUk9nXIWUqc_Dhc54Yhfw,6812
40
- edu_rdm_integration/collect_data/base/mixins.py,sha256=Rp3ECccl3vOWZH2QHR_ArAbibQMq92IMGPoP5XAfgVc,3180
42
+ edu_rdm_integration/collect_data/base/mixins.py,sha256=-KpOg8yXeu4AEEJt26iruBEa8ErQd5XXNrUOuOL9VP8,3192
41
43
  edu_rdm_integration/collect_data/base/runners.py,sha256=OjhdTmMab6dCoIZp2zmXZTK4H5Haz__QHddwGxjOuB0,2829
42
44
  edu_rdm_integration/collect_data/calculated/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
43
45
  edu_rdm_integration/collect_data/calculated/strategies.py,sha256=W9f-MlLENSoa9jHMLWTP7ktZL-7FVtkt2FLXrbmds-o,7629
@@ -58,7 +60,7 @@ edu_rdm_integration/collect_data/calculated/base/validators.py,sha256=xIaSIE9oKb
58
60
  edu_rdm_integration/collect_data/non_calculated/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
59
61
  edu_rdm_integration/collect_data/non_calculated/strategies.py,sha256=eR-z94U0XN8BBuFS789PzxCPPfUoL3a2mcZb10Psqeg,7431
60
62
  edu_rdm_integration/collect_data/non_calculated/base/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
61
- edu_rdm_integration/collect_data/non_calculated/base/caches.py,sha256=qRC92vy-oUcyokAF279CcZgWTl_x-PxsGAK-5Rb31-g,6821
63
+ edu_rdm_integration/collect_data/non_calculated/base/caches.py,sha256=uGtNhqC-JVSJHYrY9JcFfdaUpe2t3F5jZSRodQBSzt4,6845
62
64
  edu_rdm_integration/collect_data/non_calculated/base/consts.py,sha256=pds1t4eHzovm7Yz2o5je3UHqRE8gqfT2sL-IwpoBN_o,66
63
65
  edu_rdm_integration/collect_data/non_calculated/base/enums.py,sha256=BSmwrkzYwEQhz9NbZCJsldY532PqgZJzxzsVk6ue0bM,93
64
66
  edu_rdm_integration/collect_data/non_calculated/base/errors.py,sha256=dGawEQ2ItTxlFt9ynhYhpqx40Qmrlg2rOskH0DsNVnQ,297
@@ -76,7 +78,7 @@ edu_rdm_integration/enum_register/mixins.py,sha256=shLb8-9ySoY9kK1Wzb46I9wRKglQI
76
78
  edu_rdm_integration/enum_register/register.py,sha256=5OWOjK-M0Erd_5CENpBaXhVtfL0pEaDl3Bev5QKNDJc,2218
77
79
  edu_rdm_integration/export_data/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
78
80
  edu_rdm_integration/export_data/consts.py,sha256=Z6Uho8oViMc0t1tyQFZfrPFBwPx6KnhUf_iZ0qgob8o,16
79
- edu_rdm_integration/export_data/export.py,sha256=iR-eZbsU5gdWxxbB7jSrzAGlUdCrWd3VGdDD_vYWo8o,16478
81
+ edu_rdm_integration/export_data/export.py,sha256=Axij_T66EX7I_VjLeVaELuKharDNAVAWDsFlZpqf9EY,16879
80
82
  edu_rdm_integration/export_data/generators.py,sha256=yLDOcHB1PoilJwXtKGxZQhDjpeKBzEWoosahbJJ4Ba4,4020
81
83
  edu_rdm_integration/export_data/helpers.py,sha256=hU346RmQ17Ra2etFvxXI7JQlLyp_0KxH1jm-eeCqejc,2933
82
84
  edu_rdm_integration/export_data/strategies.py,sha256=ocHskG-x54U-ESX56OlZPgOibkGkuv7VKmK7hVmqj_0,6725
@@ -129,10 +131,13 @@ edu_rdm_integration/function_templates/function_export_data_template/validators.
129
131
  edu_rdm_integration/management/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
130
132
  edu_rdm_integration/management/general.py,sha256=Yem9IcU9es_xA6Ap8v2IpAjNL1CI5VkkvQffUoJhQhY,13381
131
133
  edu_rdm_integration/management/commands/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
134
+ edu_rdm_integration/management/commands/check_upload_status.py,sha256=jGqFMLMgrirG_H2OxzxNG3Z3d1xvORIzWqzKBUtr22Y,3843
135
+ edu_rdm_integration/management/commands/collect_latest_models_data.py,sha256=4ObBFqYMkX5v1saV9V7PivYBm0RMcDCGNlMGx77UAjs,750
132
136
  edu_rdm_integration/management/commands/collect_models_data.py,sha256=11RshcQiXk6Flz-8oiFuM5KSZ_8_c1111qkqmyiu5sw,1022
133
137
  edu_rdm_integration/management/commands/datamart_status.py,sha256=ImpXHwRnbcgE5pcz2_QixfEQPSLU6WH1V8Vm7xnPidU,1211
134
138
  edu_rdm_integration/management/commands/datamart_upload.py,sha256=folrNInlhBP5yLSNXtujah-rfk8oWYi1Z8i5k3bsIKw,1487
135
139
  edu_rdm_integration/management/commands/export_entities_data.py,sha256=Mas1zwsH-zRl6exbzF81Y4KyLAOJo-uurkdb6sZccuc,920
140
+ edu_rdm_integration/management/commands/export_latest_entities_data.py,sha256=LTMWvcp2if76kW4GD74x8fA5FAfdDnnuaUBFFGjqjVk,1146
136
141
  edu_rdm_integration/migrations/0001_initial.py,sha256=toNuYoHZePe5wJ6AKEW9oPOdt2OefmxDEDDJGYQIrFk,18719
137
142
  edu_rdm_integration/migrations/0002_init_data_uploadstatus.py,sha256=p8EtwowHesoRvnOxeVMPCclUaCw2aQJZ8y5Zz7xbb8Q,737
138
143
  edu_rdm_integration/migrations/0003_create_index_file_upload_status.py,sha256=TiLnqQ8bxkVI7sRa5-D3JQ6jopFYDoH1ytSxmU6USUo,735
@@ -147,9 +152,9 @@ edu_rdm_integration/uploader_log/apps.py,sha256=tYJj4-sDlq8fLOSvw18L_yys7SILpTKW
147
152
  edu_rdm_integration/uploader_log/enums.py,sha256=rgSO3BL2rh2xpfm0Pt4waQW8fB1VMJLdsGmr3SXwH_U,266
148
153
  edu_rdm_integration/uploader_log/managers.py,sha256=y5wTSMzF9hpOpIU_A7nIafL_LBU3QEie6LAYWoB-pBQ,3203
149
154
  edu_rdm_integration/uploader_log/ui.py,sha256=YM9Buqp2wxE95Wf5gvAATBzuYzDOossK1sEmvFk07cI,2110
150
- edu_rdm_integration-0.8.6.dist-info/LICENSE,sha256=uw43Gjjj-1vXWCItfSrNDpbejnOwZMrNerUh8oWbq8Q,3458
151
- edu_rdm_integration-0.8.6.dist-info/METADATA,sha256=4Zxa7ibaKf2vZmYGqYM8oYr0NcJGaHOJ8pn6O_2wJTE,48304
152
- edu_rdm_integration-0.8.6.dist-info/WHEEL,sha256=yQN5g4mg4AybRjkgi-9yy4iQEFibGQmlz78Pik5Or-A,92
153
- edu_rdm_integration-0.8.6.dist-info/namespace_packages.txt,sha256=AbpHGcgLb-kRsJGnwFEktk7uzpZOCcBY74-YBdrKVGs,1
154
- edu_rdm_integration-0.8.6.dist-info/top_level.txt,sha256=nRJV0O14UtNE-jGIYG03sohgFnZClvf57H5m6VBXe9Y,20
155
- edu_rdm_integration-0.8.6.dist-info/RECORD,,
155
+ edu_rdm_integration-0.9.0.dist-info/LICENSE,sha256=uw43Gjjj-1vXWCItfSrNDpbejnOwZMrNerUh8oWbq8Q,3458
156
+ edu_rdm_integration-0.9.0.dist-info/METADATA,sha256=OI0gv65iJLUIwzRqXz0tI8SUr2RSGYY_bKFLQGvQzJA,49762
157
+ edu_rdm_integration-0.9.0.dist-info/WHEEL,sha256=yQN5g4mg4AybRjkgi-9yy4iQEFibGQmlz78Pik5Or-A,92
158
+ edu_rdm_integration-0.9.0.dist-info/namespace_packages.txt,sha256=AbpHGcgLb-kRsJGnwFEktk7uzpZOCcBY74-YBdrKVGs,1
159
+ edu_rdm_integration-0.9.0.dist-info/top_level.txt,sha256=nRJV0O14UtNE-jGIYG03sohgFnZClvf57H5m6VBXe9Y,20
160
+ edu_rdm_integration-0.9.0.dist-info/RECORD,,