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

Feat: 1640 add remaining gtfs features to the validator #1656

Merged
merged 15 commits into from
Jan 24, 2024
Merged
Show file tree
Hide file tree
Changes from 14 commits
Commits
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
7 changes: 7 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -97,3 +97,10 @@ web/service/src/main/resources/application.properties
web/service/newrelic/newrelic.yml

processor/tests/bin

.vscode/
app/gui/bin/
app/pkg/bin/
processor/notices/bin/
processor/notices/tests/bin/
web/service/bin/
Original file line number Diff line number Diff line change
Expand Up @@ -43,17 +43,22 @@ public class FeedMetadata {

private final List<Pair<String, String>> FILE_BASED_COMPONENTS =
List.of(
new Pair<>("Pathways", GtfsPathway.FILENAME),
new Pair<>("Pathways (basic)", GtfsPathway.FILENAME),
new Pair<>("Pathways (extra)", GtfsPathway.FILENAME),
new Pair<>("Transfers", GtfsTransfer.FILENAME),
new Pair<>("Fares V1", GtfsFareAttribute.FILENAME),
new Pair<>("Fare Products", GtfsFareProduct.FILENAME),
new Pair<>("Shapes", GtfsShape.FILENAME),
new Pair<>("Frequency-Based Trip", GtfsFrequency.FILENAME),
new Pair<>("Frequencies", GtfsFrequency.FILENAME),
new Pair<>("Feed Information", GtfsFeedInfo.FILENAME),
new Pair<>("Attributions", GtfsAttribution.FILENAME),
new Pair<>("Translations", GtfsTranslation.FILENAME),
new Pair<>("Fare Media", GtfsFareMedia.FILENAME),
new Pair<>("Zone-Based Fares", GtfsStopArea.FILENAME));
new Pair<>("Zone-Based Fares", GtfsArea.FILENAME),
// new Pair<>("Route-Based Fares", GtfsNetwork.FILENAME),
new Pair<>("Transfer Fares", GtfsFareTransferRule.FILENAME),
new Pair<>("Time-Based Fares", GtfsTimeframe.FILENAME),
new Pair<>("Levels", GtfsLevel.FILENAME));

protected FeedMetadata() {}

Expand Down Expand Up @@ -148,9 +153,14 @@ private void loadSpecFeaturesBasedOnFieldPresence(GtfsFeedContainer feedContaine
loadLocationTypesComponent(feedContainer);
loadTraversalTimeComponent(feedContainer);
loadPathwayDirectionsComponent(feedContainer);
loadPathwayExtraComponent(feedContainer);
loadBlocksComponent(feedContainer);
loadRouteBasedFaresComponent(feedContainer);
loadContinuousStopsComponent(feedContainer);
loadZoneBasedComponent(feedContainer);
loadTimeBasedFaresComponent(feedContainer);
loadTransferFaresComponent(feedContainer);
loadLevelsComponent(feedContainer);
}

private void loadContinuousStopsComponent(GtfsFeedContainer feedContainer) {
Expand Down Expand Up @@ -183,10 +193,12 @@ private void loadRouteBasedFaresComponent(GtfsFeedContainer feedContainer) {
List.of(
GtfsFareLegRule::hasFromAreaId,
(Function<GtfsFareLegRule, Boolean>) GtfsFareLegRule::hasToAreaId))
&& hasAtLeastOneRecordForFields(
&& (hasAtLeastOneRecordForFields(
qcdyx marked this conversation as resolved.
Show resolved Hide resolved
feedContainer,
GtfsRoute.FILENAME,
List.of((Function<GtfsRoute, Boolean>) GtfsRoute::hasNetworkId)));
List.of((Function<GtfsRoute, Boolean>) GtfsRoute::hasNetworkId))
// || hasAtLeastOneRecordInFile(feedContainer, GtfsNetwork.FILENAME)
));
}

private void loadBlocksComponent(GtfsFeedContainer feedContainer) {
Expand All @@ -209,6 +221,27 @@ private void loadPathwayDirectionsComponent(GtfsFeedContainer feedContainer) {
(Function<GtfsPathway, Boolean>) GtfsPathway::hasReversedSignpostedAs)));
}

private void loadPathwayExtraComponent(GtfsFeedContainer feedContainer) {
specFeatures.put(
"Pathway (extra)",
hasAtLeastOneRecordForFields(
qcdyx marked this conversation as resolved.
Show resolved Hide resolved
feedContainer,
GtfsPathway.FILENAME,
List.of((Function<GtfsPathway, Boolean>) GtfsPathway::hasMaxSlope))
|| hasAtLeastOneRecordForFields(
feedContainer,
GtfsPathway.FILENAME,
List.of((Function<GtfsPathway, Boolean>) GtfsPathway::hasMinWidth))
|| hasAtLeastOneRecordForFields(
feedContainer,
GtfsPathway.FILENAME,
List.of((Function<GtfsPathway, Boolean>) GtfsPathway::hasLength))
|| hasAtLeastOneRecordForFields(
feedContainer,
GtfsPathway.FILENAME,
List.of((Function<GtfsPathway, Boolean>) GtfsPathway::hasStairCount)));
}

private void loadTraversalTimeComponent(GtfsFeedContainer feedContainer) {
specFeatures.put(
"Traversal Time",
Expand Down Expand Up @@ -306,6 +339,25 @@ private void loadRouteNamesComponent(GtfsFeedContainer feedContainer) {
(Function<GtfsRoute, Boolean>) GtfsRoute::hasRouteLongName)));
}

