Skip to content

maptic/osurs

Repository files navigation

osurs

CMake Doxygen

Optimization of space utilization in reservation systems. An algorithm for optimal distribution of reservations along a trip among available seats.

The osurs library contains the following modules with corresponding headers:

  • ds: Data structures for efficient data handling.
  • io: Input and output of networks and its structures.
  • network: Network for reservation optimization.
  • reserve: Connection routing, checking seat availability and reservation.
  • optimize: Optimizing space utilization in reservation systems.
  • olal: Optimization logic abstraction layer.
  • types: Data types of osurs.

Interdependencies:

graph RL;
  subgraph libosurs;
    osurs/types.h-->|#include|osurs/ds.h;
    osurs/network.h-->|#include|osurs/types.h;
    osurs/reserve.h-->|#include|osurs/network.h;
    osurs/io.h-->|#include|osurs/reserve.h
    osurs/optimize.h-->|#include|osurs/types.h;
    osurs/olal.h-->|#include|osurs/reserve.h;
    osurs/olal.h-->|#include|osurs/optimize.h;
  end;

Note: The core functionality of osurs is optimizing reservations. There are already powerful routing algorithms for public transport, so the algorithm included here is minimal and only serves to book reservations on already known/found connections to the right segments of the trips (without transfers).

Components and functionalities

Transit network

The network consists of nodes where vehicles stop and passengers can get on and off. A route stores the order in which the nodes are approached by a vehicle in a chain of stops. Each stop contains information about which stop is next and how long it takes to reach it. On routes, trips indicate the departure times at which a vehicle leaves from the route's root stop. Vehicle information such as capacity and reservations are stored at the trip level.

All objects of the network are located on the heap and are directly or indirectly linked to the network structure. If the memory of the network is released (delete_network()) all associated structures of the network are also cleared.

graph LR;
  subgraph network.h;
    Network-->|"nodes[]"|Node;
    Network-->|"routes[]"|Route;
    Network-->|"compositions[]"|Composition;
    Network-->|"vehicles[]"|Vehicle;
    Route-->|*root_stop|Stop;
    Stop-->|*prev|Stop;
    Stop-->|*next|Stop;
    Route-->|*root_trip|Trip;
    Trip-->|*next|Trip;
    Node-->|"routes[]"|Route;
    Stop-->|*node|Node;
    Trip-->|*vehicle|Vehicle;
    Vehicle-->|*composition|Composition;
  end;

Connections and reservations

Queried connections are not stored on the network and must be released individually to prevent a memory leak (delete_connection()). If more than one connection is possible between to nodes on the network, a connection chain is created. In a connection chain, the .prev property of the connection structure points to the previous connection or NULL if it is the root of the chain. Identically, the .next property points to the next connection or to NULL if it is the end of the chain.

graph LR;
  connection_1-->|*next|connection_2;
  connection_2-->|*next|connection_3;
  connection_3-->|*next|NULL;
  connection_3-->|*prev|connection_2;
  connection_2-->|*prev|connection_1;
  connection_1-->|*prev|NULL;

When a reservation is made, it is stored as a reservation struct on the network with a relation to the corresponding trip. The reservation exists on the heap until the entire network is released.

graph LR;
  subgraph network.h;
    Trip-->|"reservations[]"|Reservation;
    Reservation-->|*trip|Trip;
    Reservation-->|*orig|Stop;
    Reservation-->|*dest|Stop;
    Trip-->|*root_stop|Stop;
  end;
  subgraph reserve.h;
    direction LR;
    Connection-->|*orig|Stop;
    Connection-->|*dest|Stop;
    Connection-->|*trip|Trip
  end

Input and output

Networks and reservations can be persisted as separate XML files. This separation brings the advantage that the reservations (e.g. of a certain day) can be added to an already loaded network. However, it is important to delete already existing reservations from the network beforehand.

#include <stdio.h>
#include <osurs/io.h>

int main(int argc, char* argv[]) {
    Network* network = new_network();
    if (!import_network(network, "network.xml")) {
        perror("Could not load network.");
        return 1;
    }
    if (!import_reservations(network, "reservations.xml")) {
        perror("Could not load reservations.");
        return 1;
    }
    delete_network(network);
    return 0;
}

In addition, transit schedules and vehicle definition files from MATSim can be imported using import_matsim().

Optimization logic abstraction layer

The optimization logic abstraction layer provides the interface between the seat optimization algorithm and the transit schedule. The seat optimization algorithm could also be used independently from public transport for the optimization of bookings.

graph LR;
  subgraph network.h;
    Reservation;
  end;
  subgraph optimize.h;
    direction LR;
    SeatCollection-->|"seats[]"|Seat;
    Seat-->|"reservations[]"|Reservation;
  end

The optimized seat collection is calculated at the time when the seat allocation is needed. Usually a few minutes before the departure of a trip. Therefore, this information is kept outside the network and the memory of the seat collection must be released separately.

#include <osurs/io.h>
#include <osurs/olal.h>
#include <stdio.h>

int main(int argc, char *argv[]) {
    // read network
    Network *network = new_network();
    if (!import_network(network, "network.xml")) {
        perror("Could not load network.");
        return 1;
    }

    // create random reservations
    for (int i = 0; i < 1000000; ++i) {
        // search direct connection between random nodes
        Node *orig = hash_map_get_random(network->nodes);
        Node *dest = hash_map_get_random(network->nodes);
        int dep = rand() % (24 * 60 * 60) + 1;
        Connection *conn = new_connection(orig, dest, dep);
        // book reservation if enough seats are available for connection
        int seats = rand() % 3 + 1;
        Reservation *res = new_reservation(conn, seats);
        delete_connection(conn);
    }

    // optimize a random trip
    Route *route = hash_map_get_random(network->routes);
    SeatCollection *collection = optimize_trip(route->root_trip);
    print_seat_collection(collection, 0);

    // cleanup
    delete_seat_collection(collection);
    delete_network(network);

    return 0;
}

Note: Currently, only the strategy of sparsest distribution of reservations among seats is implemented. In a public transportation system, the sparsest distribution considering group reservations would be more likely.

Install the library

Build and install the library and the command line interface.

cd osurs
mkdir build && cd build
cmake ..
make
make install

To use the osurs library in a project, link with the needed sublibraries when building. For example:

gcc main.c -o main -losurs-ds -losurs-network ...

Use the osurscli to print a network and reservations or to convert a matsim transit schedule.

osurscli print tests/input/intercity_network.xml tests/input/intercity_reservations.xml
osurscli convert tests/input/matsim/transitSchedule.xml tests/input/matsim/transitVehicles.xml tmp_network.xml

Development

Trunk-based development workflow

The trunk-based development workflow uses one main branch to record the history of the project. In addition to the mainline, short-lived feature or bugfix branches are used to develop new features or fix bugs.

gitGraph
  commit
  commit tag: "v0.1.0"
  branch feature/name
  commit
  commit
  checkout main
  branch bugfix/name
  commit
  checkout main
  merge bugfix/name
  commit tag: "v0.1.1"
  checkout feature/name
  commit
  checkout main
  merge feature/name
  commit tag: "v0.2.0"

This library uses semantic versioning. Once main has acquired enough features for a release, set the new version number in the CMakeLists.txt and CHANGELOG.md. Commit and push to main and publish a release on GitHub with the version number as tag.

Style guide

Use the following naming conventions:

  • snake_case for variables and functions.
  • UpperCamelCase for type definitions.
  • UPPER_SNAKE_CASE for constants.

Place the opening brace on the same line in function definitions and flow control statements:

int function(int x) {
    if (x > 0) {
        // ...
    }
    return x;
}

Testing

Always write a unit test for new features using googletest and check for memory leaks before merging a feature into main:

valgrind -s --leak-check=full ./main

Setup dependencies

The following dependencies must be installed on the system to build and test the project:

cmake       # Build, test and package
doxygen     # Render docs
graphviz    # Graphs in docs
googletest  # Unit tests, installed via cmake
valgrind    # Detect memory leaks
libxml2     # Read and write networks
  • Ubuntu
apt install cmake doxygen graphviz valgrind libxml2-dev
  • macOS
brew install cmake doxygen graphviz valgrind libxml2

References

Project setup

Testing import and export module

Routing

IO

Python bindings

Strings

About

Optimization of space utilization in reservation systems. An algorithm for optimal distribution of reservations along a trip among available seats.

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published