Expression: replace message structure by function

pull/215/head
M66B 9 months ago
parent 538a630c31
commit 31401f9fd2

@ -170,7 +170,7 @@ public class EntityRule {
private static final int URL_TIMEOUT = 15 * 1000; // milliseconds private static final int URL_TIMEOUT = 15 * 1000; // milliseconds
private static final List<String> EXPR_VARIABLES = Collections.unmodifiableList(Arrays.asList( private static final List<String> EXPR_VARIABLES = Collections.unmodifiableList(Arrays.asList(
"to", "from", "subject", "text", "message" "to", "from", "subject", "text"
)); ));
static boolean needsHeaders(EntityMessage message, List<EntityRule> rules) { static boolean needsHeaders(EntityMessage message, List<EntityRule> rules) {
@ -500,9 +500,6 @@ public class EntityRule {
// Expression // Expression
Expression expression = getExpression(this, message, headers, html, context); Expression expression = getExpression(this, message, headers, html, context);
if (expression != null) { if (expression != null) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
boolean experiments = prefs.getBoolean("experiments", false);
if (experiments) {
if (needsHeaders(expression) && headers == null && message.headers == null) if (needsHeaders(expression) && headers == null && message.headers == null)
throw new IllegalArgumentException(context.getString(R.string.title_rule_no_headers)); throw new IllegalArgumentException(context.getString(R.string.title_rule_no_headers));
@ -512,7 +509,6 @@ public class EntityRule {
if (!Boolean.TRUE.equals(result)) if (!Boolean.TRUE.equals(result))
return false; return false;
} }
}
// Safeguard // Safeguard
if (jsender == null && if (jsender == null &&
@ -670,6 +666,37 @@ public class EntityRule {
} }
} }
@FunctionParameter(name = "value")
public static class MessageFunction extends AbstractFunction {
private EntityMessage message;
MessageFunction(EntityMessage message) {
this.message = message;
}
@Override
public EvaluationValue evaluate(
Expression expression, Token functionToken, EvaluationValue... parameterValues) {
List<Object> result = new ArrayList<>();
String name = parameterValues[0].getStringValue();
if (name != null && message != null)
try {
Field field = message.getClass().getField(name);
if (field != null) {
field.setAccessible(true);
Object value = field.get(message);
Log.i("EXPR message " + name + "=" + value);
if (value != null)
result.add(value);
}
} catch (Throwable ex) {
Log.e(ex);
}
return new EvaluationValue(result, ExpressionConfiguration.defaultConfiguration());
}
}
@InfixOperator(precedence = OPERATOR_PRECEDENCE_COMPARISON) @InfixOperator(precedence = OPERATOR_PRECEDENCE_COMPARISON)
public static class ContainsOperator extends AbstractOperator { public static class ContainsOperator extends AbstractOperator {
private boolean regex; private boolean regex;
@ -737,34 +764,11 @@ public class EntityRule {
headers = Collections.list(new InternetHeaders(bis, true).getAllHeaders()); headers = Collections.list(new InternetHeaders(bis, true).getAllHeaders());
} }
Map<String, Object> props = new HashMap<>();
if (message != null) {
for (Field field : message.getClass().getDeclaredFields())
try {
int modifiers = field.getModifiers();
if (!Modifier.isPublic(modifiers) || Modifier.isStatic(modifiers))
continue;
Class<?> type = field.getType();
if (type.isArray())
continue;
if (type.isPrimitive() ||
Boolean.class.isAssignableFrom(type) ||
Integer.class.isAssignableFrom(type) ||
Long.class.isAssignableFrom(type) ||
String.class.isAssignableFrom(type)) {
field.setAccessible(true);
Object value = field.get(message);
props.put(field.getName(), value);
} else
Log.i("EXPR skipping non primitive=" + field.getName());
} catch (Throwable ex) {
Log.w(ex);
}
}
ExpressionConfiguration configuration = ExpressionConfiguration.defaultConfiguration(); ExpressionConfiguration configuration = ExpressionConfiguration.defaultConfiguration();
configuration.getFunctionDictionary().addFunction("Header", configuration.getFunctionDictionary().addFunction("Header",
new HeaderFunction(headers)); new HeaderFunction(headers));
configuration.getFunctionDictionary().addFunction("Message",
new MessageFunction(message));
configuration.getOperatorDictionary().addOperator("Contains", configuration.getOperatorDictionary().addOperator("Contains",
new ContainsOperator(false)); new ContainsOperator(false));
configuration.getOperatorDictionary().addOperator("Matches", configuration.getOperatorDictionary().addOperator("Matches",
@ -774,17 +778,16 @@ public class EntityRule {
.with("to", to) .with("to", to)
.with("from", from) .with("from", from)
.with("subject", message == null ? null : Arrays.asList(message.subject)) .with("subject", message == null ? null : Arrays.asList(message.subject))
.with("text", doc == null ? null : Arrays.asList(doc.text())) .with("text", doc == null ? null : Arrays.asList(doc.text()));
.with("message", props);
if (message != null) { if (message != null) {
boolean hasAttachments = false; boolean hasAttachments = false;
for (String variable : expression.getUsedVariables()) for (String variable : expression.getUsedVariables())
if (!hasAttachments && "attachments".equals(variable)) { if (!hasAttachments && "hasAttachments".equals(variable)) {
hasAttachments = true; hasAttachments = true;
DB db = DB.getInstance(context); DB db = DB.getInstance(context);
List<EntityAttachment> attachments = db.attachment().getAttachments(message.id); List<EntityAttachment> attachments = db.attachment().getAttachments(message.id);
expression.with("attachments", attachments == null ? 0 : attachments.size()); expression.with("hasAttachments", attachments != null && !attachments.isEmpty());
} }
} }

@ -853,9 +853,10 @@
android:id="@+id/etExpression" android:id="@+id/etExpression"
android:layout_width="0dp" android:layout_width="0dp"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:fontFamily="monospace"
android:hint="@string/title_optional" android:hint="@string/title_optional"
android:importantForAutofill="no" android:importantForAutofill="no"
android:textAppearance="@style/TextAppearance.AppCompat.Medium" android:textAppearance="@style/TextAppearance.AppCompat.Small"
app:layout_constraintEnd_toEndOf="parent" app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent" app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/tvExpression" /> app:layout_constraintTop_toBottomOf="@id/tvExpression" />

Loading…
Cancel
Save