BR Code

A crate to parse and emit PIX BR Code. * Technical and Business specs for BR Code usage

Important Changes

Usage

toml [dependencies] brcode = "1.4.1"

Build from source

  1. Install rustup.
  2. make build-macos for macos and ios files or make build-linux for linux and android files.
  3. Files will be located at target/release/libbrcode.*, target/<target-platform>/release/libbrcode.so.

Copy files from Github Release

Shellscript to get files from release:

So sh curl -s https://api.github.com/repos/naomijub/brcode/releases/latest \ | grep "browser_download_url.*so" \ | cut -d : -f 2,3 \ | tr -d \" \ | wget -qi -

dylib sh curl -s https://api.github.com/repos/naomijub/brcode/releases/latest \ | grep "browser_download_url.*dylib" \ | cut -d : -f 2,3 \ | tr -d \" \ | wget -qi -

Example

Parse String ```rust use brcode::{from_str, Data};

fn main() { let code = "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38";

assert_eq!(from_str(code), expected());

}

fn expected() -> Vec<(usize, Data)> { vec![ (0, Data::Single("01".tostring())), (4, Data::Single("12345678901234".tostring())), (26, Data::Vector(vec![ (0, Data::Single("BR.GOV.BCB.PIX".tostring())), (1, Data::Single("123e4567-e12b-12d1-a456-426655440000".tostring()))])), (27, Data::Vector(vec![ (0, Data::Single("BR.COM.OUTRO".tostring())), (1, Data::Single("0123456789".tostring()))])), (52, Data::Single("0000".tostring())), (53, Data::Single("986".tostring())), (54, Data::Single("123.45".tostring())), (58, Data::Single("BR".tostring())), (59, Data::Single("NOME DO RECEBEDOR".tostring())), (60, Data::Single("BRASILIA".tostring())), (61, Data::Single("70074900".tostring())), (62, Data::Vector(vec![ (5, Data::Single("RP12345678-2019".tostring()))])), (80, Data::Vector(vec![( 0, Data::Single("BR.COM.OUTRO".tostring())), (1, Data::Single("0123.ABCD.3456.WXYZ".tostring()))])), (63, Data::Single("AD38".to_string()))] } ```

strtobrcode ```rust use brcode::{strtobrcode, BrCode, Template, Info, MerchantInfo, Label};

fn main() { let code = "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38";

assert_eq!(from_str(code), expected());

}

fn expected() -> BrCode { BrCode { payloadversion: 1, initiationmethod: None, merchantcategorycode: 0000u32, merchantname: "NOME DO RECEBEDOR".tostring(), merchantcity: "BRASILIA".tostring(), merchantinformation: vec![ MerchantInfo { id: 26, info: vec![ Info { id: 0, info: "BR.GOV.BCB.PIX".tostring(), }, Info { id: 1, info: "123e4567-e12b-12d1-a456-426655440000".tostring(), }, ], }, MerchantInfo { id: 27, info: vec![ Info { id: 0, info: "BR.COM.OUTRO".tostring(), }, Info { id: 1, info: "0123456789".tostring(), }, ], }, ], currency: "986".tostring(), postalcode: Some("70074900".tostring()), amount: Some(123.45), countrycode: "BR".tostring(), fieldtemplate: vec![Label { referencelabel: "RP12345678-2019".tostring(), }], crc1610: "AD38".tostring(), templates: Some(vec![Template { id: 80usize, info: vec![ Info { id: 0usize, info: "BR.COM.OUTRO".tostring(), }, Info { id: 1usize, info: "0123.ABCD.3456.WXYZ".tostring(), }, ], }]), } } ```

