This package contains necessary types and useful functions for creating data sources for use with the OpenParking project, a service focussed on collecting data on available public parking spots and providing them and derived information as open-data wherever possible.
Check out the documentation for API specifics.
Data sources are Swift packages, one or more per package. Check out swift.org for instructions on how to get Swift running on your system if it's not already set up.
Let's create a new data source for the fictional city of Bielefeld.
$ mkdir Bielefeld & cd Bielefeld
$ swift package init
Now open your newly created package manifest (Package.swift
) and add OpenParking to your list of dependencies, replacing latest
with the latest release version listed above.
// ...
dependencies: [
.package(url: "https://github.com/OpenParkingApp/OpenParking.git", from: "latest"),
]
// ...
Swift also requires that you add the relevant products to your targets as dependencies. OpenParking contains two targets, OpenParking and OpenParkingTestSupport, the latter containing some helpers to validate your data in your test target. The targets section in our manifest now looks like the following. The dependency of OpenParkingTestSupport on BielefeldTests unfortunately requires you to be a bit more verbose, since it differs from the package basename.
// ...
targets: [
.target(
name: "Bielefeld",
dependencies: ["OpenParking"]),
.testTarget(
name: "BielefeldTests",
dependencies: [
.product(name: "OpenParkingTestSupport", package: "OpenParking"),
"Bielefeld"
]),
]
// ...
Now let's head over into the main implementation in Sources/Bielefeld/Bielefeld.swift. This is where we'll define the actual data source and how it fetches its data.
To start out we're going to import OpenParking, create a new type and conform it to the Datasource
protocol. Don't forget to mark the type as public so it's accessible from outside of this module.
import Foundation
import OpenParking
public class Bielefeld: Datasource {
}
At this point your editor will likely be complaining that Bielefeld
isn't conforming to Datasource
since it requires a few attributes. These are:
- name
- slug
- infoURL
- contributor (optional)
- attributionURL (optional)
- license (optional)
Additionally the protocol requires that we implement the function func data() throws -> DataPoint
, which is the entry point for users of this data source to fetch current data.
Implementing the required attributes and a placeholder for data()
leaves us with the following. Don't forget to also add a public initializer to your type so it can be created elsewhere.
public class Bielefeld: Datasource {
public let name = "Bielefeld"
public let slug = "bielefeld"
public let infoURL = URL(string: "https://bielefeld.de")!
public init() {}
public func data() throws -> DataPoint {
fatalError("not yet implemented")
}
}
Before continuing with the actual implementation of data()
, let's quickly just add the few lines of code necessary for testing our implementation later. This is done in Tests/BielefeldTests/BielefeldTests.swift. Modify that file to look like this:
import XCTest
import OpenParkingTestSupport
import Bielefeld
final class BielefeldTests: XCTestCase {
func testDatasource() throws {
validate(datasource: Bielefeld())
}
}
That's it! validate(datasource:)
is a function from OpenParkingTestSupport that fetches live data and runs some validations to check if it makes sense.
You can run the tests via $ swift test
, you'll see any warnings and errors it encounters.
This is the interesting part and unfortunately also the one that differs for pretty much every data source. There are however some helpful utilities in the OpenParking package to make this a little easier and maintainable.
For ease of use there's two synchronous networking functions, get(url:headers:)
and post(url:headers:)
both of which return a tuple of type (Data, HTTPURLResponse)
.
Additionally there's a few helpful extensions on String
for parsing dates and unescaping HTML entities.
The main idea is to fetch current data and process that to return a DataPoint
containing information on Lot
s. For details it probably makes sense to have a look at existing data sources, e.g. Basel or Herrenberg.
One more thing to note here are the three available options of indicating issues with the data.
- If there's an inconsistency with the data that should be investigated, but is not a problem, call the
warn(_:lotName:)
function. These warnings will be aggregated and (hopefully) looked at later by whoever's running the data import. - A
DataPoint
is not a direct collection ofLot
s, but contains values of typeResult<Lot, LotError>
. Use theLotError
to highlight if data for a specific lot is broken and can not be aggregated, possibly due to missing static metadata or something else. - If all else fails and data collection has failed entirely, make use of the throwing nature of
data()
and throw a fitting error.
All of this and more should be listed in the project's documentation as well.
You'll likely want to bundle static data with your data source, including things like lot coordinates, addresses, types and anything else not being fetched together with the live data. If not present in the live data it also makes sense to include a count of total available spaces here. This kind of data is typically bundled as a GeoJSON file.
It will look something like this:
{
"type": "FeatureCollection",
"features": [{
"type": "Feature",
"geometry": {
"type": "Point",
"coordinates": [8.533417, 52.025246]
},
"properties": {
"name": "Parkplatz Alte Post",
"address": "Friedrich-Ebert-Straße",
"type": "lot",
"total": 80
}
},
// ...
]
}
Bundling this with your package is as easy as dropping the file (typically named geojson.json
) in our target's source directory, e.g.Sources/Bielefeld
for this example. To actually bundle our static data with the package we'll have to add the resource to our target in the package manifest.
.target(
name: "Bielefeld",
dependencies: ["OpenParking"],
resources: [
.process("geojson.json"),
]),
We can then have the framework automatically decode it and access its data by doing the following inside our data source.
let geodata = try self.geodata(from: .module)
This returns a GeoJson
object which we can use to retrieve our stored metadata through functions like lot(withName:)
. See the project's documentation for more.