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

locations.geojson POC phase2: End-to-end partial support of json data #1810

Merged
merged 35 commits into from
Oct 2, 2024
Merged
Show file tree
Hide file tree
Changes from 29 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
f8a966d
add base classes, main broken
davidgamez Aug 27, 2024
204f931
fix FeedMetadata compilation issues
davidgamez Aug 28, 2024
e7751f0
add generated files to gitignore
davidgamez Aug 28, 2024
1131d7d
Merge branch 'master' into poc/json-files
davidgamez Aug 28, 2024
f027e9a
update TableStatus use
davidgamez Aug 28, 2024
2eae957
add json container class
davidgamez Aug 28, 2024
ae500ef
prepare feed container to load json files
davidgamez Aug 28, 2024
acab716
skip abstract classes while initializing the gtfs descriptors
davidgamez Aug 28, 2024
138d550
add empty GtfsJsonFileLoader
davidgamez Aug 28, 2024
e1ea3f1
Corrected a NPE caused by containers missing for files not dataset
jcpitre Aug 29, 2024
435876e
We now have a working POC with a notice if feature id is present in s…
jcpitre Sep 4, 2024
75e8d80
Changed names of some classes
jcpitre Sep 4, 2024
1e9fac5
Reformatting
jcpitre Sep 4, 2024
209d203
Added new notices fields.
jcpitre Sep 4, 2024
b82b39b
Added class comments and renamed some classes.
jcpitre Sep 4, 2024
8bdab6d
Removed some commented out code
jcpitre Sep 4, 2024
6f82f03
Simplified the use of generics got geojson classes
jcpitre Sep 5, 2024
77805f3
Moved code form core to main.
jcpitre Sep 13, 2024
e3dbb2b
Added a validator for the duplicate feature id in locations..
jcpitre Sep 20, 2024
63d03e9
Merge branch 'master' into poc/json-files-phase2
jcpitre Sep 23, 2024
bffef74
add GtfsLocationsSchema and related classes
davidgamez Sep 9, 2024
c90bd7e
Started using the Schema so the rules.hyml will be generated properly.
jcpitre Sep 24, 2024
b794c3a
Added location_group_stops so the UniqueGeographyIdValidator is more …
jcpitre Sep 24, 2024
4badc74
Modified according to PR comments.
jcpitre Sep 26, 2024
2380dae
Merge branch 'master' into poc/json-files-phase2
jcpitre Sep 26, 2024
67504ab
Removed validators and notices not slated for 6.0
jcpitre Sep 27, 2024
3e6845c
Removed foreign key validations on locations_groups_stops since it's …
jcpitre Sep 27, 2024
2a423cc
Clean-up, added comments, removed unused classes.
jcpitre Sep 27, 2024
86ea2c3
Merge branch 'master' into poc/json-files-phase2
jcpitre Sep 27, 2024
63b7d62
Added some tests
jcpitre Sep 30, 2024
361d444
Remove old file with uppercase 'J' from index
jcpitre Oct 1, 2024
1ac48f5
Merge branch 'master' into poc/json-files-phase2
jcpitre Oct 1, 2024
92a9952
Added some comments
jcpitre Oct 2, 2024
ee77597
Merge branch 'master' into poc/json-files-phase2
jcpitre Oct 2, 2024
beac89b
Merge branch 'master' into poc/json-files-phase2
jcpitre Oct 2, 2024
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
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -104,5 +104,6 @@ app/pkg/bin/
processor/notices/bin/
processor/notices/tests/bin/
web/service/bin/
/web/service/execution_result.json

