sqlite3 and result ordering according to non-ASCII collations |
For a while I was not sure how to order query results in Codeigniter when using sqlite3 database, which natively lacks non-ascii text collation functions. So I resorted to sorting php arrays, which is quite suboptimal with larger datasets.
When I took a closer look, I realized that it is quite simple to handle sorting in vanilla PHP case. Below I post an example of creating a Lithuanian collation and using it with sqlite3. But I see createCollation and createFunction sqlite3 driver methods are not used in Codeigniter4 code. So I have to find a way to implement these myself. Could someone take a look at the vanilla php example below and maybe throw in a suggestion on how to optimally do that in Codeigniter? Vanilla php example PHP Code: // the locale offers proper collation order, set it if it is not default And you continue as if you were using native sqlite3 functions: PHP Code: $results = $db->query('SELECT bar FROM foo where LT_LOWER(foo) LIKE %Žemai%' ORDER BY bar ASC COLLATE LT); You can also assign this LT collation as default to individual columns when creating a table (and then omit it in select statements), but then you won't be able to read that table without declaring those additional collation methods before the basic select queries.
==
Donatas G.
Try:
PHP Code: $db->connID->createCollation('LT', function ($a, $b) { (06-17-2024, 01:38 AM)kenjis Wrote: Try: Voila! so I added these three calls to BaseController: PHP Code: setlocale(LC_ALL, 'lt_LT.UTF-8'); And then the controller method to test: PHP Code: public function collate() And I see that both ordering and case - insensitive search with utf8 characters is working. The last question would be - how do I integrate COLLATE LT into query builder. So far I failed (see the third code block, which does not order).
==
Donatas G.
It seems there is no way to generate the query `ORDER BY last_name COLLATE LT DESC` with orderBy().
https://github.com/codeigniter4/CodeIgni...1449-L1497 You need to extend the orderBy() method. (06-17-2024, 04:17 PM)kenjis Wrote: It seems there is no way to generate the query `ORDER BY last_name COLLATE LT DESC` with orderBy().Thanks @kenjis; I think I will simply go with declaring collation on creation of table, so that I do not need to declare it on queries. I might also change the collation name to utf8_lithuanian_ci to make it compatible with the MySql counterpart for easier migration if I ever decide to move it also seems that I can overwrite the native LOWER function of sqlite with my own implementation: PHP Code: $db->createFunction('LOWER', function ($str) { If I do not notice any shortcomings to such approach, then SQlite3 collation and search functionality can be made on par with mysql's using proper migration code and a few statements in the BaseController class.
==
Donatas G.
I finally managed to compile the ICU plugin for SQLite3, which can then be loaded and used instead of the PHP functions, making the search almost twice as fast. So definitely worth it with larger databases. Here is the tutorial on how to do that on Ubuntu 22.04 (and the process is the same on Ubuntu 24.04):
https://dg.lapas.info/irasas/enhancing-s...extension/
==
Donatas G. |
Welcome Guest, Not a member yet? Register Sign In |