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

Quickstart Updates #5

Open
wants to merge 9 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
8 changes: 8 additions & 0 deletions .idea/.gitignore

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

8 changes: 8 additions & 0 deletions .idea/modules.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

9 changes: 9 additions & 0 deletions .idea/passkit-golang-grpc-quickstart.iml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 6 additions & 0 deletions .idea/vcs.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

166 changes: 111 additions & 55 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,79 +1,135 @@
# passkit-golang-members-grpc-quickstart

The PassKit Golang SDK makes it quick and easy to create and install your branded membership passes for Apple Wallet and Google Pay.

This repository has following structure with each purpose.
- `certs` folder is a place to store your credential files.
- `examples` folder contains SDK methods you can use to create membership cards and engage with members.

## Table of Content
* [Installation](#installation)
* [Prerequisites](#prerequisites)
* [Quickstart](#quickstart)
* [Examples](#examples)
* [GUI Tool](#gui-tool)
* [Documentation](#documentation)
* [Check Other Passes](#check-other-passes)
* [Getting Help](#getting-help)
* [License](#license)

## Installation
Install passkit-io-go with:
```go
go get -u github.com/PassKit/passkit-golang-sdk
```
Then, import SDK with:
```go
import(
"github.com/PassKit/passkit-golang-grpc-sdk/io/members"
"github.com/PassKit/passkit-golang-grpc-sdk/io"
)
```
## Prerequisites
1. Create a PassKit account. Sign up for free [HERE](https://app.passkit.com/).

2. Generate & Download your SDK credentials by clicking the 'GENERATE NEW SDK CREDENTIALS' button from the Developer Tools page in the [portal](https://app.passkit.com/app/account/developer-tools).

## Quickstart
By completing this Quickstart, you will be able to issue a membership card for a new member.
PassKit Golang Quickstart
=======================
[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
[![Go Reference](https://pkg.go.dev/badge/github.com/PassKit/passkit-golang-grpc-sdk.svg)](https://pkg.go.dev/github.com/PassKit/passkit-golang-grpc-sdk)

### Overview

This quickstart aims to help get Golang developers up and running with the PassKit SDK as quickly as possible.

### Prerequisites

1. Ensure your followed the steps in [prerequisites](#prerequisites).
You will need the following:
- A PassKit account (signup for free at [PassKit](https://app.passkit.com))
- Your PassKit SDK Credentials (available from the [Developer Tools Page](https://app.passkit.com/app/account/developer-tools))
- Apple wallet certificate id (for flights only, available from the [certificate page](https://app.passkit.com/app/account/certificates))
![ScreenShot](images/certificate.png)
- Recommended code editor [Golang](https://go.dev/dl/) (Guide to [installation](https://go.dev/doc/install))

2. Install PassKit Golang SDK with:
```go
go get -u github.com/PassKit/passkit-golang-sdk
```
### Configuration

3. Place your SDK credential files (`certificate.pem`, `key.pem` and `ca-chain.pem`) in the certs folder in this repoo. The SDK uses these .pem files to authenticate against the PassKit server.
1. Install PassKit Golang SDK in the terminal with `go install github.com/PassKit/passkit-golang-grpc-sdk`
It should output something similar to below:
![ScreenShot](images/go-get-commands.png)

4. Now we need to decrypt your `key.pem`. At your project root directory, run `cd ./certs openssl ec -in key.pem -out key.pem`. Your `key.pem` file should look like below.
2. In the certs folder of the repository add the following three PassKit credential files:
- certificate.pem
- ca-chain.pem
- key.pem

You can disregard the key-java.pem credentials file as it is not compatible with Golang.

3. Now we need to decrypt your `key.pem`. At your project root directory in the terminal, run `cd ./certs` `openssl ec -in key.pem -out key.pem`. If you are in your root directory the terminal should show `PASSKIT-GOLANG-GRPC-QUICKSTART` to the left of the cursor and then after running the command `certs`, as shown below.
![ScreenShot](images/decrypt-key.png)
For the password use the one-time password that you used for generating the SDK credentials.

Your `key.pem` file should look like below.
![ScreenShot](https://raw.githubusercontent.com/PassKit/passkit-golang-members-quickstart/master/images/decrypted_key_pem.png)
If you do not see `Proc-Type: 4,ENCEYPTED` on line 2, you have successfully decrypted `key.pem`.
If you do not see `Proc-Type: 4,ENCEYPTED` on line 2, you have successfully decrypted `key.pem`.

4. Replace `[email protected]` in `main.go` on line 29 with your email address in order to receive the welcome email with card url which your member will also receive.
![ScreenShot](images/email.png)

5. Replace `[email protected]` in `main.go` with your email address in order to receive the welcome email with card url which your member will also receive.
5. Go back to root directory with cd ../ Then run go mod tidy , then go run main.go to create a sample membership card, coupon card and boarding pass (with default templates & tiers/offers) and issue them.

6. Go back to root directory with `cd ../..`. Then run `go run main.go` to create a sample program (with default template & tier) and issue a membership card against that.
6. If you are issuing boarding passes set PassTypeIdentifier on line 30 in create_carrier.go in the flights folder to your Apple Certificate Pass Type Id as shown in the prerequisites.
![ScreenShot](images/apple-certificate.png)

## Examples
### Membership Cards
#### Issue A Membership Card.
Follow the steps of the [Quickstart](#quickstart) to create a sample membership card and experience it in your Mobile Wallet.
Follow the steps of the configuration to get the quickstart up and running.
In `IssueMembershipCard()` the methods there are:
- CreateProgram() - takes a new program name and creates a new program
- CreateTier() - takes the programId of the program just created in the above program, creates a new template (based of default template), creates a tier, and links this tier to the program
- EnrolMember() - takes programId and tierId created by the above methods, and memberDetails, creates a new member record, and sends a welcome email to deliver membership card url

After running `go run main.go` the terminal should show:
![ScreenShot](images/issue-membership-card.png)

#### Engage With Your Members
`EngageWithMembers()` contains multiple methods you can use to engage with your members.
For example, you can update contents of digital membership card or send a push notification.
`EngageWithMembers()` contains multiple methods you can use to engage with your members. They use the information from `IssueMembershipCard()` to run the methods unless information is manually entered.
- GetSingleMember() - takes memberId and returns the record of that member
- ListMembers() - takes search conditions as pagination object and returns list of member records which match with the conditions
- CountMembers() - takes search conditions as pagination object and returns the number of members who match with the condition
- SendWelcomeEmail() - takes memberId and sends a welcome email (contains membership card url) to the member
- UpdateMember_EmailAddress() - takes memberId and memberDetails, and updates existing member record
- AddPoints() - takes a programId of an existing program and memberId of existing member to add points to chosen member
- UsePoints() - takes a programId of an existing program and memberId of existing member to use points from a chosen member
- DeleteMembers() - takes programId, tierId, memberId and memberDetails, deletes an existing member record

After running `go run main.go` the terminal should show:
![ScreenShot](images/engage-with-members.png)

### Coupons
#### Issue A Coupon.
Follow the steps of the configuration to get the quickstart up and running.
In `IssueCoupon()` the methods are:
- CreateCampaign() - takes a new campaign name and creates a new campaign
- CreateOffer() - takes a campaignId of the campaign you just created and creates a new template (based of default template), creates an offer, and links this offer to the campaign
- CreateCoupon() - takes campaignId and offerId created by the above methods, and couponDetails, creates a new coupon record, and sends a welcome email to deliver coupon card url

After running `go run main.go` the terminal should show:
![ScreenShot](images/issue-coupon.png)

#### Engage With Coupons.
`EngageWithCoupons()` contains multiple methods you can use to engage with coupons. They use the information from `IssueCoupon()` to run the methods unless informatin is manually entered.
- GetSingleCoupon() - takes couponId and returns the record of that coupon
- ListCoupons() - takes search conditions as pagination object and returns list of coupon records which match with the conditions
- CountCoupons() - takes search conditions as pagination object and returns the number of coupons who match with the condition
- UpdateCoupon() - takes a campaignId of an existing campaign and couponId of existing coupon to update that coupon
- RedeemCoupon() - takes a campaignId of an existing campaign and couponId of existing coupon to redeem that coupon
- VoidCoupon() - takes the couponId, offerId and campaignId to void an existing coupon
- DeleteCouponOffer() - takes the offerId to delete an existing offer

After running `go run main.go` the terminal should show:
![ScreenShot](images/engage-with-coupons.png)

### Boarding Passes
#### Issue A Boarding Pass.
Follow the steps of the configuration to get the quickstart up and running.
In `IssueBoardingPass()` the methods are:
- CreateTemplate() - creates the pass template for flights and boarding passes
- CreateCarrier() - takes a new carrier code and creates a new carrier
- CreateAirport() - takes a new airport code and creates a new airport.
- CreateFlight() - takes templateId , from previous method, to use as base template and uses a carrier code, created from previous method, and creates a new flight
- CreateFlightDesignator() - creates flight designator using flight code
- CreateBoardingPass() - takes templateId, from previous method, and customer details creates a new boarding pass, and sends a welcome email to deliver boarding pass url

After running `go run main.go` the terminal should show:
![ScreenShot](images/issue-boarding-pass.png)


#### Engage With Boarding Passes.
`EngageWithBoardingPass()` contains multiple methods you can use to engage with boarding passes. They use the information from `IssueBoardingPass()` to run the methods unless informatin is manually entered.
- DeleteFlight() - takes an existing flight number as well as other details and deletes the flight associated with it
- DeleteFlightDesignator() - takes an existing flight designation and deletes the flight designator associated with it
- DeleteAirports() - takes an existing airport code and deletes the airport associated with it
- DeleteCarrier() - takes an existing carrier code and deletes the carrier associated with it

After running `go run main.go` the terminal should show:
![ScreenShot](images/engage-with-boarding-passes.png)

## GUI Tool
GUI tool can be accessed from [your PassKit account](https://app.passkit.com/login).

## Documentation
* [PassKit Membership Official Documentation](https://docs.passkit.io/protocols/member)
* [PassKit Coupons Official Documentation](https://docs.passkit.io/protocols/coupon)
* [PassKit Boarding Passes Official Documentation](https://docs.passkit.io/protocols/boarding)

## Check Other Passes
* Coupons (coming soon)
* Flight Ticket (coming soon)

## Getting Help
* Email [[email protected]](email:[email protected])
* [Online chat support](https://passkit.com/)

## License
Expand Down
50 changes: 50 additions & 0 deletions examples/coupons/count_coupons.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
package coupons

import (
"fmt"
"github.com/PassKit/passkit-golang-grpc-sdk/io/single_use_coupons"
"github.com/PassKit/passkit-golang-grpc-quickstart/examples/shared"
"log"

"github.com/PassKit/passkit-golang-grpc-sdk/io"
"golang.org/x/net/context"
"google.golang.org/grpc/metadata"
)

// CountCoupons takes search conditions as pagination object and returns the number of coupons who match with the condition.
func CountCoupons(campaignId, offerId string) {
fmt.Println("Counting coupon records that match with conditions...")

// Generate a coupons module client
pkCouponsClient := single_use_coupons.NewSingleUseCouponsClient(shared.Conn)

// Generate context object to connect to the server.
ctx := context.Background()
ctx = metadata.NewOutgoingContext(ctx, nil)

//Filters coupons based on chosen filter/s e.g. based on offers, more than one field can be chosen
listRequest := &single_use_coupons.ListRequest{
CouponCampaignId: campaignId,
Filters: &io.Filters{
FilterGroups: []*io.FilterGroup{
{
Condition: io.Operator_AND,
FieldFilters: []*io.FieldFilter{
{
FilterField: "offerId",
FilterValue: offerId,
FilterOperator: "eq",
},
},
},
},
},
}

countResponse, err := pkCouponsClient.CountCouponsByCouponCampaign(ctx, listRequest)
if err != nil {
log.Fatalf("Count coupons err: %v\n", err)
}

fmt.Printf("Count result was %v.\n", countResponse.Total)
}
45 changes: 45 additions & 0 deletions examples/coupons/create_campaign.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
package coupons
import (
"fmt"
"log"

"github.com/PassKit/passkit-golang-grpc-quickstart/examples/shared"
"github.com/PassKit/passkit-golang-grpc-sdk/io"
"github.com/PassKit/passkit-golang-grpc-sdk/io/single_use_coupons"

"golang.org/x/net/context"
"google.golang.org/grpc/metadata"
)

// CreateCampaign takes a new campaign name and creates a new campaign. The method returns the campaign id.
// A campaign needs to be created because campaign functions as a class object for offer and coupons.
func CreateCampaign() string {
fmt.Println("Start creating a coupon campaign...")

// Generate PassKit Client object for Coupon protocol.
pkCouponsClient := single_use_coupons.NewSingleUseCouponsClient(shared.Conn)

// Generate context object to connect to the server.
ctx := context.Background()
ctx = metadata.NewOutgoingContext(ctx, nil)

// Create your coupon campaign object.
campaign := &single_use_coupons.CouponCampaign{
Name: "Coupon Campaign",
Status: []io.ProjectStatus{
io.ProjectStatus_PROJECT_DRAFT,
io.ProjectStatus_PROJECT_ACTIVE_FOR_OBJECT_CREATION,
},
}

// Send gRPC request to create a coupon campaign record.
campaignId, err := pkCouponsClient.CreateCouponCampaign(ctx, campaign)
if err != nil || campaignId == nil {
log.Fatalf("Create campaign err: %v", err)
}

// You need this campaign id to create Offer and Coupon objects in order to issue coupons.
log.Printf("Create Campaign Success: You have successfully created your coupon campaign. Your campaign id is %s.\n", campaignId.Id)

return campaignId.Id
}
49 changes: 49 additions & 0 deletions examples/coupons/create_coupon.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
package coupons

import (
"fmt"
"log"

"github.com/PassKit/passkit-golang-grpc-sdk/io"
"github.com/PassKit/passkit-golang-grpc-sdk/io/single_use_coupons"
"github.com/PassKit/passkit-golang-grpc-quickstart/examples/shared"

"golang.org/x/net/context"
"google.golang.org/grpc/metadata"
)

// CreateCoupon takes campaignId, offerId and couponDetails, creates a new coupon record, and sends a welcome email to deliver coupon card url.
// The method returns the coupon id. Coupon id is a part of card url.
func CreateCoupon(campaignId, offerId, emailAddress string) string {
fmt.Println("Start creating a coupon...")

// Generate a coupons module client
pkCouponsClient := single_use_coupons.NewSingleUseCouponsClient(shared.Conn)

// Generate context object to connect to the server.
ctx := context.Background()
ctx = metadata.NewOutgoingContext(ctx, nil)

// Generates coupon with mandatory fields, more fields can be added, refer to docs.passkit.io and select Coupons for the full list
newCoupon := &single_use_coupons.Coupon{
OfferId: offerId,
CampaignId: campaignId,
Person: &io.Person{
Surname: "Smith",
Forename: "Bailey",
DisplayName: "Bailey",
EmailAddress: emailAddress,
},
}

couponId, err := pkCouponsClient.CreateCoupon(ctx, newCoupon)
if err != nil {
log.Fatalf("Create coupon err: %v", err)
}

fmt.Printf("Create Coupon Success: You have successfully created a coupon. Your coupon id is %s.\n", couponId.Id)
fmt.Printf("To check this user's coupon card, please visit https://pub1.pskt.io/%s\n", couponId.Id)

return couponId.Id
}

Loading