RULES.md
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
import java.util.Optional;
import java.util.TreeMap;
import java.util.logging.Level;
import org.mobilitydata.gtfsvalidator.annotation.GtfsJson;
import org.mobilitydata.gtfsvalidator.annotation.GtfsTable;
import org.mobilitydata.gtfsvalidator.annotation.GtfsValidationNotice;
import org.mobilitydata.gtfsvalidator.annotation.GtfsValidationNotice.SectionRef;
Expand Down Expand Up @@ -125,7 +126,11 @@ public static NoticeDocComments loadComments(Class<?> noticeClass) {
private static ReferencesSchema generateReferences(GtfsValidationNotice noticeAnnotation) {
ReferencesSchema schema = new ReferencesSchema();
Arrays.stream(noticeAnnotation.files().value())
.map(NoticeSchemaGenerator::getFileIdForTableClass)
.map(
fileClass -> {
Optional<String> fileId = getFileIdForTableClass(fileClass);
return fileId.or(() -> getFileIdForJsonClass(fileClass));
})
.flatMap(Optional::stream)
.forEach(schema::addFileReference);
Arrays.stream(noticeAnnotation.bestPractices().value())
Expand All @@ -146,6 +151,11 @@ private static Optional<String> getFileIdForTableClass(Class<? extends GtfsEntit
return Optional.ofNullable(table).map(GtfsTable::value);
}

private static Optional<String> getFileIdForJsonClass(Class<? extends GtfsEntity> entityClass) {
GtfsJson annotation = entityClass.getAnnotation(GtfsJson.class);
return Optional.ofNullable(annotation).map(GtfsJson::value);
}

private static UrlReference convertUrlRef(UrlRef ref) {
return new UrlReference(ref.label(), ref.url());
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,48 +7,44 @@
import com.univocity.parsers.csv.CsvParserSettings;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import org.mobilitydata.gtfsvalidator.notice.CsvParsingFailedNotice;
import org.mobilitydata.gtfsvalidator.notice.EmptyFileNotice;
import org.mobilitydata.gtfsvalidator.notice.MissingRecommendedFileNotice;
import org.mobilitydata.gtfsvalidator.notice.MissingRequiredFileNotice;
import org.mobilitydata.gtfsvalidator.notice.NoticeContainer;
import org.mobilitydata.gtfsvalidator.parsing.CsvFile;
import org.mobilitydata.gtfsvalidator.parsing.CsvHeader;
import org.mobilitydata.gtfsvalidator.parsing.CsvRow;
import org.mobilitydata.gtfsvalidator.parsing.FieldCache;
import org.mobilitydata.gtfsvalidator.parsing.RowParser;
import org.mobilitydata.gtfsvalidator.validator.FileValidator;
import org.mobilitydata.gtfsvalidator.validator.SingleEntityValidator;
import org.mobilitydata.gtfsvalidator.validator.ValidatorProvider;
import org.mobilitydata.gtfsvalidator.validator.ValidatorUtil;

public final class AnyTableLoader {
/** This class loads csv files specifically. */
public final class CsvFileLoader extends TableLoader {

private static final FluentLogger logger = FluentLogger.forEnclosingClass();
private static final List<Class<? extends FileValidator>> singleFileValidatorsWithParsingErrors =
new ArrayList<>();
private CsvFileLoader() {}
// Create the singleton and add a method to obtain it
private static final CsvFileLoader INSTANCE = new CsvFileLoader();

private static final List<Class<? extends SingleEntityValidator>>
singleEntityValidatorsWithParsingErrors = new ArrayList<>();

public List<Class<? extends FileValidator>> getValidatorsWithParsingErrors() {
return Collections.unmodifiableList(singleFileValidatorsWithParsingErrors);
@Nonnull
public static CsvFileLoader getInstance() {
return INSTANCE;
}

public List<Class<? extends SingleEntityValidator>> getSingleEntityValidatorsWithParsingErrors() {
return Collections.unmodifiableList(singleEntityValidatorsWithParsingErrors);
}
private final FluentLogger logger = FluentLogger.forEnclosingClass();

public static GtfsTableContainer load(
GtfsTableDescriptor tableDescriptor,
@Override
public GtfsEntityContainer<?, ?> load(
GtfsFileDescriptor fileDescriptor,
ValidatorProvider validatorProvider,
InputStream csvInputStream,
NoticeContainer noticeContainer) {
GtfsTableDescriptor tableDescriptor = (GtfsTableDescriptor) fileDescriptor;
final String gtfsFilename = tableDescriptor.gtfsFilename();

CsvFile csvFile;
Expand All @@ -61,22 +57,19 @@ public static GtfsTableContainer load(
csvFile = new CsvFile(csvInputStream, gtfsFilename, settings);
} catch (TextParsingException e) {
noticeContainer.addValidationNotice(new CsvParsingFailedNotice(gtfsFilename, e));
return tableDescriptor.createContainerForInvalidStatus(
GtfsTableContainer.TableStatus.INVALID_HEADERS);
return tableDescriptor.createContainerForInvalidStatus(TableStatus.INVALID_HEADERS);
}
if (csvFile.isEmpty()) {
noticeContainer.addValidationNotice(new EmptyFileNotice(gtfsFilename));
return tableDescriptor.createContainerForInvalidStatus(
GtfsTableContainer.TableStatus.EMPTY_FILE);
return tableDescriptor.createContainerForInvalidStatus(TableStatus.EMPTY_FILE);
}
final CsvHeader header = csvFile.getHeader();
final ImmutableList<GtfsColumnDescriptor> columnDescriptors = tableDescriptor.getColumns();
final NoticeContainer headerNotices =
validateHeaders(validatorProvider, gtfsFilename, header, columnDescriptors);
noticeContainer.addAll(headerNotices);
if (headerNotices.hasValidationErrors()) {
return tableDescriptor.createContainerForInvalidStatus(
GtfsTableContainer.TableStatus.INVALID_HEADERS);
return tableDescriptor.createContainerForInvalidStatus(TableStatus.INVALID_HEADERS);
}
final int nColumns = columnDescriptors.size();
final ImmutableMap<String, GtfsFieldLoader> fieldLoadersMap = tableDescriptor.getFieldLoaders();
Expand All @@ -99,8 +92,8 @@ public static GtfsTableContainer load(
final List<GtfsEntity> entities = new ArrayList<>();
boolean hasUnparsableRows = false;
final List<SingleEntityValidator<GtfsEntity>> singleEntityValidators =
validatorProvider.createSingleEntityValidators(
tableDescriptor.getEntityClass(), singleEntityValidatorsWithParsingErrors::add);
createSingleEntityValidators(tableDescriptor.getEntityClass(), validatorProvider);

try {
for (CsvRow row : csvFile) {
if (row.getRowNumber() % 200000 == 0) {
Expand Down Expand Up @@ -133,26 +126,23 @@ public static GtfsTableContainer load(
}
} catch (TextParsingException e) {
noticeContainer.addValidationNotice(new CsvParsingFailedNotice(gtfsFilename, e));
return tableDescriptor.createContainerForInvalidStatus(
GtfsTableContainer.TableStatus.UNPARSABLE_ROWS);
return tableDescriptor.createContainerForInvalidStatus(TableStatus.UNPARSABLE_ROWS);
} finally {
logFieldCacheStats(gtfsFilename, fieldCaches, columnDescriptors);
}
if (hasUnparsableRows) {
logger.atSevere().log("Failed to parse some rows in %s", gtfsFilename);
return tableDescriptor.createContainerForInvalidStatus(
GtfsTableContainer.TableStatus.UNPARSABLE_ROWS);
return tableDescriptor.createContainerForInvalidStatus(TableStatus.UNPARSABLE_ROWS);
}
GtfsTableContainer table =
tableDescriptor.createContainerForHeaderAndEntities(header, entities, noticeContainer);

ValidatorUtil.invokeSingleFileValidators(
validatorProvider.createSingleFileValidators(
table, singleFileValidatorsWithParsingErrors::add),
noticeContainer);
createSingleFileValidators(table, validatorProvider), noticeContainer);
return table;
}

private static NoticeContainer validateHeaders(
private NoticeContainer validateHeaders(
ValidatorProvider validatorProvider,
String gtfsFilename,
CsvHeader header,
Expand All @@ -178,7 +168,7 @@ private static NoticeContainer validateHeaders(
return headerNotices;
}

private static void logFieldCacheStats(
private void logFieldCacheStats(
String gtfsFilename,
FieldCache[] fieldCaches,
ImmutableList<GtfsColumnDescriptor> columnDescriptors) {
Expand All @@ -196,25 +186,4 @@ private static void logFieldCacheStats(
}
}
}

public static GtfsTableContainer loadMissingFile(
GtfsTableDescriptor tableDescriptor,
ValidatorProvider validatorProvider,
NoticeContainer noticeContainer) {
String gtfsFilename = tableDescriptor.gtfsFilename();
GtfsTableContainer table =
tableDescriptor.createContainerForInvalidStatus(
GtfsTableContainer.TableStatus.MISSING_FILE);
if (tableDescriptor.isRecommended()) {
noticeContainer.addValidationNotice(new MissingRecommendedFileNotice(gtfsFilename));
}
if (tableDescriptor.isRequired()) {
noticeContainer.addValidationNotice(new MissingRequiredFileNotice(gtfsFilename));
}
ValidatorUtil.invokeSingleFileValidators(
validatorProvider.createSingleFileValidators(
table, singleFileValidatorsWithParsingErrors::add),
noticeContainer);
return table;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
package org.mobilitydata.gtfsvalidator.table;

import java.util.List;
import java.util.Optional;

/**
* This class is the parent of containers holding table (csv) entities and containers holding JSON
* entities
*
* @param <T> The entity for this container (e.g. GtfsCalendarDate or GtfsGeojsonFeature )
* @param <D> The descriptor for the file for the container (e.g. GtfsCalendarDateTableDescriptor or
* GtfsGeojsonFileDescriptor)
*/
public abstract class GtfsEntityContainer<T extends GtfsEntity, D extends GtfsFileDescriptor> {

private final D descriptor;
private final TableStatus tableStatus;

public GtfsEntityContainer(D descriptor, TableStatus tableStatus) {
this.tableStatus = tableStatus;
this.descriptor = descriptor;
}

public TableStatus getTableStatus() {
return tableStatus;
}

public D getDescriptor() {
return descriptor;
}

public abstract Class<T> getEntityClass();

public int entityCount() {
return getEntities().size();
}

public abstract List<T> getEntities();

public abstract String gtfsFilename();

public abstract Optional<T> byTranslationKey(String recordId, String recordSubId);

public boolean isMissingFile() {
return tableStatus == TableStatus.MISSING_FILE;
}

public boolean isParsedSuccessfully() {
switch (tableStatus) {
case PARSABLE_HEADERS_AND_ROWS:
return true;
case MISSING_FILE:
return !descriptor.isRequired();
default:
return false;
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -18,20 +18,19 @@

import com.google.common.base.Ascii;
import java.util.*;
import org.mobilitydata.gtfsvalidator.table.GtfsTableContainer.TableStatus;

/**
* Container for a whole parsed GTFS feed with all its tables.
*
* <p>The tables are kept as {@code GtfsTableContainer} instances.
* <p>The tables are kept as {@link GtfsEntityContainer} instances.
*/
public class GtfsFeedContainer {
private final Map<String, GtfsTableContainer<?>> tables = new HashMap<>();
private final Map<Class<? extends GtfsTableContainer>, GtfsTableContainer<?>> tablesByClass =
private final Map<String, GtfsEntityContainer<?, ?>> tables = new HashMap<>();
private final Map<Class<? extends GtfsEntityContainer>, GtfsEntityContainer<?, ?>> tablesByClass =
new HashMap<>();

public GtfsFeedContainer(List<GtfsTableContainer<?>> tableContainerList) {
for (GtfsTableContainer<?> table : tableContainerList) {
public GtfsFeedContainer(List<GtfsEntityContainer<?, ?>> tableContainerList) {
for (GtfsEntityContainer<?, ?> table : tableContainerList) {
tables.put(table.gtfsFilename(), table);
tablesByClass.put(table.getClass(), table);
}
Expand All @@ -49,11 +48,12 @@ public GtfsFeedContainer(List<GtfsTableContainer<?>> tableContainerList) {
* @param filename file name, including ".txt" extension
* @return GTFS table or empty if the table is not supported by schema
*/
public Optional<GtfsTableContainer<?>> getTableForFilename(String filename) {
return Optional.ofNullable(tables.getOrDefault(Ascii.toLowerCase(filename), null));
public <T extends GtfsEntityContainer<?, ?>> Optional<T> getTableForFilename(String filename) {
return (Optional<T>)
Optional.ofNullable(tables.getOrDefault(Ascii.toLowerCase(filename), null));
}

public <T extends GtfsTableContainer<?>> T getTable(Class<T> clazz) {
public <T extends GtfsEntityContainer<?, ?>> T getTable(Class<T> clazz) {
return (T) tablesByClass.get(clazz);
}

Expand All @@ -65,21 +65,21 @@ public <T extends GtfsTableContainer<?>> T getTable(Class<T> clazz) {
* @return true if all files were successfully parsed, false otherwise
*/
public boolean isParsedSuccessfully() {
for (GtfsTableContainer<?> table : tables.values()) {
for (GtfsEntityContainer<?, ?> table : tables.values()) {
if (!table.isParsedSuccessfully()) {
return false;
}
}
return true;
}

public Collection<GtfsTableContainer<?>> getTables() {
public Collection<GtfsEntityContainer<?, ?>> getTables() {
return tables.values();
}

public String tableTotalsText() {
List<String> totalList = new ArrayList<>();
for (GtfsTableContainer<?> table : tables.values()) {
for (GtfsEntityContainer<?, ?> table : tables.values()) {
if (table.getTableStatus() == TableStatus.MISSING_FILE
&& !table.getDescriptor().isRequired()) {
continue;
Expand Down
Loading
Loading