brcode::to_string from Vec<(usize, Data)>: ```rust use brcode::{ self, BrCode, };

fn main() { let actual = brcode::tostring(brcodevec()); let code = "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38";

assert_eq!(actual, code);

}

fn brcodevec() -> Vec<(usize, Data)> { vec![ (0, Data::Single("01".tostring())), (4, Data::Single("12345678901234".tostring())), ( 26, Data::Vector(vec![ (0, Data::Single("BR.GOV.BCB.PIX".tostring())), ( 1, Data::Single("123e4567-e12b-12d1-a456-426655440000".tostring()), ), ]), ), ( 27, Data::Vector(vec![ (0, Data::Single("BR.COM.OUTRO".tostring())), (1, Data::Single("0123456789".tostring())), ]), ), (52, Data::Single("0000".tostring())), (53, Data::Single("986".tostring())), (54, Data::Single("123.45".tostring())), (58, Data::Single("BR".tostring())), (59, Data::Single("NOME DO RECEBEDOR".tostring())), (60, Data::Single("BRASILIA".tostring())), (61, Data::Single("70074900".tostring())), ( 62, Data::Vector(vec![(5, Data::Single("RP12345678-2019".tostring()))]), ), ( 80, Data::Vector(vec![ (0, Data::Single("BR.COM.OUTRO".tostring())), (1, Data::Single("0123.ABCD.3456.WXYZ".tostring())), ]), ), (63, Data::Single("AD38".tostring())), ] }

```

brcode::brcodetostring for struct BrCode: ```rust use brcode::{ self, BrCode, };

fn main() { let actual = brcode::brcodetostring(brcode_value()); let code = "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38";

assert_eq!(actual, code);

}

fn brcodevalue() -> BrCode { BrCode { payloadversion: 1, initiationmethod: None, merchantaccountinformation: Some(String::from("12345678901234")), merchantcategorycode: 0000u32, merchantname: "NOME DO RECEBEDOR".tostring(), merchantcity: "BRASILIA".tostring(), merchantinformation: vec![ MerchantInfo { id: 26, info: vec![ Info { id: 0, info: "BR.GOV.BCB.PIX".tostring(), }, Info { id: 1, info: "123e4567-e12b-12d1-a456-426655440000".tostring(), }, ], }, MerchantInfo { id: 27, info: vec![ Info { id: 0, info: "BR.COM.OUTRO".tostring(), }, Info { id: 1, info: "0123456789".tostring(), }, ], }, ], currency: "986".tostring(), postalcode: Some("70074900".tostring()), amount: Some(123.45), countrycode: "BR".tostring(), fieldtemplate: vec![Label { referencelabel: "RP12345678-2019".tostring(), }], crc1610: "AD38".tostring(), templates: Some(vec![Template { id: 80usize, info: vec![ Info { id: 0usize, info: "BR.COM.OUTRO".tostring(), }, Info { id: 1usize, info: "0123.ABCD.3456.WXYZ".to_string(), }, ], }]), } } ```

BrCode functions context related

BrCode functions QrCode generation related:

Benchmark

from_str in benches/parse.rs time: [15.734 us 15.758 us 15.782 us]

strtobrcode in benches/to_brcode time: [24.886 us 24.931 us 24.977 us]

ednfrombrcode in benches/to_brcode time: [52.670 us 52.795 us 52.929 us]

jsonfrombrcode in benches/to_brcode time: [28.229 us 28.284 us 28.339 us]

both-ways using BrCode time: [33.238 us 33.555 us 33.924 us]

both-ways using Vec<(usize, Data)> time: [22.867 us 22.958 us 23.107 us]

crc16_ccitt in benches/crc16: time: [3.0738 us 3.0825 us 3.0938 us]

FFI

Clojure FFI

DOCS

BR Code as Edn call function FFI edn_from_brcode or use clojar [clj-brcode "1.1.0-SNAPSHOT"]. Example: ```clojure (ns example.core (:require [clj-brcode.core :refer :all]))

(def code "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38")

(brcode->edn code)

; {:payload-version 1, :initiation-method nil, :merchant-information [{:id 26, :info [{:id 0, :info "BR.GOV.BCB.PIX"}, {:id 1, :info "123e4567-e12b-12d1-a456-426655440000"}]}, {:id 27, :info [{:id 0, :info "BR.COM.OUTRO"}, {:id 1, :info "0123456789"}]}], :merchant-category-code 0, :merchant-name "NOME DO RECEBEDOR", :merchant-city "BRASILIA", :postal-code "70074900", :currency "986", :amount 123.45, :country-code "BR", :field-template [{:reference-label "RP12345678-2019"}], :crc1610 "AD38", :templates [{:id 80, :info [{:id 0, :info "BR.COM.OUTRO"}, {:id 1, :info "0123.ABCD.3456.WXYZ"}]}]} ```

Input: rust "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38"

