Compare commits
25 commits
updated-re
...
master
Author | SHA1 | Date | |
---|---|---|---|
|
ab59bd57fb | ||
|
65cab1b379 | ||
|
ebe9cf219c | ||
|
b7f7abaaca | ||
|
e833ae1d35 | ||
|
c0f696fc52 | ||
|
661b9dc43b | ||
|
ef0c8946b1 | ||
|
c43b6032d8 | ||
|
574da01e12 | ||
|
646bb13f39 | ||
|
b07393991a | ||
|
f0b74c210e | ||
|
a74b71af50 | ||
|
5a6590533c | ||
|
d93723861e | ||
|
574b946dce | ||
|
f5c932d16a | ||
|
cc3d82a25e | ||
|
c3804b0d98 | ||
|
46996effa6 | ||
|
5505cb60a9 | ||
|
d0ac9a3502 | ||
|
b882d0620f | ||
|
34da27df6d |
8 changed files with 192 additions and 26 deletions
34
.github/workflows/cla.yml
vendored
Normal file
34
.github/workflows/cla.yml
vendored
Normal file
|
@ -0,0 +1,34 @@
|
|||
name: "CLA Assistant"
|
||||
on:
|
||||
issue_comment:
|
||||
types: [created]
|
||||
pull_request_target:
|
||||
types: [opened,closed,synchronize]
|
||||
|
||||
jobs:
|
||||
CLAssistant:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: "CLA Assistant"
|
||||
if: (github.event.comment.body == 'recheck' || github.event.comment.body == 'I have read the CLA Document and I hereby sign the CLA') || github.event_name == 'pull_request_target'
|
||||
# Alpha Release
|
||||
uses: cla-assistant/github-action@v2.1.0-alpha
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
PERSONAL_ACCESS_TOKEN : ${{ secrets.PERSONAL_ACCESS_TOKEN }}
|
||||
with:
|
||||
path-to-signatures: 'signatures/version1/cla.json'
|
||||
path-to-document: 'https://github.com/mapsme/just_gtfs/blob/master/MIT_CLA.md'
|
||||
# branch should not be protected
|
||||
branch: 'master'
|
||||
allowlist: mesozoic-drones,tatiana-yan,bot*
|
||||
|
||||
#below are the optional inputs - If the optional inputs are not given, then default values will be taken
|
||||
#remote-organization-name: enter the remote organization name where the signatures should be stored (Default is storing the signatures in the same repository)
|
||||
#remote-repository-name: enter the remote repository name where the signatures should be stored (Default is storing the signatures in the same repository)
|
||||
#create-file-commit-message: 'For example: Creating file for storing CLA Signatures'
|
||||
#signed-commit-message: 'For example: $contributorName has signed the CLA in #$pullRequestNo'
|
||||
#custom-notsigned-prcomment: 'pull request comment with Introductory message to ask new contributors to sign'
|
||||
#custom-pr-sign-comment: 'The signature to be committed in order to sign the CLA'
|
||||
#custom-allsigned-prcomment: 'pull request comment when all contributors has signed, defaults to **CLA Assistant Lite bot** All Contributors have signed the CLA.'
|
||||
|
|
@ -11,5 +11,8 @@ set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -Wextra -Werror")
|
|||
|
||||
enable_testing()
|
||||
|
||||
add_library(just_gtfs INTERFACE)
|
||||
target_include_directories(just_gtfs INTERFACE ${CMAKE_CURRENT_LIST_DIR}/include)
|
||||
|
||||
add_subdirectory(tests)
|
||||
add_subdirectory(benchmarks)
|
||||
|
|
59
MIT_CLA.md
Normal file
59
MIT_CLA.md
Normal file
|
@ -0,0 +1,59 @@
|
|||
## Individual Contributor License Agreement (CLA)
|
||||
|
||||
**Thank you for submitting your contributions to this project.**
|
||||
|
||||
By signing this CLA, you agree that the following terms apply to all of your past, present and future contributions
|
||||
to the project.
|
||||
|
||||
### License.
|
||||
|
||||
You hereby represent that all present, past and future contributions are governed by the
|
||||
[MIT License](https://opensource.org/licenses/MIT)
|
||||
copyright statement.
|
||||
|
||||
This entails that to the extent possible under law, you transfer all copyright and related or neighboring rights
|
||||
of the code or documents you contribute to the project itself or its maintainers.
|
||||
Furthermore you also represent that you have the authority to perform the above waiver
|
||||
with respect to the entirety of you contributions.
|
||||
|
||||
### Moral Rights.
|
||||
|
||||
To the fullest extent permitted under applicable law, you hereby waive, and agree not to
|
||||
assert, all of your “moral rights” in or relating to your contributions for the benefit of the project.
|
||||
|
||||
### Third Party Content.
|
||||
|
||||
If your Contribution includes or is based on any source code, object code, bug fixes, configuration changes, tools,
|
||||
specifications, documentation, data, materials, feedback, information or other works of authorship that were not
|
||||
authored by you (“Third Party Content”) or if you are aware of any third party intellectual property or proprietary
|
||||
rights associated with your Contribution (“Third Party Rights”),
|
||||
then you agree to include with the submission of your Contribution full details respecting such Third Party
|
||||
Content and Third Party Rights, including, without limitation, identification of which aspects of your
|
||||
Contribution contain Third Party Content or are associated with Third Party Rights, the owner/author of the
|
||||
Third Party Content and Third Party Rights, where you obtained the Third Party Content, and any applicable
|
||||
third party license terms or restrictions respecting the Third Party Content and Third Party Rights. For greater
|
||||
certainty, the foregoing obligations respecting the identification of Third Party Content and Third Party Rights
|
||||
do not apply to any portion of a Project that is incorporated into your Contribution to that same Project.
|
||||
|
||||
### Representations.
|
||||
|
||||
You represent that, other than the Third Party Content and Third Party Rights identified by
|
||||
you in accordance with this Agreement, you are the sole author of your Contributions and are legally entitled
|
||||
to grant the foregoing licenses and waivers in respect of your Contributions. If your Contributions were
|
||||
created in the course of your employment with your past or present employer(s), you represent that such
|
||||
employer(s) has authorized you to make your Contributions on behalf of such employer(s) or such employer
|
||||
(s) has waived all of their right, title or interest in or to your Contributions.
|
||||
|
||||
### Disclaimer.
|
||||
|
||||
To the fullest extent permitted under applicable law, your Contributions are provided on an "as is"
|
||||
basis, without any warranties or conditions, express or implied, including, without limitation, any implied
|
||||
warranties or conditions of non-infringement, merchantability or fitness for a particular purpose. You are not
|
||||
required to provide support for your Contributions, except to the extent you desire to provide support.
|
||||
|
||||
### No Obligation.
|
||||
|
||||
You acknowledge that the maintainers of this project are under no obligation to use or incorporate your contributions
|
||||
into the project. The decision to use or incorporate your contributions into the project will be made at the
|
||||
sole discretion of the maintainers or their authorized delegates.
|
||||
|
|
@ -5,6 +5,7 @@
|
|||
[](https://shields.io/)
|
||||
[](https://lbesson.mit-license.org/)
|
||||

|
||||
[](https://github.com/CUTR-at-USF/awesome-transit)
|
||||
[](https://github.com/mapsme/just_gtfs/issues)
|
||||
|
||||
|
||||
|
|
|
@ -455,6 +455,7 @@ public:
|
|||
inline Time() = default;
|
||||
inline explicit Time(const std::string & raw_time_str);
|
||||
inline Time(uint16_t hours, uint16_t minutes, uint16_t seconds);
|
||||
inline Time(size_t seconds);
|
||||
inline bool is_provided() const;
|
||||
inline size_t get_total_seconds() const;
|
||||
inline std::tuple<uint16_t, uint16_t, uint16_t> get_hh_mm_ss() const;
|
||||
|
@ -516,8 +517,8 @@ inline Time::Time(const std::string & raw_time_str) : raw_time(raw_time_str)
|
|||
return;
|
||||
|
||||
const size_t len = raw_time.size();
|
||||
if (!(len == 7 || len == 8) || (raw_time[len - 3] != ':' && raw_time[len - 6] != ':'))
|
||||
throw InvalidFieldFormat("Time is not in [H]H:MM:SS format: " + raw_time_str);
|
||||
if (!(len >= 7 && len <= 9) || raw_time[len - 3] != ':' || raw_time[len - 6] != ':')
|
||||
throw InvalidFieldFormat("Time is not in [[H]H]H:MM:SS format: " + raw_time_str);
|
||||
|
||||
hh = static_cast<uint16_t>(std::stoi(raw_time.substr(0, len - 6)));
|
||||
mm = static_cast<uint16_t>(std::stoi(raw_time.substr(len - 5, 2)));
|
||||
|
@ -543,6 +544,13 @@ inline Time::Time(uint16_t hours, uint16_t minutes, uint16_t seconds)
|
|||
time_is_provided = true;
|
||||
}
|
||||
|
||||
inline Time::Time(size_t seconds)
|
||||
: time_is_provided(true), total_seconds(seconds),
|
||||
hh(seconds / 3600), mm((seconds % 3600) / 60), ss(seconds % 3600)
|
||||
{
|
||||
set_raw_time();
|
||||
}
|
||||
|
||||
inline bool Time::is_provided() const { return time_is_provided; }
|
||||
|
||||
inline size_t Time::get_total_seconds() const { return total_seconds; }
|
||||
|
@ -644,7 +652,7 @@ using CurrencyCode = std::string;
|
|||
using LanguageCode = std::string;
|
||||
|
||||
// Helper enums for some GTFS fields ---------------------------------------------------------------
|
||||
enum class StopLocationType
|
||||
enum class StopLocationType : int8_t
|
||||
{
|
||||
StopOrPlatform = 0,
|
||||
Station = 1,
|
||||
|
@ -654,7 +662,7 @@ enum class StopLocationType
|
|||
};
|
||||
|
||||
// The type of transportation used on a route.
|
||||
enum class RouteType
|
||||
enum class RouteType : int16_t
|
||||
{
|
||||
// GTFS route types
|
||||
Tram = 0, // Tram, Streetcar, Light rail
|
||||
|
@ -752,20 +760,20 @@ enum class RouteType
|
|||
HorseDrawnCarriage = 1702
|
||||
};
|
||||
|
||||
enum class TripDirectionId
|
||||
enum class TripDirectionId : bool
|
||||
{
|
||||
DefaultDirection = 0, // e.g. outbound
|
||||
OppositeDirection = 1 // e.g. inbound
|
||||
};
|
||||
|
||||
enum class TripAccess
|
||||
enum class TripAccess : int8_t
|
||||
{
|
||||
NoInfo = 0,
|
||||
Yes = 1,
|
||||
No = 2
|
||||
};
|
||||
|
||||
enum class StopTimeBoarding
|
||||
enum class StopTimeBoarding : int8_t
|
||||
{
|
||||
RegularlyScheduled = 0,
|
||||
No = 1, // Not available
|
||||
|
@ -773,31 +781,31 @@ enum class StopTimeBoarding
|
|||
CoordinateWithDriver = 3 // Must coordinate with driver to arrange
|
||||
};
|
||||
|
||||
enum class StopTimePoint
|
||||
enum class StopTimePoint : bool
|
||||
{
|
||||
Approximate = 0,
|
||||
Exact = 1
|
||||
};
|
||||
|
||||
enum class CalendarAvailability
|
||||
enum class CalendarAvailability : bool
|
||||
{
|
||||
NotAvailable = 0,
|
||||
Available = 1
|
||||
};
|
||||
|
||||
enum class CalendarDateException
|
||||
enum class CalendarDateException : int8_t
|
||||
{
|
||||
Added = 1, // Service has been added for the specified date
|
||||
Removed = 2
|
||||
};
|
||||
|
||||
enum class FarePayment
|
||||
enum class FarePayment : bool
|
||||
{
|
||||
OnBoard = 0,
|
||||
BeforeBoarding = 1 // Fare must be paid before boarding
|
||||
};
|
||||
|
||||
enum class FareTransfers
|
||||
enum class FareTransfers : int8_t
|
||||
{
|
||||
No = 0, // No transfers permitted on this fare
|
||||
Once = 1,
|
||||
|
@ -805,13 +813,13 @@ enum class FareTransfers
|
|||
Unlimited = 3
|
||||
};
|
||||
|
||||
enum class FrequencyTripService
|
||||
enum class FrequencyTripService : bool
|
||||
{
|
||||
FrequencyBased = 0, // Frequency-based trips
|
||||
ScheduleBased = 1 // Schedule-based trips with the exact same headway throughout the day
|
||||
};
|
||||
|
||||
enum class TransferType
|
||||
enum class TransferType : int8_t
|
||||
{
|
||||
Recommended = 0,
|
||||
Timed = 1,
|
||||
|
@ -819,7 +827,7 @@ enum class TransferType
|
|||
NotPossible = 3
|
||||
};
|
||||
|
||||
enum class PathwayMode
|
||||
enum class PathwayMode : int8_t
|
||||
{
|
||||
Walkway = 1,
|
||||
Stairs = 2,
|
||||
|
@ -830,13 +838,13 @@ enum class PathwayMode
|
|||
ExitGate = 7
|
||||
};
|
||||
|
||||
enum class PathwayDirection
|
||||
enum class PathwayDirection : bool
|
||||
{
|
||||
Unidirectional = 0,
|
||||
Bidirectional = 1
|
||||
};
|
||||
|
||||
enum class AttributionRole
|
||||
enum class AttributionRole : bool
|
||||
{
|
||||
No = 0, // Organization doesn’t have this role
|
||||
Yes = 1 // Organization does have this role
|
||||
|
@ -888,7 +896,7 @@ struct Stop
|
|||
Text stop_code;
|
||||
Text stop_desc;
|
||||
Text stop_url;
|
||||
StopLocationType location_type = StopLocationType::GenericNode;
|
||||
StopLocationType location_type = StopLocationType::StopOrPlatform;
|
||||
Text stop_timezone;
|
||||
Text wheelchair_boarding;
|
||||
Id level_id;
|
||||
|
@ -999,6 +1007,14 @@ struct FareAttributesItem
|
|||
size_t transfer_duration = 0; // Length of time in seconds before a transfer expires
|
||||
};
|
||||
|
||||
inline bool operator==(const FareAttributesItem & lhs, const FareAttributesItem & rhs)
|
||||
{
|
||||
return std::tie(lhs.fare_id, lhs.price, lhs.currency_type, lhs.payment_method, lhs.transfers,
|
||||
lhs.agency_id, lhs.transfer_duration) ==
|
||||
std::tie(rhs.fare_id, rhs.price, rhs.currency_type, rhs.payment_method, rhs.transfers,
|
||||
rhs.agency_id, rhs.transfer_duration);
|
||||
}
|
||||
|
||||
// Optional dataset file
|
||||
struct FareRule
|
||||
{
|
||||
|
@ -1340,6 +1356,7 @@ private:
|
|||
inline void write_translations(std::ofstream & out) const;
|
||||
inline void write_attributions(std::ofstream & out) const;
|
||||
|
||||
protected:
|
||||
std::string gtfs_directory;
|
||||
|
||||
Agencies agencies;
|
||||
|
@ -1840,7 +1857,7 @@ inline Result Feed::add_fare_attributes(const ParsedCsvRow & row)
|
|||
|
||||
item.currency_type = row.at("currency_type");
|
||||
set_field(item.payment_method, row, "payment_method", false);
|
||||
set_field(item.transfers, row, "transfers", false);
|
||||
set_field(item.transfers, row, "transfers");
|
||||
|
||||
// Conditionally optional:
|
||||
item.agency_id = get_value_or_default(row, "agency_id");
|
||||
|
@ -2778,9 +2795,12 @@ inline void Feed::write_fare_attributes(std::ofstream & out) const
|
|||
for (const auto & attribute : fare_attributes)
|
||||
{
|
||||
std::vector<std::string> fields{
|
||||
wrap(attribute.fare_id), wrap(attribute.price), attribute.currency_type,
|
||||
wrap(attribute.payment_method), wrap(attribute.transfers), wrap(attribute.agency_id),
|
||||
wrap(attribute.transfer_duration)};
|
||||
wrap(attribute.fare_id), wrap(attribute.price), attribute.currency_type,
|
||||
wrap(attribute.payment_method),
|
||||
// Here we handle GTFS specification corner case: "The fact that this field can be left
|
||||
// empty is an exception to the requirement that a Required field must not be empty.":
|
||||
attribute.transfers == FareTransfers::Unlimited ? "" : wrap(attribute.transfers),
|
||||
wrap(attribute.agency_id), wrap(attribute.transfer_duration)};
|
||||
write_joined(out, std::move(fields));
|
||||
}
|
||||
}
|
||||
|
|
20
signatures/version1/cla.json
Normal file
20
signatures/version1/cla.json
Normal file
|
@ -0,0 +1,20 @@
|
|||
{
|
||||
"signedContributors": [
|
||||
{
|
||||
"name": "nilsnolde",
|
||||
"id": 25637358,
|
||||
"comment_id": 1441674396,
|
||||
"created_at": "2023-02-23T12:23:33Z",
|
||||
"repoId": 250751634,
|
||||
"pullRequestNo": 18
|
||||
},
|
||||
{
|
||||
"name": "Osyotr",
|
||||
"id": 8740768,
|
||||
"comment_id": 2089322842,
|
||||
"created_at": "2024-05-02T00:13:58Z",
|
||||
"repoId": 250751634,
|
||||
"pullRequestNo": 21
|
||||
}
|
||||
]
|
||||
}
|
|
@ -1,3 +1,4 @@
|
|||
fare_id,price,currency_type,payment_method,transfers,transfer_duration
|
||||
p,1.25,USD,0,0,
|
||||
a,5.25,USD,0,0,
|
||||
a,5.25,USD,1,1,
|
||||
x,20,USD,0,,60
|
|
@ -23,6 +23,14 @@ TEST_CASE("Time in HH:MM:SS format")
|
|||
CHECK_EQ(stop_time.get_total_seconds(), 39 * 60 * 60 + 45 * 60 + 30);
|
||||
}
|
||||
|
||||
TEST_CASE("Time in HHH:MM:SS format")
|
||||
{
|
||||
Time stop_time("103:05:21");
|
||||
CHECK_EQ(stop_time.get_hh_mm_ss(), std::make_tuple(103, 5, 21));
|
||||
CHECK_EQ(stop_time.get_raw_time(), "103:05:21");
|
||||
CHECK_EQ(stop_time.get_total_seconds(), 103 * 60 * 60 + 5 * 60 + 21);
|
||||
}
|
||||
|
||||
TEST_CASE("Time from integers 1")
|
||||
{
|
||||
Time stop_time(14, 30, 0);
|
||||
|
@ -44,6 +52,7 @@ TEST_CASE("Invalid time format")
|
|||
CHECK_THROWS_AS(Time("12/10/00"), const InvalidFieldFormat &);
|
||||
CHECK_THROWS_AS(Time("12:100:00"), const InvalidFieldFormat &);
|
||||
CHECK_THROWS_AS(Time("12:10:100"), const InvalidFieldFormat &);
|
||||
CHECK_THROWS_AS(Time("12:10/10"), const InvalidFieldFormat &);
|
||||
}
|
||||
|
||||
TEST_CASE("Time not provided")
|
||||
|
@ -301,7 +310,7 @@ TEST_CASE("Read GTFS feed")
|
|||
CHECK_EQ(feed.get_attributions().size(), 1);
|
||||
CHECK_EQ(feed.get_calendar().size(), 2);
|
||||
CHECK_EQ(feed.get_calendar_dates().size(), 1);
|
||||
CHECK_EQ(feed.get_fare_attributes().size(), 2);
|
||||
CHECK_EQ(feed.get_fare_attributes().size(), 3);
|
||||
CHECK_EQ(feed.get_fare_rules().size(), 4);
|
||||
CHECK(!feed.get_feed_info().feed_publisher_name.empty());
|
||||
CHECK_EQ(feed.get_levels().size(), 3);
|
||||
|
@ -386,7 +395,7 @@ TEST_CASE("Stops")
|
|||
CHECK_EQ(stops[0].stop_id, "FUR_CREEK_RES");
|
||||
CHECK(stops[0].stop_desc.empty());
|
||||
CHECK_EQ(stops[0].stop_name, "Furnace Creek Resort (Demo)");
|
||||
CHECK_EQ(stops[0].location_type, StopLocationType::GenericNode);
|
||||
CHECK_EQ(stops[0].location_type, StopLocationType::StopOrPlatform);
|
||||
CHECK(stops[0].zone_id.empty());
|
||||
|
||||
auto const & stop = feed.get_stop("FUR_CREEK_RES");
|
||||
|
@ -485,7 +494,7 @@ TEST_CASE("Fare attributes")
|
|||
REQUIRE_EQ(feed.read_fare_attributes(), ResultCode::OK);
|
||||
|
||||
const auto & attributes = feed.get_fare_attributes();
|
||||
REQUIRE_EQ(attributes.size(), 2);
|
||||
REQUIRE_EQ(attributes.size(), 3);
|
||||
CHECK_EQ(attributes[0].fare_id, "p");
|
||||
CHECK_EQ(attributes[0].price, 1.25);
|
||||
CHECK_EQ(attributes[0].currency_type, "USD");
|
||||
|
@ -493,9 +502,28 @@ TEST_CASE("Fare attributes")
|
|||
CHECK_EQ(attributes[0].transfers, FareTransfers::No);
|
||||
CHECK_EQ(attributes[0].transfer_duration, 0);
|
||||
|
||||
CHECK_EQ(attributes[1].fare_id, "a");
|
||||
CHECK_EQ(attributes[1].price, 5.25);
|
||||
CHECK_EQ(attributes[1].currency_type, "USD");
|
||||
CHECK_EQ(attributes[1].payment_method, FarePayment::BeforeBoarding);
|
||||
CHECK_EQ(attributes[1].transfers, FareTransfers::Once);
|
||||
CHECK_EQ(attributes[1].transfer_duration, 0);
|
||||
|
||||
CHECK_EQ(attributes[2].fare_id, "x");
|
||||
CHECK_EQ(attributes[2].price, 20);
|
||||
CHECK_EQ(attributes[2].currency_type, "USD");
|
||||
CHECK_EQ(attributes[2].payment_method, FarePayment::OnBoard);
|
||||
CHECK_EQ(attributes[2].transfers, FareTransfers::Unlimited);
|
||||
CHECK_EQ(attributes[2].transfer_duration, 60);
|
||||
|
||||
const auto & attributes_for_id = feed.get_fare_attributes("a");
|
||||
REQUIRE_EQ(attributes_for_id.size(), 1);
|
||||
CHECK_EQ(attributes_for_id[0].price, 5.25);
|
||||
|
||||
REQUIRE_EQ(feed.write_fare_attributes("data/output_feed"), ResultCode::OK);
|
||||
Feed feed_copy("data/output_feed");
|
||||
REQUIRE_EQ(feed_copy.read_fare_attributes(), ResultCode::OK);
|
||||
CHECK_EQ(attributes, feed_copy.get_fare_attributes());
|
||||
}
|
||||
|
||||
TEST_CASE("Fare rules")
|
||||
|
|
Loading…
Add table
Reference in a new issue