private void loadZoneBasedComponent(GtfsFeedContainer feedContainer) {
specFeatures.put(
"Zone-Based Fares", hasAtLeastOneRecordInFile(feedContainer, GtfsArea.FILENAME));
}

private void loadTimeBasedFaresComponent(GtfsFeedContainer feedContainer) {
specFeatures.put(
"Time-Based Fares", hasAtLeastOneRecordInFile(feedContainer, GtfsTimeframe.FILENAME));
}

private void loadTransferFaresComponent(GtfsFeedContainer feedContainer) {
specFeatures.put(
"Transfer Fares", hasAtLeastOneRecordInFile(feedContainer, GtfsFareTransferRule.FILENAME));
}

private void loadLevelsComponent(GtfsFeedContainer feedContainer) {
specFeatures.put("Levels", hasAtLeastOneRecordInFile(feedContainer, GtfsLevel.FILENAME));
}

private void loadAgencyData(GtfsTableContainer<GtfsAgency> agencyTable) {
for (GtfsAgency agency : agencyTable.getEntities()) {
agencies.add(AgencyMetadata.from(agency));
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
// package org.mobilitydata.gtfsvalidator.table;
//
// import org.mobilitydata.gtfsvalidator.annotation.*;
//
// @GtfsTable("networks.txt")
// @Required
// public interface GtfsNetworkSchema extends GtfsEntity {
// @FieldType(FieldTypeEnum.ID)
// @PrimaryKey
// @Required
// String networkId();
//
// @MixedCase
// String networkName();
// }
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,6 @@ public class FareAttributeAgencyIdValidator extends FileValidator {

@Override
public void validate(NoticeContainer noticeContainer) {

// routes.agency_id is required when there are multiple agencies
int totalAgencies = agencyTable.entityCount();

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,7 @@ public void validate(NoticeContainer noticeContainer) {
// A multilanguage feed may have different agency_lang.
return;
}

for (GtfsAgency agency : agencyTable.getEntities()) {
if (agency.hasAgencyLang() && !feedLang.equals(agency.agencyLang())) {
noticeContainer.addValidationNotice(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,6 @@ public class RouteAgencyIdValidator extends FileValidator {

@Override
public void validate(NoticeContainer noticeContainer) {

// routes.agency_id is required when there are multiple agencies
int totalAgencies = agencyTable.entityCount();

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -227,7 +227,7 @@ public void containsPathwaysComponentTest() throws IOException, InterruptedExcep
+ "pathway2,stop2,stop3,2,0\n";
createDataFile("pathways.txt", pathwayContent);
validateSpecFeature(
"Pathways",
"Pathways (basic)",
true,
ImmutableList.of(GtfsPathwayTableDescriptor.class, GtfsAgencyTableDescriptor.class));
}
Expand All @@ -237,15 +237,15 @@ public void omitsPathwaysComponentTest() throws IOException, InterruptedExceptio
String pathwayContent = "pathway_id,from_stop_id,to_stop_id,pathway_mode,is_bidirectional\n";
createDataFile("pathways.txt", pathwayContent);
validateSpecFeature(
"Pathways",
"Pathways (basic)",
false,
ImmutableList.of(GtfsPathwayTableDescriptor.class, GtfsAgencyTableDescriptor.class));
}

@Test
public void omitsComponents() throws IOException, InterruptedException {
validateSpecFeature(
"Pathways",
"Pathways (basic)",
false,
ImmutableList.of(GtfsPathwayTableDescriptor.class, GtfsAgencyTableDescriptor.class));
validateSpecFeature(
Expand Down Expand Up @@ -304,7 +304,7 @@ public void containsFrequencyBasedTripComponentTest() throws IOException, Interr
"trip_id, start_time, end_time, headway_secs\n" + "dummy1, 01:01:01, 01:01:02, 1\n";
createDataFile(GtfsFrequency.FILENAME, content);
validateSpecFeature(
"Frequency-Based Trip",
"Frequencies",
true,
ImmutableList.of(GtfsFrequencyTableDescriptor.class, GtfsAgencyTableDescriptor.class));
}
Expand All @@ -314,7 +314,7 @@ public void omitsFrequencyBasedTripComponentTest() throws IOException, Interrupt
String content = "trip_id, start_time, end_time, headway_secs\n";
createDataFile(GtfsFrequency.FILENAME, content);
validateSpecFeature(
"Frequency-Based Trip",
"Frequencies",
false,
ImmutableList.of(GtfsFrequencyTableDescriptor.class, GtfsAgencyTableDescriptor.class));
}
Expand Down Expand Up @@ -406,11 +406,11 @@ public void omitsFareMediaComponentTest() throws IOException, InterruptedExcepti
@Test
public void containsZoneBasedFaresComponentTest() throws IOException, InterruptedException {
String content = "area_id, stop_id\n" + "dummyArea, dummyStop\n";
createDataFile(GtfsStopArea.FILENAME, content);
createDataFile(GtfsArea.FILENAME, content);
validateSpecFeature(
"Zone-Based Fares",
true,
ImmutableList.of(GtfsStopAreaTableDescriptor.class, GtfsAgencyTableDescriptor.class));
ImmutableList.of(GtfsAreaTableDescriptor.class, GtfsAgencyTableDescriptor.class));
}

@Test
Expand Down
Loading