Last active
June 3, 2025 13:50
-
-
Save thuykaka/24a195ea4ec770ae298e5326e3b1bac9 to your computer and use it in GitHub Desktop.
Tạo Bank QR (giống VietQR)
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| /* | |
| https://vietqr.net/portal-service/download/documents/QR_Format_T&C_v1.0_VN_092021.pdf | |
| Nội dung của mã QR động sẽ có các phần ghép lại tạo thành 1 nội dung hoàn chỉnh (mỗi phần bao gồm mã ID, độ dài của giá trị và giá trị): | |
| 1. Payload Format Indicator | |
| Ví dụ: 000210 -> ID: 00, Độ dài: 02, Giá trị: 01. | |
| 2. Point of Initiation Method | |
| Ví dụ: 010212 -> ID: 01, Độ dài: 02, Giá trị: 12. | |
| Trong đó: | |
| - 11 = QR tĩnh – áp dụng khi mã QR cho phép thực hiện nhiều lần giao dịch | |
| - 12 = QR động – áp dụng khi mã QR chỉ cho phép thực hiện một lần giao dịch | |
| 3. Consumer Account Information | |
| Ví dụ: 38530010A000000727012300069704230109mynamebvh0208QRIBFTTA -> ID: 38, Độ dài: 53, Giá trị: 0010A000000727012300069704230109mynamebvh0208QRIBFTTA | |
| Giá trị gồm 3 thành phần: | |
| - Định danh toàn cầu (GUID): 0010A000000727 -> ID: 00, Độ dài: 10, Giá trị: A000000727 | |
| - Tổ chức ngân hàng thụ hưởng: 012300069704230109mynamebvh -> ID: 01, Độ dài: 23, Giá trị: 00069704230109mynamebvh. | |
| Giá trị này lại chia làm 2 phần: | |
| - Acquier ID/BNB ID: 0006970423 -> ID: 00, Độ dài: 06, Giá trị: 970423 (970423 chính là mã BIN TPBank). | |
| - Merchant ID/Consumer ID: 0109mynamebvh, ID: 01, Độ dài: 09, Giá trị: mynamebvh (Chính là số tài khoản). | |
| - Mã dịch vụ: 0208QRIBFTTA -> ID: 02, Độ dài: 08, Giá trị: QRIBFTTA (QRIBFTTA: dịch vụ chuyển nhanh NAPAS247 bằng mã QR đến Tài khoản) | |
| 4. Transaction Currency | |
| Ví dụ: 5303704 -> ID: 53, Độ dài: 03, Giá trị: 704 (704 mã tiền tệ VN) | |
| 5. Transaction Amount | |
| Ví dụ: 5406180000 -> ID: 54, Độ dài: 06, Giá trị: 180000 (số tiền chuyển khoản) | |
| 6. Country Code | |
| Ví dụ: 5802VN -> ID: 58, Độ dài: 02, Giá trị: VN (mã quốc gia) | |
| 7. Additional Data Field Template | |
| Ví dụ: 62080804test -> ID: 62, Độ dài: 08, Giá trị: 0804test | |
| Giá trị gồm: ID: 08, Độ dài 04, Giá trị: test (là nội dung chuyển khoản) | |
| 8. CRC | |
| 6304AB76 -> ID: 63, Độ dài: 04, Giá trị: AB76 | |
| Đây là một phương pháp kiểm tra tính toàn vẹn dữ liệu trong việc truyền tải thông tin. | |
| Ở đây họ sử dụng CRC 0xFFFF. | |
| Cách tính là lấy nội dung của QRCode như ví dụ trên sẽ là 00020101021238530010A000000727012300069704230109mynamebvh0208QRIBFTTA53037045405500005802VN62080804test6304 | |
| rồi cho qua hàm tính CRC sẽ ra kết quả: AB76. | |
| */ |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| import qrcode, { type QRCodeToBufferOptions, type QRCodeToDataURLOptions } from 'qrcode'; | |
| import { z } from 'zod'; | |
| import { Jimp } from 'jimp'; | |
| type PointOfInitMethod = 'static' | 'dynamic'; | |
| type QrCodeLogoOptions = { | |
| src: string; | |
| text?: string; | |
| backgroundColor?: string; | |
| size?: { | |
| w?: number; | |
| h?: number; | |
| }; | |
| position?: 'center' | 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right'; | |
| }; | |
| // Zod schema for validation | |
| const VietQrSchema = z.object({ | |
| bankBin: z | |
| .string() | |
| .min(1, 'Bank BIN or name is required') | |
| .refine(val => /^[0-9]{6}$/.test(val) || Object.keys(BANK_BINS).includes(val.toLowerCase()), { | |
| message: 'Invalid bank BIN code or bank name', | |
| }), | |
| accNo: z | |
| .string() | |
| .min(1, 'Account number is required') | |
| .max(19, 'Account number must be at most 19 digits') | |
| .regex(/^[0-9]+$/, 'Account number must contain only digits'), | |
| amt: z | |
| .string() | |
| .min(1, 'Amount is required') | |
| .max(13, 'Amount must be at most 13 digits') | |
| .regex(/^[0-9]+$/, 'Amount must contain only digits') | |
| .refine(val => parseInt(val) > 0, { | |
| message: 'Amount must be a positive number', | |
| }), | |
| desc: z | |
| .string() | |
| .min(1, 'Description is required') | |
| .max(50, 'Description must be at most 50 characters') | |
| .regex(/^[a-zA-Z0-9\s]+$/, 'Description must contain only letters, numbers and spaces') | |
| .transform(val => val.trim()) | |
| .refine(val => val.length <= 50, { | |
| message: 'Description must be at most 50 characters after trimming', | |
| }), | |
| initMethod: z.enum(['static', 'dynamic']).optional(), | |
| }); | |
| type VietQrConfig = z.infer<typeof VietQrSchema>; | |
| // Bank BIN codes mapping | |
| const BANK_BINS: Record<string, string> = { | |
| // Ngân hàng TMCP Sài Gòn Công Thương | |
| saigonbank: '970400', | |
| sgb: '970400', | |
| 'ngan hang sai gon cong thuong': '970400', | |
| // Ngân hàng TMCP Sài Gòn Thương Tín | |
| sacombank: '970403', | |
| stb: '970403', | |
| 'ngan hang sai gon thuong tin': '970403', | |
| // Ngân hàng Nông nghiệp và Phát triển Nông thôn Việt Nam | |
| agribank: '970405', | |
| 'ngan hang nong nghiep va phat trien nong thon viet nam': '970405', | |
| // Ngân hàng TMCP Đông Á | |
| dongabank: '970406', | |
| 'ngan hang dong a': '970406', | |
| // Ngân hàng TMCP Kỹ Thương Việt Nam | |
| techcombank: '970407', | |
| tcb: '970407', | |
| 'ngan hang ky thuong viet nam': '970407', | |
| // Ngân hàng TNHH Một Thành Viên Dầu Khí Toàn Cầu | |
| gpbank: '970408', | |
| 'ngan hang dau khi toan cau': '970408', | |
| // Ngân hàng TMCP Bắc Á | |
| bab: '970409', | |
| 'ngan hang bac a': '970409', | |
| // Ngân hàng TNHH Một Thành Viên Standard Chartered | |
| stanchart: '970410', | |
| scvn: '970410', | |
| 'ngan hang standard chartered viet nam': '970410', | |
| // Ngân hàng TMCP Đại Chúng Việt Nam | |
| pvcombank: '970412', | |
| 'ngan hang dai chung viet nam': '970412', | |
| // Ngân hàng TNHH Một Thành Viên Đại Dương | |
| oceanbank: '970414', | |
| ocean: '970414', | |
| // Ngân hàng TMCP Công Thương Việt Nam | |
| vietinbank: '970415', | |
| 'ngan hang cong thuong viet nam': '970415', | |
| // Ngân hàng TMCP Á Châu | |
| acb: '970416', | |
| 'ngan hang a chau': '970416', | |
| // Ngân hàng Đầu tư và Phát triển Việt Nam | |
| bidv: '970418', | |
| 'ngan hang dau tu va phat trien viet nam': '970418', | |
| // Ngân hàng TMCP Quốc Dân | |
| ncb: '970419', | |
| 'ngan hang quoc dan': '970419', | |
| // Ngân hàng Liên doanh Việt Nga | |
| vrb: '970421', | |
| 'ngan hang lien doanh viet nga': '970421', | |
| // Ngân hàng TMCP Quân Đội | |
| mb: '970422', | |
| 'ngan hang quan doi': '970422', | |
| // Ngân hàng TMCP Tiên Phong | |
| tpbank: '970423', | |
| 'ngan hang tien phong': '970423', | |
| // Ngân hàng TNHH Một Thành Viên Shinhan Việt Nam | |
| shinhan: '970424', | |
| 'ngan hang shinhan viet nam': '970424', | |
| // Ngân hàng TMCP An Bình | |
| abbank: '970425', | |
| 'ngan hang an binh': '970425', | |
| // Ngân hàng TMCP Hàng Hải | |
| msb: '970426', | |
| 'ngan hang hang hai': '970426', | |
| // Ngân hàng TMCP Việt Á | |
| vietabank: '970427', | |
| 'ngan hang viet a': '970427', | |
| // Ngân hàng TMCP Nam Á | |
| nama: '970428', | |
| 'ngan hang nam a': '970428', | |
| // Ngân hàng TMCP Xăng dầu Petrolimex | |
| pgbank: '970430', | |
| 'ngan hang thinh vuong va phat trien': '970430', | |
| // Ngân hàng TMCP Xuất Nhập khẩu Việt Nam | |
| eximbank: '970431', | |
| 'ngan hang xuat nhap khau viet nam': '970431', | |
| // Ngân hàng TMCP Việt Nam Thịnh Vượng | |
| vpbank: '970432', | |
| 'ngan hang viet nam thinh vuong': '970432', | |
| // Ngân hàng TMCP Việt Nam Thương Tín | |
| vietbank: '970433', | |
| 'ngan hang viet nam thuong tin': '970433', | |
| // Ngân hàng TNHH Indovina | |
| indovina: '970434', | |
| 'ngan hang indovina': '970434', | |
| // Ngân hàng TMCP Ngoại thương Việt Nam | |
| vietcombank: '970436', | |
| vcb: '970436', | |
| 'ngan hang ngoai thuong viet nam': '970436', | |
| // Ngân hàng TMCP Phát triển TP.HCM | |
| hdbank: '970437', | |
| 'ngan hang phat trien tp hcm': '970437', | |
| // Ngân hàng TMCP Bảo Việt | |
| bvb: '970438', | |
| baovietbank: '970438', | |
| 'ngan hang bao viet': '970438', | |
| // Ngân hàng TNHH Một Thành Viên Public Việt Nam | |
| publicbank: '970439', | |
| pbvn: '970439', | |
| 'ngan hang public viet nam': '970439', | |
| // Ngân hàng TMCP Đông Nam Á | |
| seabank: '970440', | |
| 'ngan hang dong nam a': '970440', | |
| // Ngân hàng TMCP Quốc Tế Việt Nam | |
| vib: '970441', | |
| 'ngan hang quoc te viet nam': '970441', | |
| // Ngân hàng TNHH Một Thành Viên Hong Leong Việt Nam | |
| hongleong: '970442', | |
| hlbvn: '970442', | |
| 'ngan hang hong leong viet nam': '970442', | |
| // Ngân hàng TMCP Sài Gòn - Hà Nội | |
| shb: '970443', | |
| 'ngan hang sai gon ha noi': '970443', | |
| // Ngân hàng TNHH Một Thành Viên Xây Dựng Việt Nam | |
| cbbank: '970444', | |
| 'ngan hang xay dung viet nam': '970444', | |
| // Ngân hàng Hợp Tác Xã Việt Nam | |
| coopbank: '970446', | |
| // Ngân hàng TMCP Phương Đông | |
| ocb: '970448', | |
| // Ngân hàng TMCP Kiên Long | |
| kienlong: '970452', | |
| klb: '970452', | |
| // Ngân hàng TMCP Bản Việt | |
| bvbank: '970454', | |
| // Ngân hàng Công nghiệp Hàn Quốc - Chi nhánh Hà Nội | |
| ibk_hanoi: '970455', | |
| // Ngân hàng Industrial Bank of Korea - Chi nhánh Hồ Chí Minh | |
| ibk_hcm: '970456', | |
| // Ngân hàng TNHH Một Thành Viên Woori Bank Việt Nam | |
| wooribank: '970457', | |
| // Ngân hàng TNHH Một Thành Viên UOB Việt Nam | |
| uob: '970458', | |
| // Ngân hàng TNHH Một Thành Viên CIMB Việt Nam | |
| cimb: '970459', | |
| }; | |
| /** | |
| * Get bank BIN code from bank name | |
| * @param bankName - Bank name or code (case insensitive) | |
| * @returns Bank BIN code or undefined if not found | |
| */ | |
| function getBankBin(bankName: string): string | undefined { | |
| return BANK_BINS[bankName.toLowerCase()]; | |
| } | |
| /** | |
| * Check if a string is a valid bank BIN code | |
| * @param bin - String to check | |
| * @returns boolean indicating if the string is a valid BIN code | |
| */ | |
| function isValidBin(bin: string): boolean { | |
| return /^[0-9]{6}$/.test(bin); | |
| } | |
| /** | |
| * Get bank BIN code from either bank name or direct BIN code | |
| * @param input - Bank name or BIN code | |
| * @returns Bank BIN code or undefined if not found | |
| * @throws Error if input is invalid | |
| */ | |
| function resolveBankBin(input: string): string { | |
| // If input is a valid BIN code, use it directly | |
| if (isValidBin(input)) return input; | |
| // Try to get BIN code from bank name | |
| const bin = getBankBin(input); | |
| if (bin) return bin; | |
| throw new Error(`Invalid bank name or BIN code: ${input}, refer: https://www.sbv.gov.vn/webcenter/portal/vi/menu/trangchu/ttvnq/htmtcqht`); | |
| } | |
| class VietQr { | |
| private static readonly PFI = '000210'; // PAYLOAD_FORMAT_INDICATOR | |
| private static readonly GUID = '0010A000000727'; | |
| private static readonly SVC = '0208QRIBFTTA'; // SERVICE_CODE | |
| private static readonly CURR = '5303704'; // CURRENCY | |
| private static readonly CC = '5802VN'; // COUNTRY_CODE | |
| private readonly initMethod: string; | |
| private readonly accInfo: string; | |
| private readonly txAmt: string; | |
| private readonly addData: string; | |
| /** | |
| * Creates a new VietQR code generator instance | |
| * @param cfg Configuration object containing all required parameters | |
| * @param cfg.bankBin - Bank BIN code or bank name from https://www.sbv.gov.vn/webcenter/portal/vi/menu/trangchu/ttvnq/htmtcqht | |
| * @param cfg.accNo - Recipient's bank account number | |
| * @param cfg.amt - Transfer amount (positive number, max 13 digits) | |
| * @param cfg.desc - Transfer description (max 50 characters, no special characters) | |
| * @param cfg.initMethod - QR type: 'static' or 'dynamic' | |
| * @throws Error if validation fails | |
| */ | |
| constructor(cfg: VietQrConfig) { | |
| // Validate input using Zod schema | |
| const validatedConfig = VietQrSchema.parse(cfg); | |
| const bankBin = resolveBankBin(validatedConfig.bankBin); | |
| this.initMethod = this.buildInitMethod(validatedConfig.initMethod); | |
| this.accInfo = this.buildAccInfo(bankBin, validatedConfig.accNo); | |
| this.txAmt = this.buildTxAmt(validatedConfig.amt); | |
| this.addData = this.buildAddData(validatedConfig.desc); | |
| } | |
| private static getSize(txt: string): string { | |
| return txt.length.toString().padStart(2, '0'); | |
| } | |
| private buildInitMethod(method?: PointOfInitMethod): string { | |
| return method === 'static' ? '010211' : '010212'; | |
| } | |
| private buildAccInfo(bin: string, acc: string): string { | |
| const binSize = VietQr.getSize(bin); | |
| const binPart = `00${binSize}${bin}`; | |
| const accSize = VietQr.getSize(acc); | |
| const accPart = `01${accSize}${acc}`; | |
| const org = `01${VietQr.getSize(`${binPart}${accPart}`)}${binPart}${accPart}`; | |
| const info = `${VietQr.GUID}${org}${VietQr.SVC}`; | |
| return `38${VietQr.getSize(info)}${info}`; | |
| } | |
| private buildTxAmt(amt: string): string { | |
| return `54${VietQr.getSize(amt)}${amt}`; | |
| } | |
| private buildAddData(desc: string): string { | |
| const val = `08${VietQr.getSize(desc)}${desc}`; | |
| return `62${VietQr.getSize(val)}${val}`; | |
| } | |
| private static calcCrc(txt: string): string { | |
| let crc = 0xffff; | |
| for (let i = 0; i < txt.length; i++) { | |
| crc ^= txt.charCodeAt(i) << 8; | |
| for (let j = 0; j < 8; j++) { | |
| crc = (crc & 0x8000) !== 0 ? ((crc << 1) ^ 0x1021) & 0xffff : (crc << 1) & 0xffff; | |
| } | |
| } | |
| return crc.toString(16).toUpperCase().padStart(4, '0'); | |
| } | |
| /** | |
| * Generates the QR code string | |
| * @returns The complete QR code string | |
| */ | |
| public getQrContent(): string { | |
| const content = [VietQr.PFI, this.initMethod, this.accInfo, VietQr.CURR, this.txAmt, VietQr.CC, this.addData, '6304'].join(''); | |
| return `${content}${VietQr.calcCrc(content)}`; | |
| } | |
| public async getQrDataUrl(options: QRCodeToDataURLOptions): Promise<string> { | |
| return qrcode.toDataURL(this.getQrContent(), { width: 250, ...options }); | |
| } | |
| private detectUrlType(url: string): 'url' | 'base64' | undefined { | |
| if (/^data:image\/png;base64,/.test(url)) return 'base64'; | |
| if (z.string().url().safeParse(url).success) return 'url'; | |
| return undefined; | |
| } | |
| public async getQrWithLogo(qrCodeOptions: QRCodeToBufferOptions, logoOptions: QrCodeLogoOptions): Promise<string> { | |
| const qrImage = await Jimp.read(await qrcode.toBuffer(this.getQrContent(), { width: 250, ...qrCodeOptions })); | |
| const logoSrc = logoOptions.src; | |
| const logoType = this.detectUrlType(logoSrc); | |
| if (!logoType) throw new Error('Invalid logo source'); | |
| const logoImage = await Jimp.read(logoType === 'url' ? logoSrc : Buffer.from(logoSrc.split(',')[1]!, 'base64')); | |
| const logoRatio = logoImage.bitmap.height / logoImage.bitmap.width; | |
| const logoWidth = Math.min(logoOptions.size?.w ?? qrImage.bitmap.width * 0.25, qrImage.bitmap.width * 0.25); | |
| const logoHeight = logoOptions.size?.h ?? Math.round(logoWidth * logoRatio); | |
| logoImage.resize({ w: logoWidth, h: logoHeight }); | |
| // Center the logo in the QR code | |
| const x = qrImage.bitmap.width / 2 - logoWidth / 2; | |
| const y = qrImage.bitmap.height / 2 - logoHeight / 2; | |
| if (!logoOptions.backgroundColor) { | |
| qrImage.composite(logoImage, x, y); | |
| } else { | |
| const backgroundImage = new Jimp({ width: logoWidth, height: logoHeight, color: logoOptions.backgroundColor }); | |
| backgroundImage.composite(logoImage, 0, 0); | |
| qrImage.composite(backgroundImage, x, y); | |
| } | |
| return qrImage.getBase64('image/png'); | |
| } | |
| } | |
| export { VietQr, type VietQrConfig }; |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| const vietQr = new VietQr({ bankBin: '970415', accNo: '0982447171', amt: '50000', desc: 'thuyck' }); | |
| const qrOptions = {}; | |
| const logoOptions = { | |
| src: `https://www.google.com/images/branding/googlelogo/1x/googlelogo_color_272x92dp.png`, | |
| backgroundColor: '#ffffff', | |
| }; | |
| const url = await vietQr.getQrWithLogo(qrOptions, logoOptions); | |
| console.log('url:', url); | |
| //  |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment