Ticket T724378
Visible to All Users

How to use the DefaultForceCaseInsensitiveForAnySource option with PostgreSQL

created 6 years ago

[DevExpress Support Team: CLONED FROM T385990: How to make the Grid's filter to be a case- and accent-insensitive in Server Mode]
Hi Uriah,

I have been using MSSQL and everything was working fine technically but due to cost factor I have been exploring PostgreSQL which seems to no cost solution for me and performance wise much better as well. But I am running in to few difficulties, one of which is the case-sensitivity.

In my web XAF app, when the user searches any text then it has to be case sensitive. I looked at this post of yours and found that there is a global setting for making the search case in-sensitive. I have applied this static property and set it to true in the Global.asax file but that doesn't seems to be working either. Could you please guide me on how to get rid of this case sensitivity for Postgres

C#
protected void Application_Start(Object sender, EventArgs e) { DevExpress.Data.Helpers.ServerModeCore.DefaultForceCaseInsensitiveForAnySource = true; }

Answers approved by DevExpress Support

created 6 years ago (modified 4 years ago)

Hi Farooq,

The DevExpress.Data.Helpers.ServerModeCore.DefaultForceCaseInsensitiveForAnySource option is in effect only for Server Mode data sources. Other components do not respect this option. Please see alternative solutions in the How to make the Grid's filter to be a case- and accent-insensitive in Server Mode article.

Please note that  the FilterController component in an XAF application provides an event that you can use to substitute the default filter expression with a custom one: CustomGetFilters. Using this event, you can implement case-insensitive filtering for the Full Text Search Action. For example, convert all text values to upper case. Please refer to the following ticket where we provided a corresponding example: How can I activate "case insensitive" search in lookup views.

Update:

I found that PostgreSQL requires you to prepare SQL query parameters and specify the database type to correctly filter by a column of the citext type (Using Dapper and Postgresql - citext data type). By default, XPO does not take this specificity into account (citext is not in the Data Types Supported by XPO list) and the filtering will not work property (values will be correctly loaded from the database, though).

To overcome this difficulty, you can implement your own PostgreSqlConnectionProvider descendant (How to create a custom XPO connection provider and then use it in an XAF application) and override its CreateParameter method.

C#
protected override IDataParameter CreateParameter(IDbCommand command, object value, string name, DBColumnType dbType, string dbTypeName, int size) { NpgsqlParameter param = (NpgsqlParameter)base.CreateParameter(command, value, name, dbType, dbTypeName, size); if(param.DbType == DbType.String && dbTypeName == null) param.NpgsqlDbType = NpgsqlDbType.Citext; return param; }

This solution implies that all text columns in the database use the citex type. To force XPO to always use this type when creating new columns, override the GetSqlCreateColumnTypeForString method in your custom connection provider.

C#
protected override string GetSqlCreateColumnTypeForString(DBTable table, DBColumn column) { return "citext"; }

According to the PostgreSQL documentation (8.3. Character Types), there is no performance difference between varcharvarchar(n) , and text types.

    Show previous comments (3)

      Hi Uriah,

      Thanks for your efforts in digging out a workaround, I have also been doing that during the weekend and it is really sad that a basic funcationility is so hard to achieve in the Postgres.

      Now, If we consider converting CITEXT column type then we would have to change every text column of every table and I believe that is too difficult and hard to maintain as we are using the power of ORM with XAF that takes care of DB updates. Is there any possibilities to manage the column datatype with the DB updater tool when it detects the connection is to Postgres?

      Using the FilterController event and overriding may serve as a workaround but then the Column Filters, Filter Builder, etc… will still have the issue.

      DevExpress Support Team 6 years ago

        Hi Farooq,

        I found that PostgreSQL requires you to prepare SQL query parameters and specify the database type to correctly filter by a column of the citext type (Using Dapper and Postgresql - citext data type). By default, XPO does not take this specificity into account (citext is not in the Data Types Supported by XPO list) and the filtering will not work property (values will be correctly loaded from the database, though).

        To overcome this difficulty, you can implement your own PostgreSqlConnectionProvider descendant (How to create a custom XPO connection provider and then use it in an XAF application) and override its CreateParameter method.

        C#
        protected override IDataParameter CreateParameter(IDbCommand command, object value, string name, DBColumnType dbType, string dbTypeName, int size) { NpgsqlParameter param = (NpgsqlParameter)base.CreateParameter(command, value, name, dbType, dbTypeName, size); if(param.DbType == DbType.String && dbTypeName == null) param.NpgsqlDbType = NpgsqlDbType.Citext; return param; }

        This solution implies that all text columns in the database use the citex type. To force XPO to always use this type when creating new columns, override the GetSqlCreateColumnTypeForString method in your custom connection provider.

        C#
        protected override string GetSqlCreateColumnTypeForString(DBTable table, DBColumn column) { return "citext"; }

        According to the PostgreSQL documentation (8.3. Character Types), there is no performance difference between varchar, varchar(n) , and text types.

        >> Is there any possibilities to manage the column datatype with the DB updater tool when it detects the connection is to Postgres?

        The DB Updater tool invokes the UpdateDatabaseBeforeUpdateSchema and UpdateDatabaseAfterUpdateSchema methods in the Updater class defined in your module. Technically, you can programmatically determine the connection type (for example, add a custom property to an XafApplication descendant and initialize it in the CreateDefaultObjectSpaceProvider method) and execute SQL queries using the ExecuteNonQueryCommand method (How to: Handle Renamings and Deletions of Business Classes and their Properties). However, this approach is cumbersome and if your application is not intended to work in an unknown environment, it is better to update the legacy database manually.

        J J
        Joche Ojeda - DevExpress MVP 5 years ago

          To avoid problems with searches, I recreated XAF search controller, the idea is to convert the values to compare to uppercase as well as the value from the column by using the function to upper on the server side.

          Here you can find an article about my experiences with the search controller in XAF https://www.jocheojeda.com/2018/12/01/xaf-filtercontroller-case-insensitive-search/ feel free to take and/or modify the code as you needed

          Disclaimer: The information provided on DevExpress.com and affiliated web properties (including the DevExpress Support Center) is provided "as is" without warranty of any kind. Developer Express Inc disclaims all warranties, either express or implied, including the warranties of merchantability and fitness for a particular purpose. Please refer to the DevExpress.com Website Terms of Use for more information in this regard.

          Confidential Information: Developer Express Inc does not wish to receive, will not act to procure, nor will it solicit, confidential or proprietary materials and information from you through the DevExpress Support Center or its web properties. Any and all materials or information divulged during chats, email communications, online discussions, Support Center tickets, or made available to Developer Express Inc in any manner will be deemed NOT to be confidential by Developer Express Inc. Please refer to the DevExpress.com Website Terms of Use for more information in this regard.