Expected Edn: clojure {:payload-version 1,:initiation-method nil, :merchant-information [ {:id 26, :info [{ :id 0, :info "BR.GOV.BCB.PIX",}, {:id 1, :info "123e4567-e12b-12d1-a456-426655440000",}]}, {:id 27, :info [{ :id 0, :info "BR.COM.OUTRO",}, {:id 1, :info "0123456789",}]} ],:merchant-category-code 0, :merchant-name "NOME DO RECEBEDOR", :merchant-city "BRASILIA", :postal-code "70074900", :currency "986", :amount 123.45, :country-code "BR", :field-template [{ :reference-label "RP12345678-2019", }], :crc1610 "AD38", :templates [ { :id 80, :info [{ :id 0, :info "BR.COM.OUTRO", },{ :id 1, :info "0123.ABCD.3456.WXYZ", }], }] }

Edn as BR Code call function FFI edn_to_brcode or use clojar [clj-brcode "1.1.0-SNAPSHOT"]. Example: ```clojure (ns example.core (:require [clj-brcode.core :refer :all]))

(def edn {:payload-version 1, :initiation-method nil, :merchant-information [{:id 26, :info [{:id 0, :info "BR.GOV.BCB.PIX"}, {:id 1, :info "123e4567-e12b-12d1-a456-426655440000"}]}, {:id 27, :info [{:id 0, :info "BR.COM.OUTRO"}, {:id 1, :info "0123456789"}]}], :merchant-category-code 0, :merchant-name "NOME DO RECEBEDOR", :merchant-city "BRASILIA", :postal-code "70074900", :currency "986", :amount 123.45, :country-code "BR", :field-template [{:reference-label "RP12345678-2019"}], :crc1610 "AD38", :templates [{:id 80, :info [{:id 0, :info "BR.COM.OUTRO"}, {:id 1, :info "0123.ABCD.3456.WXYZ"}]}]})

(brcode->edn edn)

; "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38" ```

Other available functions: - json->brcode - brcode->json - crc16-ccitt

Clojure Benchmark with Criterium

brcode->edn Evaluation count : 4644 in 6 samples of 774 calls. Execution time mean : 131.416626 µs Execution time std-deviation : 2.218919 µs Execution time lower quantile : 130.073353 µs ( 2.5%) Execution time upper quantile : 135.212868 µs (97.5%) Overhead used : 8.079635 ns

edn->brcode Evaluation count : 3816 in 6 samples of 636 calls. Execution time mean : 157.407924 µs Execution time std-deviation : 3.556917 µs Execution time lower quantile : 154.338082 µs ( 2.5%) Execution time upper quantile : 162.800564 µs (97.5%) Overhead used : 8.102766 ns

(-> brcode brcode->edn edn->brcode) Evaluation count : 1920 in 6 samples of 320 calls. Execution time mean : 344.903181 µs Execution time std-deviation : 26.518055 µs Execution time lower quantile : 328.923528 µs ( 2.5%) Execution time upper quantile : 390.059255 µs (97.5%) Overhead used : 8.071450 ns

Node FFI

DOCS

BR Code as Json call function parse. Example: ```js const brcode = require('node-brecode');

const code = "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38";

console.log(brcode.parse(code)); // {"payloadversion":1,"initiationmethod":null, // "merchantinformation":[ // {"id":26,"info":[{"id":0,"info":"BR.GOV.BCB.PIX"},{"id":1,"info":"123e4567-e12b-12d1-a456-426655440000"}]}, // {"id":27,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123456789"}]} // ], // "merchantcategorycode":0,"merchantname":"NOME DO RECEBEDOR","merchantcity":"BRASILIA","postalcode":"70074900", // "currency":"986","amount":123.45,"countrycode":"BR","fieldtemplate":[{"reference_label":"RP12345678-2019"}], // "crc1610":"AD38","templates":[ // {"id":80,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123.ABCD.3456.WXYZ"}]} // ]} ```

Input: rust "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38"

