@@ -31,128 +31,98 @@ import io.spine.core.Where
31
31
import io.spine.option.GoesOption
32
32
import io.spine.protodata.Compilation
33
33
import io.spine.protodata.ast.Field
34
+ import io.spine.protodata.ast.FieldType
34
35
import io.spine.protodata.ast.File
35
36
import io.spine.protodata.ast.MessageType
36
37
import io.spine.protodata.ast.PrimitiveType.TYPE_BYTES
37
38
import io.spine.protodata.ast.PrimitiveType.TYPE_STRING
38
- import io.spine.protodata.ast.Span
39
39
import io.spine.protodata.ast.event.FieldOptionDiscovered
40
40
import io.spine.protodata.ast.field
41
41
import io.spine.protodata.ast.qualifiedName
42
- import io.spine.protodata.ast.toPath
43
42
import io.spine.protodata.ast.unpack
43
+ import io.spine.protodata.check
44
44
import io.spine.protodata.plugin.Policy
45
- import io.spine.server.event.NoReaction
45
+ import io.spine.server.event.Just
46
46
import io.spine.server.event.React
47
- import io.spine.server.event.asA
48
- import io.spine.server.tuple.EitherOf2
47
+ import io.spine.server.event.just
49
48
import io.spine.validation.event.GoesFieldDiscovered
50
49
import io.spine.validation.event.goesFieldDiscovered
50
+ import io.spine.validation.protodata.findField
51
+ import io.spine.validation.protodata.message
51
52
52
53
/* *
53
- * A policy to add a validation rule to a type whenever the `(goes)` field option
54
- * is discovered.
54
+ * Controls whether a field should be validated with the `(goes)` option.
55
55
*
56
- * This option, when being applied to a target field `A`, declares a dependency to
57
- * a field `B`. So, whenever `A` is set, `B` also must be set.
56
+ * Whenever a filed marked with `(goes)` option is discovered,
57
+ * emits [GoesFieldDiscovered] if the following conditions are met:
58
58
*
59
- * Upon discovering a field with the mentioned option, the police emits the following
60
- * composite rule for `A`: `(A isNot Set) OR (B is Set)`.
59
+ * 1. The field type is supported by the option.
60
+ * 2. The companion field is present in the message.
61
+ * 3. The companion field and the target field are different fields.
62
+ * 4. The companion field type is supported by the option.
61
63
*
62
- * Please note, this police relies on implementation of `required` option to determine
63
- * whether the field is set. Thus, inheriting its behavior regarding the supported
64
- * field types and specification about when a field of a specific type is considered
65
- * to be set.
64
+ * Ant violation of the above conditions leads to a compilation error.
66
65
*/
67
66
internal class GoesPolicy : Policy <FieldOptionDiscovered >() {
68
67
69
- // TODO:2025-02-18:yevhenii.nadtochii: Make non-nullable in ProtoData.
70
68
override val typeSystem by lazy { super .typeSystem!! }
71
69
72
70
@React
73
71
override fun whenever (
74
72
@External @Where(field = OPTION_NAME , equals = GOES )
75
73
event : FieldOptionDiscovered
76
- ): EitherOf2 <GoesFieldDiscovered , NoReaction > {
74
+ ): Just <GoesFieldDiscovered > {
77
75
val field = event.subject
78
76
val file = event.file
79
77
checkFieldType(field, file)
80
78
81
- // TODO:2025-02-18:yevhenii.nadtochii: Use a shortcut for `defaultMessage`.
82
79
val option = event.option.unpack<GoesOption >()
83
- val declaringType = field.declaringType
84
- val declaringMessage = typeSystem.findMessage(declaringType)!! .first
80
+ val declaringMessage = typeSystem.message(field.declaringType)
85
81
val companionName = option.with
86
82
checkFieldExists(declaringMessage, companionName, field, file)
87
83
88
84
val companionField = declaringMessage.field(companionName)
89
- checkDistinct (field, companionField, file)
85
+ checkFieldsDistinct (field, companionField, file)
90
86
checkCompanionType(companionField, file)
91
87
92
- val message = option.errorMsg.ifEmpty { DefaultErrorMessage .from( option.descriptorForType) }
88
+ val message = option.errorMsg.ifEmpty { option.descriptorForType.defaultMessage }
93
89
return goesFieldDiscovered {
94
90
id = field.id()
95
91
errorMessage = message
96
92
companion = companionField
97
93
subject = field
98
- }.asA ()
94
+ }.just ()
99
95
}
100
96
}
101
97
102
- public fun Field.id (): FieldId = fieldId {
103
- type = declaringType
104
- name = this @id.name
105
- }
106
-
107
- private fun checkFieldType (field : Field , file : File ) {
108
- val type = field.type
109
- if (type.isPrimitive && type.primitive !in SUPPORTED_PRIMITIVES ) {
110
- compilationError(file, field.span) {
111
- " The field type `${field.type} ` of the `${field.qualifiedName} ` field " +
112
- " is not supported by the `($GOES )` option."
113
- }
98
+ private fun checkFieldType (field : Field , file : File ) =
99
+ Compilation .check(field.type.isSupported(), file, field.span) {
100
+ " The field type `${field.type} ` of the `${field.qualifiedName} ` field " +
101
+ " is not supported by the `($GOES )` option."
114
102
}
115
- }
116
103
117
- private fun checkCompanionType (field : Field , file : File ) {
118
- val type = field.type
119
- if (type.isPrimitive && type.primitive !in SUPPORTED_PRIMITIVES ) {
120
- compilationError(file, field.span) {
121
- " The field type `${field.type} ` of the companion `${field.qualifiedName} ` field " +
122
- " is not supported by the `($GOES )` option."
123
- }
104
+ private fun checkCompanionType (companion : Field , file : File ) =
105
+ Compilation .check(companion .type.isSupported(), file, companion .span) {
106
+ " The field type `${companion .type} ` of the companion `${companion .qualifiedName} ` field " +
107
+ " is not supported by the `($GOES )` option."
124
108
}
125
- }
126
109
127
- private fun checkFieldExists (message : MessageType , companion : String , field : Field , file : File ) {
128
- if (message.fieldList.find { it.name.value == companion } == null ) {
129
- compilationError(file, field.span) {
130
- " The message `${message.name.qualifiedName} ` does not have `$companion ` field " +
131
- " declared as companion of `${field.name.value} ` by the `($GOES )` option."
132
- }
110
+ private fun checkFieldExists (message : MessageType , companion : String , field : Field , file : File ) =
111
+ Compilation .check(message.findField(companion ) != null , file, field.span) {
112
+ " The message `${message.name.qualifiedName} ` does not have `$companion ` field " +
113
+ " declared as companion of `${field.name.value} ` by the `($GOES )` option."
133
114
}
134
- }
135
115
136
- /* *
137
- * Checks that the given [field] and its [companion] are distinct fields.
138
- */
139
- private fun checkDistinct (field : Field , companion : Field , file : File ) {
140
- if (field == companion ) {
141
- compilationError(file, field.span) {
142
- " The `($GOES )` option can not use the marked field as its own companion. " +
143
- " Self-referencing is prohibited. Please specify another field. " +
144
- " The invalid field: `${field.qualifiedName} `."
145
- }
116
+ private fun checkFieldsDistinct (field : Field , companion : Field , file : File ) =
117
+ Compilation .check(field != companion , file, field.span) {
118
+ " The `($GOES )` option can not use the marked field as its own companion. " +
119
+ " Self-referencing is prohibited. Please specify another field. " +
120
+ " The invalid field: `${field.qualifiedName} `."
146
121
}
147
- }
122
+
123
+ private fun FieldType.isSupported (): Boolean =
124
+ ! isPrimitive || primitive in SUPPORTED_PRIMITIVES
148
125
149
126
private val SUPPORTED_PRIMITIVES = listOf (
150
127
TYPE_STRING , TYPE_BYTES
151
128
)
152
-
153
- private fun compilationError (file : File , span : Span , message : () -> String ): Nothing =
154
- Compilation .error(
155
- file.toPath().toFile(),
156
- span.startLine, span.startColumn,
157
- message()
158
- )
0 commit comments