com.zeoflow.depot.processor.InsertionMethodProcessor.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of depot-compiler Show documentation
Show all versions of depot-compiler Show documentation
The Depot persistence library provides an abstraction layer over SQLite to allow for more robust database access while using the full power of SQLite.
The newest version!
/*
* Copyright (C) 2021 ZeoFlow SRL
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
@file:Suppress("PLATFORM_CLASS_MAPPED_TO_KOTLIN")
package com.zeoflow.depot.processor
import com.zeoflow.depot.OnConflictStrategy.IGNORE
import com.zeoflow.depot.OnConflictStrategy.REPLACE
import com.zeoflow.depot.compiler.processing.XMethodElement
import com.zeoflow.depot.compiler.processing.XType
import com.zeoflow.depot.vo.InsertionMethod
import com.zeoflow.depot.vo.findFieldByColumnName
class InsertionMethodProcessor(
baseContext: Context,
val containing: XType,
val executableElement: XMethodElement
) {
val context = baseContext.fork(executableElement)
fun process(): InsertionMethod {
val delegate = ShortcutMethodProcessor(context, containing, executableElement)
val annotation = delegate.extractAnnotation(
com.zeoflow.depot.Insert::class,
ProcessorErrors.MISSING_INSERT_ANNOTATION
)
val onConflict = annotation?.value?.onConflict ?: OnConflictProcessor.INVALID_ON_CONFLICT
context.checker.check(
onConflict in REPLACE..IGNORE,
executableElement, ProcessorErrors.INVALID_ON_CONFLICT_VALUE
)
val returnType = delegate.extractReturnType()
val returnTypeName = returnType.typeName
context.checker.notUnbound(
returnTypeName, executableElement,
ProcessorErrors.CANNOT_USE_UNBOUND_GENERICS_IN_INSERTION_METHODS
)
val (entities, params) = delegate.extractParams(
targetEntityType = annotation?.getAsType("entity"),
missingParamError = ProcessorErrors.INSERTION_DOES_NOT_HAVE_ANY_PARAMETERS_TO_INSERT,
onValidatePartialEntity = { entity, pojo ->
val missingPrimaryKeys = entity.primaryKey.fields.any {
pojo.findFieldByColumnName(it.columnName) == null
}
context.checker.check(
entity.primaryKey.autoGenerateId || !missingPrimaryKeys,
executableElement,
ProcessorErrors.missingPrimaryKeysInPartialEntityForInsert(
partialEntityName = pojo.typeName.toString(),
primaryKeyNames = entity.primaryKey.fields.columnNames
)
)
// Verify all non null columns without a default value are in the POJO otherwise
// the INSERT will fail with a NOT NULL constraint.
val missingRequiredFields = (entity.fields - entity.primaryKey.fields).filter {
it.nonNull && it.defaultValue == null &&
pojo.findFieldByColumnName(it.columnName) == null
}
context.checker.check(
missingRequiredFields.isEmpty(),
executableElement,
ProcessorErrors.missingRequiredColumnsInPartialEntity(
partialEntityName = pojo.typeName.toString(),
missingColumnNames = missingRequiredFields.map { it.columnName }
)
)
}
)
val methodBinder = delegate.findInsertMethodBinder(returnType, params)
context.checker.check(
methodBinder.adapter != null,
executableElement,
ProcessorErrors.CANNOT_FIND_INSERT_RESULT_ADAPTER
)
return InsertionMethod(
element = executableElement,
name = executableElement.name,
returnType = returnType,
entities = entities,
parameters = params,
onConflict = onConflict,
methodBinder = methodBinder
)
}
}