Лучший SQL Builder – используем jOOQ на Android

habr.png

Лучший SQL Builder. Используем jOOQ на Android


Введение

При разработке Android-приложений вполне естественным считается использовать SQLite базу данных в качестве основного хранилища. Обычно, базы данных на мобильных устройствах имеют весьма простенькие схемы и состоят из 10–15 таблиц. Для подобных случаев подходит почти любой SQL Builder, ORM, и даже голый SQLite API.

Но, увы, не всем разработчикам везет, и порой на нашу долю выпадает описывать большие модели данных, использовать хранимые процедуры, настраивать работу с кастомными типами данных или писать 10 INNER JOIN в запросе за очень толстой сущностью. Так не повезло и вашему покорному слуге, из чего и появился материал для данной статьи. Что же, суровые времена требуют суровых мер. Итак, накатываем jOOQ на Android.


Все бы хорошо, но

Но есть два факта, с которыми нужно будет совладать. Первый из них подстерегает нас на самом начале работы с jOOQ: на этапе идеологическом. Для того, чтобы инициировать процесс кодогенерации, нужно, собственно, заиметь базу данных, к которой jooq plugin подключится. Данная проблема решается легко, создаем template-проект с описанием gradle task для генерации, после чего создаем БД локально, прописываем в конфигах пути, запускаем плагин и копируем полученные исходники к себе в проект.

Далее, допустим мы сгенерировали все необходимые классы. Просто так скопировать их в Android-проект мы не сможем — будут требоваться дополнительные зависимости, первая из которых — на javax аннотации. Варианта два, оба банальные. Либо добавляем библиотеку (org.glassfish: javax.annotation), либо — используем замечательный инструмент — find & replace in scope.

И вот казалось бы, все хорошо, все предварительные настройки сделаны, классы скопированы и импортированы в проект. Возможно вам даже удастся запустить приложение, и есть шанс, что оно заработает. Если вы обязаны поддерживать Android API Level < 24 – не ведитесь, на это наш путь еще не заканчивается. Дело заключается в том, что jOOQ на текущий момент в open-source версии во многом использует Java 8, которая, как известно, с Android дружит весьма условно. Эта проблема также решается двумя вариантами: либо покупаем jOOQ, пишем в саппорт и слезно выпрашиваем версию на Java 6 или Java 7 (у них есть, судя по статьям в сети), либо же, если у вас, как и у меня, нет жесткой необходимости обладать всеми последними нововведениями библиотеки, равно как и желания платить, то есть второй путь. jOOQ начал переходить на Java 8 не так давно. Последняя из версий до миграции является 3.6.0, что значит, что мы можем использовать генератор с параметром groovy version = '3.6.0' и поддерживать старые версии устройств.

И последнее, что ждет энтузиастов, пошедших по этой тропинке отчаяния. В Android в принципе нет JDBC, что значит, что пришло время скрестив пальцы искать 3rd-party solutions. К счастью, подобная библиотека есть — SQLDroid.

Все. Основные этапы и действия на них бегло расписаны. Теперь перейдем к коду, тут все в целом довольно логично, но, дабы сократить ваше время, приведу примеры из собственного проекта.


Кодогенерация

Настройка jOOQ плагина будет выглядеть следующим образом:

buildScript {
    repositories {
        mavenCentral()
    }

    dependencies {
        classpath "nu.studer:gradle-jooq-plugin:$jooq_plugin_version"
    }
}

apply plugin: 'nu.studer.jooq'

dependencies {
    jooqRuntime "org.xerial:sqlite-jdbc:$xerial_version"
}

jooq {
    version = '3.6.0'
    edition = 'OSS'

    dev(sourceSets.main) {
        jdbc {
            driver = 'org.sqlite.JDBC'
            url = 'jdbc:sqlite:/Path/To/Database/database.db3'
        }

        generator {
            name = 'org.jooq.util.DefaultGenerator'
            strategy {
                name = 'org.jooq.util.DefaultGeneratorStrategy'
            }
            database {
                name = 'org.jooq.util.sqlite.SQLiteDatabase'
            }
            generate {
                relations = true
                deprecated = false
                records = true
                immutablePojos = true
                fluentSetters = true
            }
            target {
                packageName = 'com.example.mypackage.data.database'
            }
        }
    }
}


Android

Необходимые зависимости:

implementation "org.jooq:jooq:$jooq_version"
implementation "org.sqldroid:sqldroid:$sqldroid_version"
implementation "org.glassfish:javax.annotation:$javax_annotations_version"

