com.squareup.sqldelight.intellij.lang.SqlDelightCompletionContributor.kt Maven / Gradle / Ivy
/*
* Copyright (C) 2016 Square, Inc.
*
* 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.
*/
package com.squareup.sqldelight.intellij.lang
import com.intellij.codeInsight.completion.CompletionContributor
import com.intellij.codeInsight.completion.CompletionInitializationContext
import com.intellij.codeInsight.completion.CompletionParameters
import com.intellij.codeInsight.completion.CompletionProvider
import com.intellij.codeInsight.completion.CompletionResultSet
import com.intellij.codeInsight.completion.CompletionType
import com.intellij.codeInsight.lookup.LookupElementBuilder
import com.intellij.patterns.ElementPattern
import com.intellij.patterns.ElementPatternCondition
import com.intellij.patterns.InitialPatternCondition
import com.intellij.psi.PsiElement
import com.intellij.util.ProcessingContext
import com.squareup.javapoet.TypeName
import com.squareup.sqldelight.SqliteLexer
import com.squareup.sqldelight.SqliteParser
import com.squareup.sqldelight.intellij.SqlDelightManager
import com.squareup.sqldelight.intellij.util.elementAt
import com.squareup.sqldelight.resolution.ResolutionError
import com.squareup.sqldelight.resolution.Resolver
import com.squareup.sqldelight.types.SymbolTable
import com.squareup.sqldelight.validation.SqlDelightValidator
import org.antlr.v4.runtime.ANTLRInputStream
import org.antlr.v4.runtime.CommonTokenStream
import org.antlr.v4.runtime.ParserRuleContext
private val DUMMY_IDENTIFIER = "sql_delight_dummy_identifier"
class SqlDelightCompletionContributor : CompletionContributor() {
init {
extend(
CompletionType.BASIC,
SqlDelightPattern(),
SqlDelightCompletionProvider()
)
}
override fun beforeCompletion(context: CompletionInitializationContext) {
context.dummyIdentifier = DUMMY_IDENTIFIER
}
}
private class SqlDelightCompletionProvider : CompletionProvider() {
override fun addCompletions(parameters: CompletionParameters, context: ProcessingContext?,
result: CompletionResultSet) {
val manager = SqlDelightManager.getInstance(parameters.originalFile) ?: return
if (parameters.originalFile !is SqliteFile) return
val lexer = SqliteLexer(ANTLRInputStream(parameters.position.containingFile.text))
val parser = com.squareup.sqldelight.SqliteParser(CommonTokenStream(lexer))
parser.parse().elementAt(parameters.offset)?.getAvailableValues(result, manager)
// No reason to do any other completion for SQLDelight files. Might save some time.
result.stopHere()
}
private fun ParserRuleContext.getAvailableValues(
result: CompletionResultSet, manager: SqlDelightManager
) {
try {
var symbolTable = manager.symbolTable
if (this is SqliteParser.Create_table_stmtContext) {
// It's likely the create table statement will fail to compile, so we need
// to give a fake type for it in the symbol table so resolution will run.
symbolTable += SymbolTable(tableTypes = mapOf(table_name().text to TypeName.OBJECT), tag = this)
}
val resolver = Resolver(symbolTable)
SqlDelightValidator().validate(this, resolver)
result.addAllElements(resolver.errors
.filter { it.originatingElement.text.endsWith(DUMMY_IDENTIFIER) }
.flatMap { lookupElements(it) })
} catch (e: Throwable) {
}
}
/**
* Given a resolution error, returns the possible column/table names that would resolve
* that error.
*/
private fun lookupElements(error: ResolutionError) =
when (error) {
is ResolutionError.TableNameNotFound -> error.availableTableNames
.map {
LookupElementBuilder.create(it)
}
is ResolutionError.ColumnNameNotFound -> error.availableColumns
.flatMap { it.columnNames() }
.filterNotNull()
.distinct()
.map {
LookupElementBuilder.create(it)
}
is ResolutionError.ColumnOrTableNameNotFound -> error.availableColumns
.filter { error.tableName == null || error.tableName == it.name }
.flatMap { it.columnNames() + it.tableNames() }
.filterNotNull()
.distinct()
.map {
LookupElementBuilder.create(it)
}
else -> emptyList()
}
}
/**
* The completion provider acts on all PsiElements so this pattern is effectively always true.
* Since the completion provider is already scoped to only .sq files the behavior should be safe.
*/
private class SqlDelightPattern : ElementPattern {
override fun accepts(o: Any?, context: ProcessingContext?) = true
override fun accepts(o: Any?) = true
override fun getCondition() = ElementPatternCondition(
object : InitialPatternCondition(PsiElement::class.java) {})
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy