Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Android] attempt to write a readonly database #1137

Open
haashem opened this issue Oct 27, 2024 · 9 comments
Open

[Android] attempt to write a readonly database #1137

haashem opened this issue Oct 27, 2024 · 9 comments

Comments

@haashem
Copy link

haashem commented Oct 27, 2024

Hi,

I use sqflite to cache network responses, and delete the expired caches weekly.
After publishing the app, I got below crash reports in Firebase for some Android devices:

Fatal Exception: io.flutter.plugins.firebase.crashlytics.FlutterError
DatabaseException(attempt to write a readonly database (code 1032 SQLITE_READONLY_DBMOVED[1032])) sql 'INSERT OR REPLACE INTO network_cache ("key", response, timestamp) VALUES (?, ?, ?)' args [https://api.myapp.com/api/app?platform=a..., {"request":{"method":"HttpMethod.get","url":"https..., 2024-10-26T03:56:12.376333]

Screenshot 2024-10-27 at 11 32 12

Here is how I create the db:

static Future<SqliteCacheDatabase> init() async {
    final dbDirectory = await getApplicationCacheDirectory();
    dbDirectory.create(recursive: true);
    final db = await openDatabase(
      join(dbDirectory.path, 'sqflite_cache.db'),
      onCreate: (db, version) {
        return db.execute('''
        CREATE TABLE $networkCacheTableName(
          key TEXT PRIMARY KEY,
          response TEXT,
          timestamp TEXT
        )
      ''');
      },
      version: 1,
    );
    return SqliteCacheDatabase._(db);
  }

I don't delete the db file after creation.

Do you think it can be result of parallel writes in the same table?

  @override
  Future<void> insert(CachedResponse cachedResponse, {required String key}) =>
      _database.instance.insert(
        SqliteCacheDatabase.networkCacheTableName,
        {
          'key': key,
          'response': jsonEncode(cachedResponse.response),
          'timestamp': cachedResponse.timestamp.toIso8601String(),
        },
        conflictAlgorithm: ConflictAlgorithm.replace,
      );

@alextekartik
Copy link
Contributor

sqflite on iOS and Android uses whatever SQLite bundled on the platform. It seems the UPSERT (ON CONFLICT) feature was added in SQLite 3.24 which might not be present in some Android version: See

You can do a "SELECT sqlite_version()" to check the version.

I see 2 solutions:

  • Do not use on CONFLICT and instead use transaction to catch the error on the insert and then perform the update
  • Try sqflite_common_ffi along with sqlite3 and sqlite3 flutter libs. I have not tried it much but it should work

@alextekartik
Copy link
Contributor

Forget that, I see the report is on Android 14 which should be fine. Parallel writes should be fine. sqflite serializes calls anyway. Still puzzled...my only "feeling" would be not to use the cache directory but instead use getDatabasesPath on Android and manage on your side to not make it too big. I don't know what is the Android behavior as when the cache is cleared (low storage situation, mabye your database is very big) but indeed if Android decides to delete the files, the error reported might happen. One alternative solution would be to catch this error and re-open the database when it happens. I'll try later (not very soon) to reproduce the issue by filling a db on the cache or by forcing file deletion but so far sqlite does not provide any automatic mechanism to handle this issue.

@haashem
Copy link
Author

haashem commented Nov 19, 2024

Surprisingly I faced the same error on iOS simulator while app was in state of launch, this is the first time I see this error during debug.

Screenshot 2024-11-19 at 19 32 32

@alextekartik
Copy link
Contributor

Do you confirm that you create the directory before opening the database ? (there is a missing await for dbDirectory.create in some code you posted above)

@haashem
Copy link
Author

haashem commented Nov 21, 2024

Wow 🤩,
great catch! I didn't notice it!
I must await dbDirectory.create(recursive: true);

I should look for a linter check to enforce explicitly use await to unawaited()!

@alextekartik
Copy link
Contributor

unawaited_futures ! Defintly a good lint to add

@haashem
Copy link
Author

haashem commented Nov 22, 2024

Thanks for your point. I close the issue for now. if I face the error again, will let you know. but based on your point it should resolve the error.

@haashem haashem closed this as completed Nov 22, 2024
@haashem
Copy link
Author

haashem commented Jan 7, 2025

Hi again,
The issue regressed again, and await dbDirectory.create(recursive: true); didn't fix.
The scenario is I cache network responses and when the app moves to background, I delete the expired caches.

When the db becomes readonly? do you think the db becomes readonly when a delete operation is in progress? this issue is only happening on Android.

@haashem haashem reopened this Jan 7, 2025
@alextekartik
Copy link
Contributor

as I said above, one reason could be the usage of the application cache directory that Android might cleanup this folder at any time. Can you try using getDatabasesPath() instead on Android?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants