Skip to content

Commit

Permalink
feat: add rules param.name, param.type
Browse files Browse the repository at this point in the history
  • Loading branch information
tangcent committed Apr 14, 2024
1 parent 0cb6bb6 commit 776aa41
Show file tree
Hide file tree
Showing 11 changed files with 199 additions and 55 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -99,6 +99,16 @@ object ClassExportRuleKeys {
BooleanRuleMode.ANY
)

val PARAM_NAME: RuleKey<String> = SimpleRuleKey(
"param.name",
StringRuleMode.SINGLE
)

val PARAM_TYPE: RuleKey<String> = SimpleRuleKey(
"param.type",
StringRuleMode.SINGLE
)

val PARAM_DEFAULT_VALUE: RuleKey<String> = SimpleRuleKey(
"param.default.value",
StringRuleMode.MERGE_DISTINCT
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,9 @@ import com.itangcent.intellij.jvm.element.ExplicitParameter
import kotlin.reflect.KClass

interface ExportContext : Extensible {
/**
* the parent context, allowing navigation up the context hierarchy.
*/
fun parent(): ExportContext?

/**
Expand All @@ -23,22 +26,45 @@ interface ExportContext : Extensible {
fun psi(): PsiElement
}

/**
* Extends ExportContext for contexts dealing with variables (methods or parameters).
*/
interface VariableExportContext : ExportContext {

/**
* the name of the variable.
*/
fun name(): String

/**
* the type of the variable, which may be null if the type is not resolved.
*/
fun type(): DuckType?

/**
* the explicit element representation of the variable.
*/
fun element(): ExplicitElement<*>

/**
* Sets a resolved name for the variable, typically used for renaming.
*/
fun setResolvedName(name: String)
}

//region kits of ExportContext

/**
* find specific contexts by type.
*/
@Suppress("UNCHECKED_CAST")
fun <T : ExportContext> ExportContext.findContext(condition: KClass<T>): T? {
return findContext { condition.isInstance(it) } as? T
}

/**
* find specific contexts by condition.
*/
fun ExportContext.findContext(condition: (ExportContext) -> Boolean): ExportContext? {
var exportContext: ExportContext? = this
while (exportContext != null) {
Expand All @@ -50,39 +76,58 @@ fun ExportContext.findContext(condition: (ExportContext) -> Boolean): ExportCont
return null
}

fun <T> ExportContext.findExt(attr: String): T? {
var exportContext: ExportContext? = this
while (exportContext != null) {
exportContext.getExt<T>(attr)?.let { return it }
exportContext = exportContext.parent()
}
return null
}

//endregion

/**
* Base context with no parent, typically used for top-level classes.
*/
abstract class RootExportContext :
SimpleExtensible(), ExportContext {
override fun parent(): ExportContext? {
return null
}
}

/**
* General purpose context implementation with a specified parent context.
*/
abstract class AbstractExportContext(private val parent: ExportContext) :
SimpleExtensible(), ExportContext {
SimpleExtensible(), VariableExportContext {

private var resolvedName: String? = null

override fun parent(): ExportContext? {
return this.parent
}

/**
* Returns the name of the element.
*
* @return the element name.
*/
override fun name(): String {
return resolvedName ?: element().name()
}

override fun setResolvedName(name: String) {
this.resolvedName = name
}
}

/**
* Context specifically for a class
*/
class ClassExportContext(val cls: PsiClass) : RootExportContext() {
override fun psi(): PsiClass {
return cls
}
}

/**
* Context for a method, containing specifics about the method being exported.
*/
class MethodExportContext(
private val parent: ExportContext,
parent: ExportContext,
private val method: ExplicitMethod
) : AbstractExportContext(parent), VariableExportContext {

Expand Down Expand Up @@ -113,19 +158,27 @@ class MethodExportContext(
}
}

class ParameterExportContext(
private val parent: ExportContext,
private val parameter: ExplicitParameter
) : AbstractExportContext(parent), VariableExportContext {
/**
* Context for a parameter, containing specifics about the parameter being exported.
*/
interface ParameterExportContext : VariableExportContext {

/**
* Returns the name of the element.
*
* @return the element name.
*/
override fun name(): String {
return parameter.name()
}
override fun element(): ExplicitParameter

override fun psi(): PsiParameter
}

fun ParameterExportContext(
parent: ExportContext,
parameter: ExplicitParameter
): ParameterExportContext {
return ParameterExportContextImpl(parent, parameter)
}

class ParameterExportContextImpl(
parent: ExportContext,
private val parameter: ExplicitParameter
) : AbstractExportContext(parent), ParameterExportContext {

/**
* Returns the type of the variable.
Expand All @@ -145,18 +198,30 @@ class ParameterExportContext(
}
}

/**
* retrieve ClassExportContext based on the current context.
*/
fun ExportContext.classContext(): ClassExportContext? {
return this.findContext(ClassExportContext::class)
}

/**
* retrieve MethodExportContext based on the current context.
*/
fun ExportContext.methodContext(): MethodExportContext? {
return this.findContext(MethodExportContext::class)
}

/**
* retrieve ParameterExportContext based on the current context.
*/
fun ExportContext.paramContext(): ParameterExportContext? {
return this.findContext(ParameterExportContext::class)
}

/**
* Searches for an extended property, first locally then up the context hierarchy.
*/
fun <T> ExportContext.searchExt(attr: String): T? {
this.getExt<T>(attr)?.let { return it }
this.parent()?.searchExt<T>(attr)?.let { return it }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -509,7 +509,10 @@ abstract class RequestClassExporter : ClassExporter {
continue
}

parsedParams.add(ParameterExportContext(methodExportContext, param).also { it.raw() })
parsedParams.add(ParameterExportContext(methodExportContext, param)
.adaptive()
.also { it.raw() }
)
} finally {
ruleComputer.computer(ClassExportRuleKeys.API_PARAM_AFTER, param)
}
Expand Down Expand Up @@ -539,7 +542,7 @@ abstract class RequestClassExporter : ClassExporter {
processMethodParameter(
request,
parameterExportContext,
KVUtils.getUltimateComment(paramDocComment, parameterExportContext.name())
KVUtils.getUltimateComment(paramDocComment, parameterExportContext.rawName())
.append(readParamDoc(parameterExportContext.element()))
)
} finally {
Expand Down Expand Up @@ -589,7 +592,6 @@ abstract class RequestClassExporter : ClassExporter {
return
}

@Suppress("UNCHECKED_CAST")
protected open fun addParamAsQuery(
parameterExportContext: VariableExportContext,
request: Request, typeObject: Any?, paramDesc: String? = null,
Expand All @@ -600,7 +602,7 @@ abstract class RequestClassExporter : ClassExporter {
if (typeObject == Magics.FILE_STR) {
requestBuilderListener.addFormFileParam(
parameterExportContext,
request, parameterExportContext.paramName(),
request, parameterExportContext.name(),
parameterExportContext.required()
?: ruleComputer.computer(ClassExportRuleKeys.PARAM_REQUIRED, parameterExportContext.element())
?: false, paramDesc
Expand All @@ -612,7 +614,7 @@ abstract class RequestClassExporter : ClassExporter {
requestBuilderListener.addParam(
parameterExportContext,
request,
parameterExportContext.paramName(),
parameterExportContext.name(),
tinyQueryParam(parameterExportContext.defaultVal() ?: typeObject),
parameterExportContext.required()
?: ruleComputer.computer(ClassExportRuleKeys.PARAM_REQUIRED, parameterExportContext.element())
Expand Down Expand Up @@ -683,7 +685,6 @@ abstract class RequestClassExporter : ClassExporter {
}
}

@Suppress("UNCHECKED_CAST")
protected open fun addParamAsForm(
parameterExportContext: VariableExportContext,
request: Request, typeObject: Any?, paramDesc: String? = null,
Expand All @@ -693,7 +694,7 @@ abstract class RequestClassExporter : ClassExporter {
if (typeObject == Magics.FILE_STR) {
requestBuilderListener.addFormFileParam(
parameterExportContext,
request, parameterExportContext.paramName(),
request, parameterExportContext.name(),
ruleComputer.computer(
ClassExportRuleKeys.PARAM_REQUIRED,
parameterExportContext.element()
Expand Down Expand Up @@ -769,7 +770,7 @@ abstract class RequestClassExporter : ClassExporter {
} else {
requestBuilderListener.addFormParam(
parameterExportContext,
request, parameterExportContext.paramName(), tinyQueryParam(typeObject)?.toString(),
request, parameterExportContext.name(), tinyQueryParam(typeObject)?.toString(),
parameterExportContext.required()
?: ruleComputer.computer(ClassExportRuleKeys.PARAM_REQUIRED, parameterExportContext.element())
?: false, paramDesc
Expand Down Expand Up @@ -830,12 +831,36 @@ abstract class RequestClassExporter : ClassExporter {
}

//region extent of ParameterExportContext
fun VariableExportContext.setParamName(name: String) {
this.setExt("param_name", name)

fun ParameterExportContext.adaptive(): ParameterExportContext {
return AdaptiveParameterContext(this)
}

inner class AdaptiveParameterContext(
val delegate: ParameterExportContext
) : ParameterExportContext by delegate {

private val nameByRule by lazy {
ruleComputer.computer(ClassExportRuleKeys.PARAM_NAME, delegate.element())
}

private val typeByRule by lazy {
ruleComputer.computer(ClassExportRuleKeys.PARAM_TYPE, delegate.element())?.let {
duckTypeHelper.findDuckType(it, delegate.psi())
}
}

override fun name(): String {
return nameByRule ?: delegate.name()
}

override fun type(): DuckType? {
return typeByRule ?: delegate.type()
}
}

fun VariableExportContext.paramName(): String {
return this.getExt<String>("param_name") ?: this.name()
fun VariableExportContext.rawName(): String {
return this.element().name()
}

fun ExportContext.setRequired(name: Boolean) {
Expand Down
Loading

0 comments on commit 776aa41

Please sign in to comment.