Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
25 changes: 0 additions & 25 deletions config.json.exemple

This file was deleted.

2 changes: 0 additions & 2 deletions sinc-db.bat

This file was deleted.

29 changes: 29 additions & 0 deletions sincronizacao.log
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
2026-05-15 11:16:13 | INFO | === SINC: PROD LEITURA ➔ Banco Local ===
2026-05-15 11:16:13 | INFO | package: mysql.connector.plugins
2026-05-15 11:16:13 | INFO | plugin_name: mysql_native_password
2026-05-15 11:16:13 | INFO | AUTHENTICATION_PLUGIN_CLASS: MySQLNativePasswordAuthPlugin
2026-05-15 11:16:13 | ERROR | Tabela 'tabela1' não encontrada na origem (PROD LEITURA).
2026-05-15 11:16:13 | ERROR | Tabela 'tabela2' não encontrada na origem (PROD LEITURA).
2026-05-15 11:16:13 | ERROR | Tabela 'tabela3' não encontrada na origem (PROD LEITURA).
2026-05-15 11:16:14 | ERROR | Tabela 'tabela4' não encontrada na origem (PROD LEITURA).
2026-05-15 11:16:14 | ERROR | Tabela 'tabela5' não encontrada na origem (PROD LEITURA).
2026-05-15 11:16:34 | INFO | === SINC: PROD LEITURA ➔ Banco Local ===
2026-05-15 11:16:34 | INFO | package: mysql.connector.plugins
2026-05-15 11:16:34 | INFO | plugin_name: mysql_native_password
2026-05-15 11:16:34 | INFO | AUTHENTICATION_PLUGIN_CLASS: MySQLNativePasswordAuthPlugin
2026-05-15 11:16:34 | ERROR | Erro em periodo: [WinError 2] O sistema não pode encontrar o arquivo especificado
2026-05-15 11:17:05 | INFO | === SINC: PROD LEITURA ➔ Banco Local ===
2026-05-15 11:17:05 | INFO | package: mysql.connector.plugins
2026-05-15 11:17:05 | INFO | plugin_name: mysql_native_password
2026-05-15 11:17:05 | INFO | AUTHENTICATION_PLUGIN_CLASS: MySQLNativePasswordAuthPlugin
2026-05-15 11:17:05 | ERROR | Erro em periodo: [WinError 2] O sistema não pode encontrar o arquivo especificado
2026-05-15 11:17:12 | INFO | === SINC: PROD LEITURA ➔ Banco Local ===
2026-05-15 11:17:12 | INFO | package: mysql.connector.plugins
2026-05-15 11:17:12 | INFO | plugin_name: mysql_native_password
2026-05-15 11:17:12 | INFO | AUTHENTICATION_PLUGIN_CLASS: MySQLNativePasswordAuthPlugin
2026-05-15 11:17:12 | ERROR | Erro em periodo: [WinError 2] O sistema não pode encontrar o arquivo especificado
2026-05-15 11:23:31 | INFO | === SINC: PROD LEITURA ➔ Banco Local ===
2026-05-15 11:23:31 | INFO | package: mysql.connector.plugins
2026-05-15 11:23:31 | INFO | plugin_name: mysql_native_password
2026-05-15 11:23:31 | INFO | AUTHENTICATION_PLUGIN_CLASS: MySQLNativePasswordAuthPlugin
2026-05-15 11:23:32 | INFO | SUCCESS: periodo sincronizada.
20 changes: 15 additions & 5 deletions sync-db-gui.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,18 @@ def load_config(self):
self.destino = self.config["destino"]
self.mysql_bin = self.settings["mysql_bin_path"]

mysql_client = self.settings.get("mysql_client", "local")
if mysql_client == "docker":
_container = self.settings["docker_container"]
_bin = self.settings.get("docker_mysql_bin", "/usr/bin")
self.mysqldump_base_cmd = ["docker", "exec", "-i", _container,
f"{_bin}/{self.settings.get('docker_mysqldump_cmd', 'mysqldump')}"]
self.mysql_base_cmd = ["docker", "exec", "-i", _container,
f"{_bin}/{self.settings.get('docker_mysql_cmd', 'mysql')}"]
else:
self.mysqldump_base_cmd = [os.path.join(self.mysql_bin, "mysqldump.exe")]
self.mysql_base_cmd = [os.path.join(self.mysql_bin, "mysql.exe")]

def get_conn(self, cfg):
params = cfg.copy()
params.pop('alias', None)
Expand Down Expand Up @@ -84,8 +96,7 @@ def sync_table(self, table, progress_callback):

# Dump
self.log(f"Baixando dados de {table}...")
dump_cmd = [
os.path.join(self.mysql_bin, "mysqldump.exe"),
dump_cmd = self.mysqldump_base_cmd + [
"-h", self.origem['host'], "-u", self.origem['user'], f"-p{self.origem['password']}"
]
if not needs_creation: dump_cmd.append("--no-create-info")
Expand All @@ -104,9 +115,8 @@ def sync_table(self, table, progress_callback):
# Import com Feedback
self.log(f"Importando {table}...")
file_size = os.path.getsize(temp_sql)
import_cmd = [
os.path.join(self.mysql_bin, "mysql.exe"),
"-h", self.destino['host'], "-u", self.destino['user'], f"-p{self.destino['password']}",
import_cmd = self.mysql_base_cmd + [
"-h", self.destino['host'], "-u", self.destino['user'], f"-p{self.destino['password']}",
"--default-character-set=latin1", self.destino['database']
]

Expand Down
2 changes: 2 additions & 0 deletions sync-db.bat
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
@echo off
python "C:\scripts\database_table_cloning_tool\sync-db.py" %*
21 changes: 16 additions & 5 deletions sync-db.py
Original file line number Diff line number Diff line change
Expand Up @@ -39,9 +39,20 @@ def load_config():
CONFIG = load_config()

# Atalhos para facilidade
MYSQL_BIN_PATH = CONFIG["settings"]["mysql_bin_path"]
MYSQLDUMP_EXE = os.path.join(MYSQL_BIN_PATH, "mysqldump.exe")
MYSQL_EXE = os.path.join(MYSQL_BIN_PATH, "mysql.exe")
SETTINGS = CONFIG["settings"]
MYSQL_CLIENT = SETTINGS.get("mysql_client", "local")
MYSQL_BIN_PATH = SETTINGS["mysql_bin_path"]

if MYSQL_CLIENT == "docker":
_container = SETTINGS["docker_container"]
_bin = SETTINGS.get("docker_mysql_bin", "/usr/bin")
_dump_cmd = SETTINGS.get("docker_mysqldump_cmd", "mysqldump")
_mysql_cmd = SETTINGS.get("docker_mysql_cmd", "mysql")
MYSQLDUMP_BASE_CMD = ["docker", "exec", "-i", _container, f"{_bin}/{_dump_cmd}"]
MYSQL_BASE_CMD = ["docker", "exec", "-i", _container, f"{_bin}/{_mysql_cmd}"]
else:
MYSQLDUMP_BASE_CMD = [os.path.join(MYSQL_BIN_PATH, "mysqldump.exe")]
MYSQL_BASE_CMD = [os.path.join(MYSQL_BIN_PATH, "mysql.exe")]

# O arquivo CSV padrão e o LOG também ficam na pasta do script para serem centrais
DEFAULT_CSV_FILE = os.path.join(SCRIPT_DIR, CONFIG["settings"]["default_csv_file"])
Expand Down Expand Up @@ -207,7 +218,7 @@ def sync_data(table, needs_creation=False):
try:
with open(temp_sql, "w", encoding='utf-8') as f: f.write(header_sql)

dump_cmd = [MYSQLDUMP_EXE, "-h", ORIGEM_CONFIG['host'], "-u", ORIGEM_CONFIG['user'], f"-p{ORIGEM_CONFIG['password']}"]
dump_cmd = MYSQLDUMP_BASE_CMD + ["-h", ORIGEM_CONFIG['host'], "-u", ORIGEM_CONFIG['user'], f"-p{ORIGEM_CONFIG['password']}"]
if not needs_creation: dump_cmd.append("--no-create-info")
dump_cmd.extend(["--complete-insert", "--skip-add-locks", "--skip-comments", "--single-transaction", "--quick", "--default-character-set=latin1", ORIGEM_CONFIG['database'], table])

Expand All @@ -224,7 +235,7 @@ def sync_data(table, needs_creation=False):

# Import
file_size = os.path.getsize(temp_sql)
import_cmd = [MYSQL_EXE, "-h", DESTINO_CONFIG['host'], "-u", DESTINO_CONFIG['user'], f"-p{DESTINO_CONFIG['password']}", "--default-character-set=latin1", DESTINO_CONFIG['database']]
import_cmd = MYSQL_BASE_CMD + ["-h", DESTINO_CONFIG['host'], "-u", DESTINO_CONFIG['user'], f"-p{DESTINO_CONFIG['password']}", "--default-character-set=latin1", DESTINO_CONFIG['database']]

if HAS_RICH:
with subprocess.Popen(import_cmd, stdin=subprocess.PIPE, stderr=subprocess.PIPE) as proc:
Expand Down