diff --git a/sync/doc/changelog.rst b/sync/doc/changelog.rst
index 6ff2ee99..08d2165a 100644
--- a/sync/doc/changelog.rst
+++ b/sync/doc/changelog.rst
@@ -2,6 +2,7 @@
-------
- **New:** Use prime numbers for major releases ;-)
+- **New:** Support data files
- **Improvement:** make links dependent on project
`7.0.0`
diff --git a/sync/models/__init__.py b/sync/models/__init__.py
index d0b38e2a..46411344 100644
--- a/sync/models/__init__.py
+++ b/sync/models/__init__.py
@@ -10,6 +10,7 @@
from . import sync_project_demo
from . import sync_task
from . import sync_job
+from . import sync_data
from . import ir_logging
from . import ir_actions
from . import ir_attachment
diff --git a/sync/models/sync_data.py b/sync/models/sync_data.py
new file mode 100644
index 00000000..e26933b5
--- /dev/null
+++ b/sync/models/sync_data.py
@@ -0,0 +1,45 @@
+# Copyright 2024 Ivan Yelizariev
+import base64
+import csv
+import json
+from io import StringIO
+
+import yaml
+
+from odoo import fields, models
+
+
+class SyncData(models.Model):
+ _name = "sync.data"
+ _description = "Sync Data File"
+
+ name = fields.Char("Technical name")
+ project_id = fields.Many2one("sync.project", ondelete="cascade")
+ file_name = fields.Char("File Name")
+ file_content = fields.Binary("File Content")
+
+ def csv(self, *args, **kwargs):
+ """Parse CSV file from binary field."""
+ if self.file_content:
+ file_content = base64.b64decode(self.file_content)
+ file_content = file_content.decode("utf-8")
+ file_like_object = StringIO(file_content)
+ reader = csv.DictReader(file_like_object, *args, **kwargs)
+ return [row for row in reader]
+ return []
+
+ def json(self):
+ """Parse JSON file from binary field."""
+ if self.file_content:
+ file_content = base64.b64decode(self.file_content)
+ file_content = file_content.decode("utf-8")
+ return json.loads(file_content)
+ return {}
+
+ def yaml(self):
+ """Parse YAML file from binary field."""
+ if self.file_content:
+ file_content = base64.b64decode(self.file_content)
+ file_content = file_content.decode("utf-8")
+ return yaml.safe_load(file_content)
+ return None
diff --git a/sync/models/sync_project.py b/sync/models/sync_project.py
index d4a8885f..34cfc5cf 100644
--- a/sync/models/sync_project.py
+++ b/sync/models/sync_project.py
@@ -5,8 +5,10 @@
import base64
import logging
+import os
from datetime import datetime
+import urllib3
from pytz import timezone
from odoo import api, fields, models
@@ -107,6 +109,7 @@ class SyncProject(models.Model):
log_count = fields.Integer(compute="_compute_log_count")
link_ids = fields.One2many("sync.link", "project_id")
link_count = fields.Integer(compute="_compute_link_count")
+ data_ids = fields.One2many("sync.data", "project_id")
def copy(self, default=None):
default = dict(default or {})
@@ -319,6 +322,10 @@ def record2image(record, fname="image_1920"):
for w in self.task_ids.mapped("webhook_ids"):
WEBHOOKS[w.trigger_name] = w.website_url
+ DATA = AttrDict()
+ for d in self.data_ids:
+ DATA[d.name] = d
+
core_eval_context = {
"MAGIC": MAGIC,
"SECRETS": SECRETS,
@@ -330,6 +337,7 @@ def record2image(record, fname="image_1920"):
"CORE": CORE,
"PARAMS": PARAMS,
"WEBHOOKS": WEBHOOKS,
+ "DATA": DATA,
}
LIB = eval_export(safe_eval, self.common_code, lib_eval_context)
@@ -499,6 +507,38 @@ def magic_upgrade(self):
if gist_files.get(file_name):
vals[field_name] = gist_files[file_name]
+ # [DATA]
+ http = urllib3.PoolManager()
+ for file_info in gist_content["files"].values():
+ # e.g. "data.emoji.csv"
+ file_name = file_info["filename"]
+ if not file_name.startswith("data."):
+ continue
+ raw_url = file_info["raw_url"]
+ response = http.request("GET", raw_url)
+ if response.status == 200:
+ file_content = response.data
+ file_content = base64.b64encode(file_content)
+ else:
+ raise Exception(
+ f"Failed to fetch raw content from {raw_url}. Status code: {response.status}"
+ )
+
+ technical_name = file_name
+ technical_name = technical_name[len("data.") :]
+ technical_name = os.path.splitext(technical_name)[0]
+ technical_name = technical_name.replace(".", "_")
+
+ data_vals = {
+ "name": technical_name,
+ "project_id": self.id,
+ "file_name": file_name,
+ "file_content": file_content,
+ }
+ self.env["sync.data"]._create_or_update_by_xmlid(
+ data_vals, file_name, namespace=gist_id
+ )
+
# Tasks 🦋
for file_name in gist_files:
# e.g. "task.setup.py"
diff --git a/sync/security/ir.model.access.csv b/sync/security/ir.model.access.csv
index 5964da44..2ac9e400 100644
--- a/sync/security/ir.model.access.csv
+++ b/sync/security/ir.model.access.csv
@@ -5,6 +5,9 @@ access_sync_project_manager,sync.project manager,model_sync_project,sync_group_m
access_sync_task_user,sync.task user,model_sync_task,sync_group_user,1,0,0,0
access_sync_task_dev,sync.task dev,model_sync_task,sync_group_dev,1,1,1,1
access_sync_task_manager,sync.task manager,model_sync_task,sync_group_manager,1,1,1,1
+access_sync_data_user,sync.data user,model_sync_data,sync_group_user,1,0,0,0
+access_sync_data_dev,sync.data dev,model_sync_data,sync_group_dev,1,1,1,1
+access_sync_data_manager,sync.data manager,model_sync_data,sync_group_manager,1,1,1,1
access_sync_trigger_automation_user,sync.trigger.automation user,model_sync_trigger_automation,sync_group_user,1,0,0,0
access_sync_trigger_automation_dev,sync.trigger.automation dev,model_sync_trigger_automation,sync_group_dev,1,1,1,1
access_sync_trigger_automation_manager,sync.trigger.automation manager,model_sync_trigger_automation,sync_group_manager,1,1,1,1
diff --git a/sync/views/sync_project_views.xml b/sync/views/sync_project_views.xml
index 498c0ec0..e81a3e2a 100644
--- a/sync/views/sync_project_views.xml
+++ b/sync/views/sync_project_views.xml
@@ -214,28 +214,6 @@
-
@@ -275,6 +253,26 @@
+
+
+
+
+
+
+
+
+
+
+ Hint:
+
+ Documentation
+
+
+
+
diff --git a/sync/views/sync_task_views.xml b/sync/views/sync_task_views.xml
index 9d661c97..b6512973 100644
--- a/sync/views/sync_task_views.xml
+++ b/sync/views/sync_task_views.xml
@@ -68,6 +68,13 @@
Hint: Updating this code won't change the triggers.
Instead, update the gist file.
+
+
+ Documentation
+