Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Inject defaults into the planConnection query in the GTFS GraphQL schema #6339

Open
wants to merge 25 commits into
base: dev-2.x
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
c10fb01
Draft for injecting defaults into GraphQL schema
optionsome Dec 17, 2024
2c42d14
Don't rely on directive for injecting defaults
optionsome Dec 17, 2024
c51b975
Use dependency injection for providing GraphQL schema
optionsome Dec 30, 2024
f639ce5
Only construct schema if feature flag is on
optionsome Dec 30, 2024
38b5a5b
Use a slightly prettier method for getting parent name
optionsome Dec 30, 2024
698349e
Replace preferences with route request
optionsome Dec 31, 2024
7162c7a
Add structure for injecting defaults for arguments and use it for
optionsome Dec 31, 2024
280fd45
Add default for search window and allow resetting it to null
optionsome Dec 31, 2024
45eea03
Add the rest of defaults
optionsome Dec 31, 2024
d298811
Fix schema comment about duration being seconds
optionsome Dec 31, 2024
5fe8da8
Move schema tests to new file
optionsome Dec 31, 2024
4f89618
Add tests for default injection
optionsome Dec 31, 2024
6f4a6fb
Merge remote-tracking branch 'upstream/dev-2.x' into plan-connection-…
optionsome Dec 31, 2024
c105fc2
Fix comment
optionsome Dec 31, 2024
3beb9be
Use schema visitor instead of directive wiring
optionsome Jan 3, 2025
813bbbf
Initialize schema on server start up
optionsome Jan 7, 2025
9157359
Remove unnecessary annotation
optionsome Jan 8, 2025
7cd815e
Split factory method into two versions
optionsome Jan 8, 2025
3b3d7ff
Merge remote-tracking branch 'upstream/dev-2.x' into plan-connection-…
optionsome Jan 8, 2025
9f72862
Add javadoc
optionsome Jan 8, 2025
f7585cc
Remove service wrapper
optionsome Jan 10, 2025
69de270
Fix comment
optionsome Jan 10, 2025
fc7c77f
Add missing annotation
optionsome Jan 14, 2025
76db9b5
Refactor defaults
optionsome Jan 14, 2025
a06ed19
Merge remote-tracking branch 'upstream/dev-2.x' into plan-connection-…
optionsome Jan 21, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,192 @@
package org.opentripplanner.apis.gtfs;

import graphql.language.BooleanValue;
import graphql.language.FloatValue;
import graphql.language.IntValue;
import graphql.language.StringValue;
import graphql.language.Value;
import graphql.schema.GraphQLArgument;
import graphql.schema.GraphQLInputObjectField;
import graphql.schema.GraphQLNamedSchemaElement;
import graphql.schema.GraphQLSchemaElement;
import graphql.schema.GraphQLTypeVisitor;
import graphql.schema.GraphQLTypeVisitorStub;
import graphql.util.TraversalControl;
import graphql.util.TraverserContext;
import java.util.HashMap;
import java.util.Map;
import javax.annotation.Nullable;
import org.opentripplanner.routing.api.request.RouteRequest;