Expected Json: json {"payload_version":1,"initiation_method":null, "merchant_information":[ {"id":26,"info":[{"id":0,"info":"BR.GOV.BCB.PIX"},{"id":1,"info":"123e4567-e12b-12d1-a456-426655440000"}]}, {"id":27,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123456789"}]} ], "merchant_category_code":0,"merchant_name":"NOME DO RECEBEDOR","merchant_city":"BRASILIA","postal_code":"70074900", "currency":"986","amount":123.45,"country_code":"BR","field_template":[{"reference_label":"RP12345678-2019"}], "crc1610":"AD38","templates":[ {"id":80,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123.ABCD.3456.WXYZ"}]} ]}

Json as BR Code call function emit. Example: ```js const brcode = require('node-brcode'); const json = {"payloadversion":1,"initiationmethod":null, "merchantinformation":[ {"id":26,"info":[{"id":0,"info":"BR.GOV.BCB.PIX"},{"id":1,"info":"123e4567-e12b-12d1-a456-426655440000"}]}, {"id":27,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123456789"}]} ], "merchantcategorycode":0,"merchantname":"NOME DO RECEBEDOR","merchantcity":"BRASILIA","postalcode":"70074900", "currency":"986","amount":123.45,"countrycode":"BR","fieldtemplate":[{"reference_label":"RP12345678-2019"}], "crc1610":"AD38","templates":[ {"id":80,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123.ABCD.3456.WXYZ"}]} ]};

console.log(brcode.emit(json)) // "00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38" ```

Other available functions: - crc16Ccitt

Node Benchmark with microbench

parse js { source: 'function() { return parse(code, 1000); }', raw: [ 0, 3202224 ], duration: '3 ms 202 μs 224 ns', name: 'parser' }

emit js { source: 'function() { return emit(json, 1000); }', raw: [ 0, 3386206 ], duration: '3 ms 386 μs 206 ns', name: 'emitter' }

parse(emit(json)) js { source: 'function() { return parse(emit(json), 1000); }', raw: [ 0, 4309501 ], duration: '4 ms 309 μs 501 ns', name: 'both-ways' }

Dart FFI

DOCS

Parse ```dart import 'package:dartbrcode/dartbrcode.dart';

final json = '{"payloadversion":1,"initiationmethod":null,"merchantaccountinformation":"12345678901234","merchantinformation":[{"id":26,"info":[{"id":0,"info":"BR.GOV.BCB.PIX"},{"id":1,"info":"123e4567-e12b-12d1-a456-426655440000"}]},{"id":27,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123456789"}]}],"merchantcategorycode":0,"merchantname":"NOME DO RECEBEDOR","merchantcity":"BRASILIA","postalcode":"70074900","currency":"986","amount":123.45,"countrycode":"BR","fieldtemplate":[{"reference_label":"RP12345678-2019"}],"crc1610":"AD38","templates":[{"id":80,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123.ABCD.3456.WXYZ"}]}]}';

void main() { jsonToBrcode(json); // '00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38' } ```

Emit ```dart import 'package:dartbrcode/dartbrcode.dart';

final brcode = '00020104141234567890123426580014BR.GOV.BCB.PIX0136123e4567-e12b-12d1-a456-42665544000027300012BR.COM.OUTRO011001234567895204000053039865406123.455802BR5917NOME DO RECEBEDOR6008BRASILIA61087007490062190515RP12345678-201980390012BR.COM.OUTRO01190123.ABCD.3456.WXYZ6304AD38';

void main() { jsonFromBrcode(brcode); // '{"payloadversion":1,"initiationmethod":null,"merchantaccountinformation":"12345678901234","merchantinformation":[{"id":26,"info":[{"id":0,"info":"BR.GOV.BCB.PIX"},{"id":1,"info":"123e4567-e12b-12d1-a456-426655440000"}]},{"id":27,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123456789"}]}],"merchantcategorycode":0,"merchantname":"NOME DO RECEBEDOR","merchantcity":"BRASILIA","postalcode":"70074900","currency":"986","amount":123.45,"countrycode":"BR","fieldtemplate":[{"reference_label":"RP12345678-2019"}],"crc1610":"AD38","templates":[{"id":80,"info":[{"id":0,"info":"BR.COM.OUTRO"},{"id":1,"info":"0123.ABCD.3456.WXYZ"}]}]}' } ```

Other available functions: - crc16Ccitt

Benchmarks

brcodeToJson For 100 runs: peak: 327 us, bottom: 069 us, avg: ~101 us

brcodeToJson For 10 runs: 378.68780764861793 us.

Goals