Keyboard shortcuts

Press or to navigate between chapters

Press S or / to search in the book

Press ? to show this help

Press Esc to hide this help

Voucher in Marketplace

Description / Background

Current packages on our platforms are linked to the allotments we receive from the partners. When the given allotment is full while rooms are still available at the hotels/restaurants, we lose the opportunity to sell more.

Glossary

Private (https://app.clickup.com/9003122396/docs/8ca1fpw-35796/8ca1fpw-41516)

Objectives

  • One voucher number = one voucher
  • Prepayment 100%
  • Use pricing tier based on quantity per booking (e.g. Price per 10 vouchers)
  • Voucher number format = VC-XXXXXX
  • Voucher can't be canceled or refund
  • Voucher in marketplace is set up as a package via Package module on Admin dashboard
  • This package type does not link to allotment
  • Admin can set up the quantity of the voucher for each package
  • Admin can set up the maximum vouchers allowed for each order
  • Admin can set up the selling period (Start - End date)
  • Admin can set up the validity period (Start - End date)
  • Admin can set up the full and discounted price
  • Admin can set up the voucher description (the same way we set up the menu)
  • User can buy the vouchers through Hungry Hub during the selling period
  • User can mix and match among Voucher package types per order
  • User can pay using QR and CC
  • User receives review link after redeem the voucher
    • Dine-in = same as the existing time
    • Xperience = 3 days after, one more time if not reviewed 30 days after
  • Owner receive the e-voucher with the voucher number and order ID via email
  • User must redeem the voucher within the validity period
  • User can earn Hungry Points by buying the Vouchers
  • Owner can see stats of sold/redeemed vouchers for each package (Quantity in orders and covers and revenue)
  • Owner can access to the Voucher Redemption page
  • Owner redeem the voucher by inputting the voucher numbers from the e-voucher or scan the QR code
  • Owner can keep track of redeemed vouchers as Voucher redemption list
  • Owner can add Guest details during the voucher redemption if the voucher buyer and the person staying/dining in is different
  • Admin can Hide VIM by picking "Hide in Store Page" on VIM setting
  • Singapore user now can book Voucher In Marketplace

Location

From Admin dashboard:

Packages menu ➝ Voucher in Marketplace

From Owner dashboard:

Voucher in Market Place menu ➝ Open Voucher in Market Place List to redeem the voucher(s)

How to find Voucher in Marketplace Order(s)

From Admin Dashboard:

Booking menu ➝ Voucher Order List

Admin can download the VIM report there

From Owner Dashboard:

Voucher in Market Place menu ➝ Voucher in Market Place List If owner wants to check the voucher that has been redeemed open Voucher in Market Place menu ➝ Voucher in Market Place History

Owner can add guest detail on Voucher In Market Place History if needed

How to set VIM

  1. Go to Admin Dashboard
  2. Click Packages menu
  3. Click Voucher in Market Place sub menu
  4. Direct to Voucher in Market Place Overview

[

hungryhub.com

https://hungryhub.com/admin/ticket_groups?locale=en

](https://hungryhub.com/admin/ticket_groups?locale=en)

  1. Find Add New button
  2. Click Add New button
  3. Direct to Create Voucher in Market Place page

[

hungryhub.com

https://hungryhub.com/admin/ticket_groups/new?locale=en

](https://hungryhub.com/admin/ticket_groups/new?locale=en)

  1. Fill the fields (country, voucher name (TH and EN), cover image (for the voucher), custom labels, description (TH and EN))
  2. Select the restaurant(s) where we want to have the VIM
  3. Select the Package Type
  4. Choose the image file and set the Rank for TNC
  5. Set the Selling Period, Validity Period, Original Price (THB), Discount Price (THB), Voucher Quantity, Limit per Order, Payment Type, Commission, Accept Gift Card, Pre-Payment 100%, Allow Mix and Match Package

Make sure that end selling period and end validity period is not the same, validity period must be ended after selling period

  1. Fill commission
  2. Visibility: Select Show in Store page to have the VIM displayed on Store page / Restaurant Detail page
  3. Click Save button
  4. Check on the selected restaurant(s) to make sure the VIM set is displayed

VIM Customisation setting

Admin can limit users buying by checking the "Limit Users", the number that admin input here will be the maximum number of users buying.

VIM behavior

  1. VIM with expired Selling Period would still be displayed on Restaurant Detail page
  2. Sold Out VIM would still be displayed on Restaurant Detail page
  3. VIM with today as the end of Selling Period would still be displayed on Restaurant Detail page
  4. VIM with expired Validity Period should not be displayed on Restaurant Detail page
  5. Quota Vouchers will return if VIM was canceled and VIM can be purchased again
  6. On the Quota and Quota witness admin pages, the quota amount will increase if VIM is canceled

Sequence Diagram / Flow

[

Flowchart Maker & Online Diagram Software

draw.io is free online diagram software for making flowcharts, process diagrams, org charts, UML, ER and network diagrams

https://app.diagrams.net/#G1HoljNWFbjmE0RaJn4JI20k3uDg0iEjCd

](https://app.diagrams.net/#G1HoljNWFbjmE0RaJn4JI20k3uDg0iEjCd)

ERD

[

drive.google.com

https://drive.google.com/file/d/1fFGycNXDGZLr7O8CpIdFJdDgunfO1VE2/view

](https://drive.google.com/file/d/1fFGycNXDGZLr7O8CpIdFJdDgunfO1VE2/view)

Backend Implementation

  • Add new model ticket and ticket group (we call it ticket on database)
  • implement earn point every voucher marketplace transaction
  • voucher marketplace effect on loyalty program
  • add send email confirmation, after buying the voucher

Frontend Implementation

Lock API

POST: {{ base_api }}/ticket_transactions/lock.json params:

{
	"restaurant_id": "997",
	"ticket_groups": [
		{
			"id": "5",
			"quantity": 1
		}
	],
	"access_token": "Pn1bJ0FbA989JU2JeUmQQj2wOfmTtrneoUg7SwebVb0"  (blank for guest)
}

Response:

{
	"data": {
		"transaction_id": 468,
		"expired_at": "2023-09-20T08:53:57Z"
	},
	"success": true,
	"message": "lock transaction success"
}

if failed

{
	"success": false,
	"message": "Sorry, the quantity is not sufficient.",
	"data": null
}

Cancel API:

PUT : {{ base_api }}/ticket_transactions/558/cancel.json

triggered when back from payment page and cancel QR payment

Create transaction API:

POST: {{ base_api }}/ticket_transactions**/submit.json** This is a new API different than previous one

{
	"restaurant_id": "997",
	"ticket_transaction_id": 558,
	"payment_type": "promptpay",
	"gb_primepay_card": {},
	"provider": "hungryhub",
	"channel": "web",
	"source": "website",
	"access_token": "Pn1bJ0FbA989JU2JeUmQQj2wOfmTtrneoUg7SwebVb0"
}

Design

https://www.figma.com/file/ymdLN2ZKzfI0oyX4kps2GQ/Partner-Portal-Desktop?type=design&node-id=2355-13026&mode=design

[

Email UI design (All)

Created with Figma

https://www.figma.com/file/Ob2Y8ifLbZUGxLy61epbIM/Email-UI-design-(All)?type=design&node-id=878-12&mode=design

](https://www.figma.com/file/Ob2Y8ifLbZUGxLy61epbIM/Email-UI-design-(All)?type=design&node-id=878-12&mode=design)

API Blueprint

Voucher in Marketplace table name is : Ticket Groups GET Ticket Groups URL: {{ base_api }}/ticket_groups.json Payload:

    { 	"restaurant_id" : 997 }

GET User Tickets URL: {{ base_api }}/users/tickets.json: Payload:

  {   "access_token": "Oi_60629gs9fIwc-ZClNFdrx-y09-b2AgBazpsXe0vw", 	"section_type": "unredeemed/redeemed", 	"minor_version": "{{ minor_version  }}" }

Improvement:

1. Create a secondary quota data using Redis

Createa a new Ruby class that responsible to track voucher quota, similar to InvWitness example in Booking feature https://hungryhub.com/admin/restaurants/933/inventories?date=2023-09-15&locale=en Update the redis data when TicketGroup has been created/updated/destroyed, when a new order created/updated/destroyed

2. Update Voucher List Page

Private (https://app.clickup.com/t/860rku97k) we should check the voucher quota on this page, if the quota is empty, then say sold out

  1. make the BUY button to be disabled first
  2. get the availability API from backend

Read data from Redis, not DB

[
  { id: xx, quota: 10 }
]

GET {{ base_api }}/ticket_groups/availability.json?restaurant_id=997 Response:

{
	"data": [
		{
			"id": "5",
			"quota": "0"
		},
		{
			"id": "6",
			"quota": "100"
		},
		{
			"id": "7",
			"quota": "81"
		},
		{
			"id": "8",
			"quota": "10"
		}
	],
	"success": true
}

Compare available ticket with active ticket, if quota null make it sold out. Example result:

3.Update CONFIRM/NEXT button in Voucher List Page

Private (https://app.clickup.com/t/860rku9fq) once user click the confirm button, create a temporary order id to lock the order and reduce the quota.

update Redis first, then update the DB order active value should be false

system will cancel the order if it's still pending for 10 minutes

quota = quota - selected quantity
active = false

Lock API POST: {{ base_api }}/ticket_transactions/lock.json params:

{
	"restaurant_id": "997",
	"ticket_groups": [
		{
			"id": "5",
			"quantity": 1
		}
	],
	"access_token": "Pn1bJ0FbA989JU2JeUmQQj2wOfmTtrneoUg7SwebVb0"  (blank for guest)
}

Response:

{
	"data": {
		"transaction_id": 468, 
		"expired_at": "2023-09-20T08:53:57Z"
	},
	"success": true,
	"message": "lock transaction success"
}

transaction_id is used for create ticket transaction

4. Extend Ticket Lock Timer

Private (https://app.clickup.com/t/860rtqwyk) The expired_at is used for cooldown timer on payment page. When we click "I need more time" will hit this API: PUT: {{ base_api }}/ticket_transactions/946/extend_session.json

{
    "client_type": "web"
}

Response:

{
	"data": {
		"transaction_id": 716,
		"expired_at": "2023-09-26T03:05:40Z"
	},
	"success": true,
	"message": "Extend session transaction success"
}

5. Cancel Ticket Lock and Transactions

Private (https://app.clickup.com/t/860rqct1t) cancel the order ID once user click back button from payment page and also on QR Payment Cancel API: PUT : {{ base_api }}/ticket_transactions/558/cancel.json

{
    "client_type": "web"
}

6. Update Payload Payment Page

Private (https://app.clickup.com/t/860rku9qw) change the active value to be true once the order has been paid

Before updating the active status to be true Check the availability again in Redis and Database if Redis says not available, then reject the order request

another step is checking on DB level DB says not available (quota already zero), then reject the order request

Create transaction API: POST: {{ base_api }}/ticket_transactions**/submit.json** This API is similar with our ticket_transactions API. But we replace ticket_group with ticket_transaction_id

{
	"restaurant_id": "997",
	"ticket_transaction_id": 558,
	"payment_type": "promptpay",
	"gb_primepay_card": {},
	"provider": "hungryhub",
	"channel": "web",
	"source": "website",
	"access_token": "Pn1bJ0FbA989JU2JeUmQQj2wOfmTtrneoUg7SwebVb0"
}

Response:

{
	"data": {
		"id": "506",
		"type": "ticket_transactions",
		"attributes": {
			"encrypted_id": "3Ed0Y",
			"phone": "66912392189",
			"status_as_symbol": "waiting_for_payment",
			"user_id": 188442,
			"qr_code_for_payment": "https://hh-engineering.my.id/uploads/externals/omise/source/qr_code/20042/qrcode-196-2023-09-19025811UTC20230919-614791-v415wp.png",
			"total_price": {
				"price": 1600,
				"currency": "THB",
				"format": "฿1,600"
			},
			"name": "Test First Time",
			"email": "firsttimer@gmail.com",
			"qrcode": "",
			"qr_code_for_payment_expired_at": "2023-09-19T03:00:17Z",
			"hungry_points": 64,
			"payment_type": null,
			"charge_price": {
				"price": 1600,
				"currency": "THB",
				"format": "฿1,600"
			}
		},
		"relationships": {
			"tickets": {
				"data": [
					{
						"id": "826",
						"type": "tickets"
					},
					{
						"id": "827",
						"type": "tickets"
					}
				]
			},
			"ticket_bundles": {
				"data": [
					{
						"id": "575",
						"type": "ticket_bundles"
					}
				]
			}
		}
	},
	"included": [
		{
			"id": "826",
			"type": "tickets",
			"attributes": {
				"encrypted_id": "Ooyk3",
				"redeemed_at": null,
				"ticket_group_id": 5,
				"ticket_code": "VC-D08C4EBB70",
				"valid_start_date": "2023-03-24",
				"valid_end_date": "2031-11-30",
				"active": false,
				"name": "Ayce For You",
				"amount": {
					"format": "฿800",
					"amount": 800.0,
					"currency": "THB"
				},
				"qrcode": ""
			}
		},
		{
			"id": "827",
			"type": "tickets",
			"attributes": {
				"encrypted_id": "ON92R",
				"redeemed_at": null,
				"ticket_group_id": 5,
				"ticket_code": "VC-2FFE0C72BC",
				"valid_start_date": "2023-03-24",
				"valid_end_date": "2031-11-30",
				"active": false,
				"name": "Ayce For You",
				"amount": {
					"format": "฿800",
					"amount": 800.0,
					"currency": "THB"
				},
				"qrcode": ""
			}
		},
		{
			"id": "575",
			"type": "ticket_bundles",
			"attributes": {
				"ticket_group_bundle": null,
				"quantity": 2,
				"discount_percent": null,
				"name": "Ayce For You",
				"description": "I have to be in a room by your house in a room with the new code for your phone was on your desk at work but you need a new job and you need a job at a job and a job at work at a job and I will call the doctor and tell them they will not have the right ",
				"price_per_ticket": {
					"price": 800,
					"currency": "THB",
					"format": "฿800"
				},
				"total_price": {
					"price": 1600,
					"currency": "THB",
					"format": "฿1,600"
				}
			}
		}
	],
	"success": true,
	"message": "Purchase vouchers success"
}


Locking System

to prevent oversold issue, we need to implement a locking system for VIM

implementation details: Private (https://app.clickup.com/t/860rjen10) FE: BE: Private (https://app.clickup.com/t/860rjenbv)

there are few pending items that we will continue on the next phase: https://github.com/hungryhub-team/hh-server/pull/4995

Quota Witness ada kaitan dg locking system, hrus sama dg quota biasa, quota witness dari redis, validasi quota di level quota witness dlu biar cepet

Hide store page = VIM is invisible

ketika bikin suatu ticket group sistem bikin list ticket pool, klo quantitynya 1-10, maka bikin pool dengan 10 member misal 105-ticket-1 105-ticket-2 105-ticket-3 sampe 105-ticket-10ketika ngedit ticket group quota, maka sistem akan generate lagi misalnya di edit jd 5 maka available ticket pool nya adalah 105-ticket-1pool pertama, kedua, dan ketiga itu gak ada hubungannya jika ada pool baru, maka pool lama akan di hapus member dari masing2 pool itu gak ada hubungannyaitu cuma ideku biar gak ada kasus oversold aja kasus oversold bisa di identifikasi klo ada member yg duplicate dalam pool yg samamungkin di rename aja nama member poolnya, biar gak bikin bingungmisalnya"{ticket-groupid}-{pool timestamp}-{member number}"