/**
* GraphQL type visitor that injects default values to input fields and query arguments from code
* and configuration.
*/
public class DefaultValueInjector extends GraphQLTypeVisitorStub implements GraphQLTypeVisitor {

private final Map<String, Value> defaultForKey;

public DefaultValueInjector(RouteRequest defaultRouteRequest) {
this.defaultForKey = createDefaultMapping(defaultRouteRequest);
}

@Override
public TraversalControl visitGraphQLArgument(
GraphQLArgument argument,
TraverserContext<GraphQLSchemaElement> context
) {
var defaultValue = getDefaultValueForSchemaObject(context, argument.getName());
if (defaultValue != null) {
return changeNode(
context,
argument.transform(builder -> builder.defaultValueLiteral(defaultValue).build())
);
}
return TraversalControl.CONTINUE;
}

@Override
public TraversalControl visitGraphQLInputObjectField(
GraphQLInputObjectField field,
TraverserContext<GraphQLSchemaElement> context
) {
var defaultValue = getDefaultValueForSchemaObject(context, field.getName());
if (defaultValue != null) {
return changeNode(
context,
field.transform(builder -> builder.defaultValueLiteral(defaultValue).build())
);
}
return TraversalControl.CONTINUE;
}

private Value getDefaultValueForSchemaObject(
TraverserContext<GraphQLSchemaElement> context,
String name
) {
// Arguments and input fields always have a parent
var parent = (GraphQLNamedSchemaElement) context.getParentNode();
var parentName = parent.getName();
var key = parentName + "." + name;
return defaultForKey.get(key);
}

private static Map<String, Value> createDefaultMapping(RouteRequest defaultRouteRequest) {
var builder = new DefaultMappingBuilder();
var preferences = defaultRouteRequest.preferences();
return builder
.intReq("planConnection.first", defaultRouteRequest.numItineraries())
.stringOpt("planConnection.searchWindow", defaultRouteRequest.searchWindow())
.stringReq("AlightPreferencesInput.slack", preferences.transit().alightSlack().defaultValue())
.intReq(
"BicycleParkingPreferencesInput.unpreferredCost",
preferences.bike().parking().unpreferredVehicleParkingTagCost().toSeconds()
)
.intReq("BicyclePreferencesInput.boardCost", preferences.bike().boardCost())
.floatReq("BicyclePreferencesInput.reluctance", preferences.bike().reluctance())
.floatReq("BicyclePreferencesInput.speed", preferences.bike().speed())
.intReq(
"BicycleWalkPreferencesCostInput.mountDismountCost",
preferences.bike().walking().mountDismountCost().toSeconds()
)
.floatReq(
"BicycleWalkPreferencesCostInput.reluctance",
preferences.bike().walking().reluctance()
)
.stringReq(
"BicycleWalkPreferencesInput.mountDismountTime",
preferences.bike().walking().mountDismountTime()
)
.floatReq("BicycleWalkPreferencesInput.speed", preferences.bike().walking().speed())
.stringReq("BoardPreferencesInput.slack", preferences.transit().boardSlack().defaultValue())
.floatReq("BoardPreferencesInput.waitReluctance", preferences.transfer().waitReluctance())
.intReq(
"CarParkingPreferencesInput.unpreferredCost",
preferences.car().parking().unpreferredVehicleParkingTagCost().toSeconds()
)
.floatReq("CarPreferencesInput.reluctance", preferences.car().reluctance())
.boolReq(
"DestinationBicyclePolicyInput.allowKeeping",
preferences.bike().rental().allowArrivingInRentedVehicleAtDestination()
)
.intReq(
"DestinationBicyclePolicyInput.keepingCost",
preferences.bike().rental().arrivingInRentalVehicleAtDestinationCost().toSeconds()
)
.boolReq(
"DestinationScooterPolicyInput.allowKeeping",
preferences.scooter().rental().allowArrivingInRentedVehicleAtDestination()
)
.intReq(
"DestinationScooterPolicyInput.keepingCost",
preferences.scooter().rental().arrivingInRentalVehicleAtDestinationCost().toSeconds()
)
.floatReq("ScooterPreferencesInput.reluctance", preferences.scooter().reluctance())
.floatReq("ScooterPreferencesInput.speed", preferences.scooter().speed())
.boolReq(
"TimetablePreferencesInput.excludeRealTimeUpdates",
preferences.transit().ignoreRealtimeUpdates()
)
.boolReq(
"TimetablePreferencesInput.includePlannedCancellations",
preferences.transit().includePlannedCancellations()
)
.boolReq(
"TimetablePreferencesInput.includeRealTimeCancellations",
preferences.transit().includeRealtimeCancellations()
)
.intReq("TransferPreferencesInput.cost", preferences.transfer().cost())
.intReq(
"TransferPreferencesInput.maximumAdditionalTransfers",
preferences.transfer().maxAdditionalTransfers()
)
// Max transfers are wrong in the internal model but fixed in the API mapping
.intReq(
"TransferPreferencesInput.maximumTransfers",
preferences.transfer().maxTransfers() - 1
)
.stringReq("TransferPreferencesInput.slack", preferences.transfer().slack())
.intReq("WalkPreferencesInput.boardCost", preferences.walk().boardCost())
.floatReq("WalkPreferencesInput.reluctance", preferences.walk().reluctance())
.floatReq("WalkPreferencesInput.safetyFactor", preferences.walk().safetyFactor())
.floatReq("WalkPreferencesInput.speed", preferences.walk().speed())
.boolReq("WheelchairPreferencesInput.enabled", defaultRouteRequest.wheelchair())
.build();
}

private static class DefaultMappingBuilder {

private final Map<String, Value> defaultValueForKey = new HashMap<String, Value>();

public DefaultMappingBuilder intReq(String key, int value) {
defaultValueForKey.put(key, IntValue.of(value));
return this;
}

public DefaultMappingBuilder floatReq(String key, double value) {
defaultValueForKey.put(key, FloatValue.of(value));
return this;
}

public DefaultMappingBuilder stringReq(String key, Object value) {
defaultValueForKey.put(key, StringValue.of(value.toString()));
return this;
}

public DefaultMappingBuilder stringOpt(String key, @Nullable Object value) {
if (value != null) {
defaultValueForKey.put(key, StringValue.of(value.toString()));
}
return this;
}

public DefaultMappingBuilder boolReq(String key, boolean value) {
defaultValueForKey.put(key, BooleanValue.of(value));
return this;
}

public Map<String, Value> build() {
return defaultValueForKey;
}
}
}
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
package org.opentripplanner.apis.gtfs;

import graphql.schema.GraphQLSchema;
import org.opentripplanner.routing.api.RoutingService;
import org.opentripplanner.routing.api.request.RouteRequest;
import org.opentripplanner.routing.fares.FareService;
Expand All @@ -17,6 +18,7 @@ public record GraphQLRequestContext(
VehicleRentalService vehicleRentalService,
VehicleParkingService vehicleParkingService,
RealtimeVehicleService realTimeVehicleService,
GraphQLSchema schema,
GraphFinder graphFinder,
RouteRequest defaultRouteRequest
) {
Expand All @@ -28,6 +30,7 @@ public static GraphQLRequestContext ofServerContext(OtpServerRequestContext cont
context.vehicleRentalService(),
context.vehicleParkingService(),
context.realtimeVehicleService(),
context.schema(),
context.graphFinder(),
context.defaultRouteRequest()
);
Expand Down
Loading
Loading