А теперь исходники класса-обертки, для работы с jOOQ через SQLiteOpenHelper. В целом, без него можно было бы обойтись, но так куда удобнее (на мой взгляд), чтобы благополучно пользоваться и одним, и вторым API.

class DatabaseAdapter(private val context: Context)
    : SQLiteOpenHelper(context, DATABASE_NAME, null, DATABASE_VERSION) {

    companion object {

        private const val DATABASE_NAME     = "database"
        private const val DATABASE_VERSION  = 1

        @JvmStatic private val OPEN_OPTIONS = mapOf(
                "cache" to "shared",
                "journal_mode" to "WAL",
                "synchronous" to "ON",
                "foreign_keys" to "ON")
    }

    val connectionLock: ReentrantLock = ReentrantLock(true)
    val configuration: Configuration by lazy {
        connectionLock.withLock {
            // ensure the database exists,
            // all upgrades are performed,
            // and connection is ready to be set
            val database = context.openOrCreateDatabase(
                DATABASE_NAME, 
                Context.MODE_PRIVATE, 
                null)
            if (database.isOpen) {
                database.close()
            }

            // register SQLDroid driver to be used for establishing connections
            // with our database
            DriverManager.registerDriver(
                Class.forName("org.sqldroid.SQLDroidDriver")
                    .newInstance() as Driver)

            DefaultConfiguration()
                    .set(SQLiteSource(
                        context, 
                        OPEN_OPTIONS, 
                        "database", 
                        arrayOf("databases")))
                    .set(SQLDialect.SQLITE)
        }
    }

    override fun onCreate(db: SQLiteDatabase) {
        // acquire monitor until the database connection is created
        // this is important as otherwise transactions might be tryingg to run
        // concurrently that will lead to crashes
        connectionLock.withLock {
            // TODO: Create tables
        }
    }

    override fun onOpen(db: SQLiteDatabase) {
        // acquire monitor until the database connection is established
        // this is important as otherwise transactions might be tryingg to run
        // concurrently that will lead to crashes
        connectionLock.withLock {
            super.onOpen(db)
        }
    }

    override fun onUpgrade(db: SQLiteDatabase, oldVersion: Int, newVersion: Int) {
        // acquire monitor until the database is upgraded
        // this is important as otherwise transactions might be tryingg to run
        // concurrently that will lead to crashes
        connectionLock.withLock {

        }
    }

    infix inline fun  transaction(noinline f: (Configuration) -> T): Observable
            = Observable.create { emitter ->
        val tryResult = Try {
            connectionLock.withLock {
                DSL.using(configuration).transactionResult(f)
            }
        }

        when (tryResult) {
            is Try.Success -> {
                emitter.onNext(tryResult.value)
                emitter.onComplete()
            }
            is Try.Failure -> {
                emitter.onError(tryResult.exception)
            }
        }
    }

    fun invalidate() {
        connectionLock.withLock {
            // TODO: Drop tables, vacuum and create tables
        }
    }

    private class SQLiteSource(val context: Context,
                               val options: Map,
                               val database: String,
                               val fragments: Array): DroidDataSource() {

        override fun getConnection(): Connection
                = openConnection(options)

        private fun openConnection(options: Map = emptyMap()): Connection {
            return DriverManager.getConnection(StringBuilder().apply {
                append("jdbc:sqldroid:")
                append(context.applicationInfo.dataDir)
                append("/")
                append(buildFragments(fragments))
                append(database)
                append("?")
                append(buildOptions(options))
            }.toString())
        }

        private fun buildFragments(fragments: Array)
                = when (fragments.isEmpty()) {
            true  -> ""
            false -> "${fragments.joinToString("/")}/"
        }

        private fun buildOptions(options: Map)
                = options.mapTo(mutableListOf()) { entry ->
            "${entry.key}=${entry.value}"
        }
                .joinToString(separator = "&")
    }
}


Вместо заключения

Как оказалось, настройка jOOQ в Android — не такой уж и сложный процесс. Достаточно проделать его один раз, а далее можно смело заниматься копипастом из старых проектов.

И небольшой бонус, который дает jOOQ тем, кто его использует. Как видно из примера, пи открытии подключения используется cached mode. В чем же цимес? Android SDK SQLite API не предоставляет возможности работать с БД в данном режиме, сильно ограничивая нас в организации межпроцессного взаимодействия в приложениях. Теперь же — можно смело использовать данный режим, что уже само по себе может послужить причиной перехода на этот замечательный фреймворк.

© Habrahabr.ru