Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/drallgood/jpasskit
jPasskit is an Java™ implementation of the Apple™ PassKit Web Service.
https://github.com/drallgood/jpasskit
apple boardingpass generator java passkit reward server wallet webservice
Last synced: 4 days ago
JSON representation
jPasskit is an Java™ implementation of the Apple™ PassKit Web Service.
- Host: GitHub
- URL: https://github.com/drallgood/jpasskit
- Owner: drallgood
- License: apache-2.0
- Created: 2012-09-17T09:56:45.000Z (over 12 years ago)
- Default Branch: master
- Last Pushed: 2024-07-18T13:07:23.000Z (6 months ago)
- Last Synced: 2025-01-10T17:02:03.191Z (11 days ago)
- Topics: apple, boardingpass, generator, java, passkit, reward, server, wallet, webservice
- Language: Java
- Homepage:
- Size: 2.3 MB
- Stars: 280
- Watchers: 29
- Forks: 110
- Open Issues: 9
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
- Codeowners: CODEOWNERS
Awesome Lists containing this project
README
# jpasskit
jPasskit is an Java™ implementation of the Apple™ PassKit Web Service.
There are two separate projects:
1. jPasskit - Which contains the Pass objects and useful utilities. It is designed to be included in existing Webservices, thus not including any request handling.
2. jPasskit Server - Which contains an additional REST Webservice, that can be included in existing Applications that do not use their own Webservice already. Note: This is still no standalone implementation. Things like storing passes and handling device (un)registrations are left open for the Application to implement.**Current stable release:** 0.4.2
**Development Version:** 0.4.3-SNAPSHOT [![Build Status][image-1]][1]
[![Codacy Badge][image-2]][2]
[![DepShield Badge][image-3]][3]## Installation
### Using Maven
de.brendamour
jpasskit
or:
de.brendamour
jpasskit.server
**The released artifacts are now available at Maven Central**
Snapshot versions can be found here: https://oss.sonatype.org/content/repositories/snapshots/
### Using Gradle
dependencies {
api("de.brendamour:jpasskit:0.4.0")
}or:
dependencies {
api("de.brendamour:jpasskit.server:0.4.0")
}**The released artifacts are now available at Maven Central**
Snapshot versions can be found here: https://oss.sonatype.org/content/repositories/snapshots/
## Using jPasskit
Using jPasskit is pretty straight forward:
### Creating a Pass
The class PKPass is the toplevel class. It represents the pass.json file. Everything else can just be added like one would add it on the JSON side.
Example:
PKPass pass = PKPass.builder()
.pass(
PKGenericPass.builder()
.passType(PKPassType.PKStoreCard)
.primaryFieldBuilder(
PKField.builder()
.key("balance")
.label("balance")
.value(20.0)
.currencyCode("EUR")
)
)
.barcodeBuilder(
PKBarcode.builder()
.format(PKBarcodeFormat.PKBarcodeFormatQR)
.message("ABCDEFG")
.messageEncoding(Charset.forName("utf-8"))
)
.formatVersion(1)
.passTypeIdentifier("pass.some.passTypeIdentifier")
.serialNumber("000000001")
.teamIdentifier("myTeamId")
.organizationName("OrgName")
.logoText("MyPass")
.description("My PassBook")
.backgroundColor(Color.BLACK)
.foregroundColor("rgb(255,255,255 )")
// ... and more initializations ...
.build();### Providing pass templates
Usually, passes contain additional information that need to be included in the final, signed pass, e.g.
- Images (icons, logos, background images)
- TranslationsjPasskit provides a flexible mechanism to provide such templates:
- as a folder (using `PKPassTemplateFolder`)
- as a set of streams (using `PKPassTemplateInMemory`)
- using your own implementation (implementing `IPKPassTemplate`)#### Folder-based templates (standard approach)
In order to use an existing folder on the file system as you pass's template, you create an instance of `PKPassTemplateFolder` using the path to your folder as argument:
IPKPassTemplate pkPassTemplateFolder = new PKPassTemplateFolder(PASS_TEMPLATE_FOLDER);
The content of this directory is defined in the PassKit Developer Documentation.
That's it. When signing the pass, the contents of this folder will be copied into the pass.
#### Dynamic templates (in memory)
This approach requires more code, but is also more flexible. The template is stored as a list of input streams, whose content gets copied into the pass when it is signed and packaged.
It does not matter, where the stream comes from, but the data needs to be available when the template is used.
For convenience, we provide methods to add several additional data types to the template:pkPassTemplateInMemory.addFile(PKPassTemplateInMemory.PK_BACKGROUND, stream);
pkPassTemplateInMemory.addFile(PKPassTemplateInMemory.PK_BACKGROUND_RETINA, stringBuffer);
pkPassTemplateInMemory.addFile(PKPassTemplateInMemory.PK_ICON, file);
pkPassTemplateInMemory.addFile(PKPassTemplateInMemory.PK_ICON_RETINA, url);As you can see, we're also providing static variables for the most common file names.
You can also add an optional locale parameter to the call, in which case the file will automatically be added only for the given language:
pkPassTemplateInMemory.addFile(PKPassTemplateInMemory.PK_ICON_RETINA, Locale.ENGLISH, url);
//content from URL will be placed in "en.lproj/[email protected]"**Note:** There are no checks, that the content of a provided file is valid. So if you'd provide a PDF file but store it as icon.png, it will not work.
### Signing and Zipping a Pass
The PKSigningUtil contains all necessary methods to:
1. Load the Pass Certificate
2. Load the Apple Worldwide Developer Relations CA (AppleWWDRCA)
3. Create the pass.json file
4. Hash all files in the Pass directory and create the Manifest file (manifest.json)
5. Sign the Manifest file
6. ZIP the finished Pass directoryExample to do it all in one step:
PKSigningInformation pkSigningInformation = new PKSigningInformationUtil().loadSigningInformationFromPKCS12AndIntermediateCertificate(keyStorePath, keyStorePassword, appleWWDRCA);
PKPassTemplateFolder passTemplate = new PKPassTemplateFolder(template_path);
PKFileBasedSigningUtil pkSigningUtil = new PKFileBasedSigningUtil();
byte[] signedAndZippedPkPassArchive = pkSigningUtil.createSignedAndZippedPkPassArchive(pass, passTemplate, pkSigningInformation);
## Using the jPasskit ServerThe jPasskit Server doesn't provide a full fledged PassKit Web Service but merely the basics you need implement your own standalone server. Things like storing passes and registrations still need to be implemented according to your own needs (or added to an existing Application).
### Setup
The set up and start the Server you need two things:
1. Create a Java Property object containing at least the three keys 'rest.bindIP', 'rest.bindPort' and 'rest.ssl.enabled'.
**Note:**
For the Production mode, you'll have to enable SSL and provide the following 4 keys:
- rest.ssl.keystore.path : The path to the keystore where the SSL certificate for this server is stored
- rest.ssl.keystore.type : The type of this keystore (e.g. PKCS12 or JKS)
- rest.ssl.keystore.password : The password to access the keystore
- rest.ssl.key.password : The password to access the private key
Apple requires all production passes to use SSL.
2. An implementation of IPKRestletServerResourceFactory.The IPKRestletServerResourceFactory is used to create instances of three classes: PKDeviceResource. PKPassResource, PKLogResource. You'll need to subclass each of these and provide your own implementations.
*PKDeviceResource* is used to register/unregister devices and to get the serialNumbers of changed passes.
*PKPassResource* is used to fetch the latest version of a pass.
_PKPersonalizePassResource_ is used to store the signup information for a rewards program (see https://developer.apple.com/library/prerelease/content/documentation/UserExperience/Conceptual/PassKit_PG/PassPersonalization.html)
*PKLogResource* is used for the log messages, that the devices send in case of an error.
Then you create the server instance:
Properties serverConfigurationProperties = new Properties();
serverConfigurationProperties.put("rest.bindIP", "::");
serverConfigurationProperties.put("rest.bindPort", "8082");
IPKRestletServerResourceFactory pkRestletServerResourceFactory = new MyOwnPKRestletServerResourceFactory();
PKRestServer pkRestServer = new PKRestServer(serverConfigurationProperties, pkRestletServerResourceFactory);
try {
pkRestServer.start();
} catch (Exception e) {
e.printStackTrace();
}
That's it. Your web service is running. Just point your passes to the URL where the server is running.### About Personalized Passes and Rewards Programs
Apple provides a handy, albeit short, guide about how this works: https://developer.apple.com/library/prerelease/content/documentation/UserExperience/Conceptual/PassKit\_PG/PassPersonalization.html
The process in broad strokes works as follows:
1. You create a personalizable pass (e.g. using jPasskit) for a user using a unique `serialNumber`
2. The user adds the pass, and completes the signup form
3. Your server gets a request to the personalization endpoint at `webServiceURL/version/passes/passTypeIdentifier/serialNumber/personalize` which ends up in the `PKPersonalizePassResource`.
4. You store the provided information and link it to the `serialNumber`
5. Next time the user's device downloads a new version of the pass, you provide a custom pass with his information (Make sure you DON'T provide a personalizable pass this time!)[1]: https://github.com/drallgood/jpasskit/actions/workflows/maven.yml
[2]: https://app.codacy.com/app/drallgood/jpasskit?utm_source=github.com&utm_medium=referral&utm_content=drallgood/jpasskit&utm_campaign=Badge_Grade_Settings
[3]: https://depshield.github.io[image-1]: https://github.com/drallgood/jpasskit/actions/workflows/gradle.yml/badge.svg
[image-2]: https://app.codacy.com/project/badge/Grade/e8d0e390a9c74e1babcac19b45ba89c5
[image-3]: https://depshield.sonatype.org/badges/drallgood/jpasskit/depshield.svg