|
| 1 | +import builtins |
1 | 2 | import sqlite3
|
2 | 3 | from unittest.mock import MagicMock, patch
|
3 | 4 |
|
4 | 5 | import pytest
|
| 6 | +from pytest_mock import MockerFixture |
5 | 7 |
|
| 8 | +from mysql_to_sqlite3.sqlite_utils import CollatingSequences |
6 | 9 | from mysql_to_sqlite3.transporter import MySQLtoSQLite
|
7 | 10 |
|
8 | 11 |
|
@@ -152,6 +155,155 @@ def test_transfer_exception_handling(self, mock_sqlite_connect: MagicMock, mock_
|
152 | 155 | # Verify that foreign keys are re-enabled in the finally block
|
153 | 156 | mock_sqlite_cursor.execute.assert_called_with("PRAGMA foreign_keys=ON")
|
154 | 157 |
|
| 158 | + @patch("mysql_to_sqlite3.transporter.sqlite3.connect") |
| 159 | + @patch("mysql_to_sqlite3.transporter.mysql.connector.connect") |
| 160 | + def test_sqlite_strict_supported_keeps_flag( |
| 161 | + self, |
| 162 | + mock_mysql_connect: MagicMock, |
| 163 | + mock_sqlite_connect: MagicMock, |
| 164 | + mocker: MockerFixture, |
| 165 | + ) -> None: |
| 166 | + """Ensure STRICT mode remains enabled when SQLite supports it.""" |
| 167 | + |
| 168 | + class FakeMySQLConnection: |
| 169 | + def __init__(self) -> None: |
| 170 | + self.database = None |
| 171 | + |
| 172 | + def is_connected(self) -> bool: |
| 173 | + return True |
| 174 | + |
| 175 | + def cursor(self, *args, **kwargs) -> MagicMock: |
| 176 | + return MagicMock() |
| 177 | + |
| 178 | + mock_logger = MagicMock() |
| 179 | + mocker.patch.object(MySQLtoSQLite, "_setup_logger", return_value=mock_logger) |
| 180 | + mocker.patch("mysql_to_sqlite3.transporter.sqlite3.sqlite_version", "3.38.0") |
| 181 | + mock_mysql_connect.return_value = FakeMySQLConnection() |
| 182 | + |
| 183 | + mock_sqlite_cursor = MagicMock() |
| 184 | + mock_sqlite_connection = MagicMock() |
| 185 | + mock_sqlite_connection.cursor.return_value = mock_sqlite_cursor |
| 186 | + mock_sqlite_connect.return_value = mock_sqlite_connection |
| 187 | + |
| 188 | + from mysql_to_sqlite3 import transporter as transporter_module |
| 189 | + |
| 190 | + original_isinstance = builtins.isinstance |
| 191 | + |
| 192 | + def fake_isinstance(obj: object, classinfo: object) -> bool: |
| 193 | + if classinfo is transporter_module.MySQLConnectionAbstract: |
| 194 | + return True |
| 195 | + return original_isinstance(obj, classinfo) |
| 196 | + |
| 197 | + mocker.patch("mysql_to_sqlite3.transporter.isinstance", side_effect=fake_isinstance) |
| 198 | + |
| 199 | + instance = MySQLtoSQLite( |
| 200 | + sqlite_file="file.db", |
| 201 | + mysql_user="user", |
| 202 | + mysql_password=None, |
| 203 | + mysql_database="db", |
| 204 | + mysql_host="localhost", |
| 205 | + mysql_port=3306, |
| 206 | + sqlite_strict=True, |
| 207 | + ) |
| 208 | + |
| 209 | + assert instance._sqlite_strict is True |
| 210 | + mock_logger.warning.assert_not_called() |
| 211 | + |
| 212 | + @patch("mysql_to_sqlite3.transporter.sqlite3.connect") |
| 213 | + @patch("mysql_to_sqlite3.transporter.mysql.connector.connect") |
| 214 | + def test_sqlite_strict_unsupported_disables_flag( |
| 215 | + self, |
| 216 | + mock_mysql_connect: MagicMock, |
| 217 | + mock_sqlite_connect: MagicMock, |
| 218 | + mocker: MockerFixture, |
| 219 | + ) -> None: |
| 220 | + """Ensure STRICT mode is disabled with a warning on old SQLite versions.""" |
| 221 | + |
| 222 | + class FakeMySQLConnection: |
| 223 | + def __init__(self) -> None: |
| 224 | + self.database = None |
| 225 | + |
| 226 | + def is_connected(self) -> bool: |
| 227 | + return True |
| 228 | + |
| 229 | + def cursor(self, *args, **kwargs) -> MagicMock: |
| 230 | + return MagicMock() |
| 231 | + |
| 232 | + mock_logger = MagicMock() |
| 233 | + mocker.patch.object(MySQLtoSQLite, "_setup_logger", return_value=mock_logger) |
| 234 | + mocker.patch("mysql_to_sqlite3.transporter.sqlite3.sqlite_version", "3.36.0") |
| 235 | + mock_mysql_connect.return_value = FakeMySQLConnection() |
| 236 | + |
| 237 | + mock_sqlite_cursor = MagicMock() |
| 238 | + mock_sqlite_connection = MagicMock() |
| 239 | + mock_sqlite_connection.cursor.return_value = mock_sqlite_cursor |
| 240 | + mock_sqlite_connect.return_value = mock_sqlite_connection |
| 241 | + |
| 242 | + from mysql_to_sqlite3 import transporter as transporter_module |
| 243 | + |
| 244 | + original_isinstance = builtins.isinstance |
| 245 | + |
| 246 | + def fake_isinstance(obj: object, classinfo: object) -> bool: |
| 247 | + if classinfo is transporter_module.MySQLConnectionAbstract: |
| 248 | + return True |
| 249 | + return original_isinstance(obj, classinfo) |
| 250 | + |
| 251 | + mocker.patch("mysql_to_sqlite3.transporter.isinstance", side_effect=fake_isinstance) |
| 252 | + |
| 253 | + instance = MySQLtoSQLite( |
| 254 | + sqlite_file="file.db", |
| 255 | + mysql_user="user", |
| 256 | + mysql_password=None, |
| 257 | + mysql_database="db", |
| 258 | + mysql_host="localhost", |
| 259 | + mysql_port=3306, |
| 260 | + sqlite_strict=True, |
| 261 | + ) |
| 262 | + |
| 263 | + assert instance._sqlite_strict is False |
| 264 | + mock_logger.warning.assert_called_once() |
| 265 | + |
| 266 | + def test_build_create_table_sql_appends_strict(self) -> None: |
| 267 | + """Ensure STRICT is appended to CREATE TABLE statements when enabled.""" |
| 268 | + with patch.object(MySQLtoSQLite, "__init__", return_value=None): |
| 269 | + instance = MySQLtoSQLite() |
| 270 | + |
| 271 | + instance._sqlite_strict = True |
| 272 | + instance._sqlite_json1_extension_enabled = False |
| 273 | + instance._mysql_cur_dict = MagicMock() |
| 274 | + instance._mysql_cur_dict.fetchall.side_effect = [ |
| 275 | + [ |
| 276 | + { |
| 277 | + "Field": "id", |
| 278 | + "Type": "INTEGER", |
| 279 | + "Null": "NO", |
| 280 | + "Default": None, |
| 281 | + "Key": "PRI", |
| 282 | + "Extra": "auto_increment", |
| 283 | + }, |
| 284 | + { |
| 285 | + "Field": "name", |
| 286 | + "Type": "TEXT", |
| 287 | + "Null": "NO", |
| 288 | + "Default": None, |
| 289 | + "Key": "", |
| 290 | + "Extra": "", |
| 291 | + }, |
| 292 | + ], |
| 293 | + [], |
| 294 | + ] |
| 295 | + instance._mysql_cur_dict.fetchone.return_value = {"count": 0} |
| 296 | + instance._mysql_database = "db" |
| 297 | + instance._collation = CollatingSequences.BINARY |
| 298 | + instance._prefix_indices = False |
| 299 | + instance._without_tables = False |
| 300 | + instance._without_foreign_keys = True |
| 301 | + instance._logger = MagicMock() |
| 302 | + |
| 303 | + sql = instance._build_create_table_sql("products") |
| 304 | + |
| 305 | + assert "STRICT;" in sql |
| 306 | + |
155 | 307 | def test_constructor_missing_mysql_database(self) -> None:
|
156 | 308 | """Test constructor raises ValueError if mysql_database is missing."""
|
157 | 309 | from mysql_to_sqlite3.transporter import MySQLtoSQLite
|
|
0 commit comments