Squeak.ru - шаблоны программирования

Ключевые проблемы уникального соединения Grails

Учитывая класс GORM:

    class PriceSheet {

    Client client
    Population population
    Product product
    RevenueModelType modelType

    BigDecimal price

    static constraints = {
        client(unique: ['population', 'product', 'modelType'])
    }
}

Я хочу, чтобы прайс-лист сохранялся / обновлялся только в том случае, если клиент, население, продукт и тип модели уникальны. (Должен быть только один элемент прейскуранта для комбинации клиента, населения, продукта и типа модели).

Ключ создается в mySQL.

Моя проблема в том, что проверка grails проходит, но не удается сохранить.

    priceSheetInstance.validate()

    if (priceSheetInstance.hasErrors()) {
        respond priceSheetInstance.errors, view:'create'
        return
    }

    priceSheetInstance.save flush:true

Есть идеи или предложения? Я ставлю отладчик на точку останова после проверки и вижу, что ошибки пусты.

Grails 2.3.10

04.08.2015

Ответы:


1

Во-первых, вам нужно сообщить GORM, какие свойства составляют ваш составной первичный ключ, а затем ваш доменный класс должен реализовать Serializable. Конечный результат будет примерно таким:

import org.apache.commons.lang.builder.HashCodeBuilder

class PriceSheet implements Serializable {
    Client client
    Population population
    Product product
    RevenueModelType modelType

    BigDecimal price

    static constraints = {        
    }

    static mapping = {
        id composite: ['client', 'population', 'product', 'modelType']
    }

    boolean equals(other) {
        if(!(other instanceof PriceSheet)) return false
        other.client == client && other.population == population && other.product == product && other.modelType == modelType
    }

    int hashCode() {
        def builder = new HashCodeBuilder()

        builder.with {
            append client
            append population
            append product
            append modelType
        }

        builder.toHashCode()
    }    
}

Подробнее о составных ключах GORM можно узнать здесь .

05.08.2015
  • Спасибо за вариант, но я бы предпочел, чтобы первичный ключ не был составным. Согласно документам, я должен быть в состоянии сделать это, поскольку пытаюсь использовать уникальный валидатор 07.08.2015
  • Да, но вы должны иметь это в виду (прямо из документации): возможно (хотя на практике это маловероятно), что проверка уникальности пройдет, но последующее сохранение не удастся. Если другое сохранение или обновление произойдет с обновлением базы данных между проверкой Grails и фактическим сохранением вашего экземпляра, вызов завершится ошибкой. Единственный способ предотвратить это - использовать транзакцию на уровне изоляции SERIALIZABLE, но это очень плохо для производительности. Возможно, вам лучше вызвать save (), а затем проверить наличие ошибок. 07.08.2015

  • 2

    Окончательное решение заключалось в переносе NewSession в контроллер для сохранений:

        PriceSheet.withNewSession {
    
            priceSheetInstance.validate()
    
            // needed to call save in order to check the unique compound key
            // validate alone wasn't working
            // also needed to wrap withNewSession above
            // PriceSheet GORM object implements Serializable as well
            // spent way more time than expected on this.... wrestled with Grails
    
            if (priceSheetInstance.hasErrors() || (priceSheetInstance.save(failOnError: true) && priceSheetInstance.hasErrors())) {
                respond priceSheetInstance.errors, view:'create'
                return
            }
    
            request.withFormat {
                form multipartForm {
                    flash.message = message(code: 'default.created.message', args: [message(code: 'priceSheet.label', default: 'Price Sheet'), priceSheetInstance?.id])
                    redirect (action: 'index')
                }
                '*' { respond priceSheetInstance, [status: CREATED] }
            }
        }
    

    Мне не нужно было оборачивать с помощью NewSession для обновлений ... на самом деле, обертывание с помощью NewSession для обновлений вызывало StaleObjectExceptions

    07.08.2015
    Новые материалы

    Угловая структура архитектуры
    Обратите внимание, что эта статья устарела, я решил создать новую с лучшей структурой и с учетом автономных компонентов: https://medium.com/@marekpanti/angular-standalone-architecture-b645edd0d54a..

    «Данные, которые большинство людей используют для обучения своих моделей искусственного интеллекта, поставляются со встроенным…
    Первоначально опубликовано HalkTalks: https://hacktown.com.br/blog/blog/os-dados-que-a-maioria-das-pessoas-usa-para-treinar-seus-modelos-de-inteligencia-artificial- ja-vem-com-um-vies-embutido/..

    Сильный ИИ против слабого ИИ: различия парадигм искусственного интеллекта
    В последние годы изучению и развитию искусственного интеллекта (ИИ) уделяется большое внимание и прогресс. Сильный ИИ и Слабый ИИ — две основные парадигмы в области искусственного интеллекта...

    Правильный способ добавить Firebase в ваш проект React с помощью React Hooks
    React + Firebase - это мощная комбинация для быстрого и безопасного создания приложений, от проверки концепции до массового производства. Раньше (знаете, несколько месяцев назад) добавление..

    Создайте API с помощью Python FastAPI
    Создание API с помощью Python становится очень простым при использовании пакета FastAPI. После установки и импорта вы можете создать приложение FastAPI и указать несколько конечных точек. Каждой..

    Веселье с прокси-сервером JavaScript
    Прокси-серверы JavaScript — это чистый сахар, если вы хотите создать некоторую общую логику в своих приложениях, чтобы облегчить себе жизнь. Вот один пример: Связь клиент-сервер Мы..

    Получить бесплатный хостинг для разработчиков | Разместите свой сайт за несколько шагов 🔥
    Статические веб-сайты — это веб-страницы с фиксированным содержанием и его постоянным содержанием. Но теперь статические сайты также обрабатывают динамические данные с помощью API и запросов...