Skip to content

Commit

Permalink
Merge pull request #3 from mercadolibre/feature/add-sizable-bitmap-ca…
Browse files Browse the repository at this point in the history
…pability-to-composite

Feature/add sizable bitmap capability to composite
  • Loading branch information
FacundoMora authored Feb 1, 2023
2 parents bb0b3da + dbd6ae1 commit b0e41ef
Show file tree
Hide file tree
Showing 4 changed files with 183 additions and 28 deletions.
52 changes: 37 additions & 15 deletions field/bitmap.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,31 +6,53 @@ import (
"github.com/moov-io/iso8583/utils"
)

const minBitmapLength = 8 // 64 bit, 8 bytes, or 16 hex digits
const maxBitmaps = 3
const (
defaultBitmapSize = 8
defaultMaxBitmaps = 3
)

var _ Field = (*Bitmap)(nil)

// NOTE: Bitmap does not support JSON encoding or decoding.
type Bitmap struct {
spec *Spec
bitmap *utils.Bitmap
data *Bitmap
spec *Spec
bitmap *utils.Bitmap
data *Bitmap
options *BitmapOptions
}

type BitmapOptions struct {
MaxBitmaps int
BitmapSize int
}

func NewBitmap(spec *Spec) *Bitmap {
return &Bitmap{
spec: spec,
bitmap: utils.NewBitmap(64 * maxBitmaps),
}
bm := &Bitmap{}
bm.SetSpec(spec)
return bm
}

func (f *Bitmap) Spec() *Spec {
return f.spec
}

func (f *Bitmap) SetSpec(spec *Spec) {
bmo := (*BitmapOptions)(nil)
if spec != nil {
bmo = spec.BitmapOptions
}
if bmo == nil {
bmo = &BitmapOptions{}
}
if bmo.MaxBitmaps == 0 {
bmo.MaxBitmaps = defaultMaxBitmaps
}
if bmo.BitmapSize == 0 {
bmo.BitmapSize = defaultBitmapSize
}
f.options = bmo
f.spec = spec
f.Reset()
}

func (f *Bitmap) SetBytes(b []byte) error {
Expand Down Expand Up @@ -66,7 +88,7 @@ func (f *Bitmap) Pack() ([]byte, error) {
return nil, fmt.Errorf("failed to retrieve bytes: %w", err)
}

data = data[0 : 8*count]
data = data[0 : f.options.BitmapSize*count]

packed, err := f.spec.Enc.Encode(data)
if err != nil {
Expand All @@ -81,7 +103,7 @@ func (f *Bitmap) Pack() ([]byte, error) {
// if secondary bitmap presents (bit 1 is set) we return 16 bytes (or 32 for hex encoding)
// and so on for maxBitmaps
func (f *Bitmap) Unpack(data []byte) (int, error) {
minLen, _, err := f.spec.Pref.DecodeLength(minBitmapLength, data)
minLen, _, err := f.spec.Pref.DecodeLength(f.options.BitmapSize, data)
if err != nil {
return 0, fmt.Errorf("failed to decode length: %w", err)
}
Expand All @@ -90,7 +112,7 @@ func (f *Bitmap) Unpack(data []byte) (int, error) {
read := 0

// read max
for i := 0; i < maxBitmaps; i++ {
for i := 0; i < f.options.MaxBitmaps; i++ {
decoded, readDecoded, err := f.spec.Enc.Decode(data[read:], minLen)
if err != nil {
return 0, fmt.Errorf("failed to decode content for %d bitmap: %w", i+1, err)
Expand Down Expand Up @@ -150,7 +172,7 @@ func (f *Bitmap) Marshal(data interface{}) error {
}

func (f *Bitmap) Reset() {
f.bitmap = utils.NewBitmap(64 * maxBitmaps)
f.bitmap = utils.NewBitmap(f.options.BitmapSize * 8 * f.options.MaxBitmaps)
}

func (f *Bitmap) Set(i int) {
Expand All @@ -167,7 +189,7 @@ func (f *Bitmap) Len() int {

func (f *Bitmap) bitmapsCount() int {
count := 1
for i := 0; i < maxBitmaps; i++ {
for i := 0; i < f.options.MaxBitmaps-1; i++ {
if f.IsSet(i*64 + 1) {
count += 1
}
Expand All @@ -184,7 +206,7 @@ func (f *Bitmap) setBitmapFields() bool {
// bitmap bit 65

// start from the 2nd bitmap as for the 1st bitmap we don't need to set any bits
for bitmapIndex := 2; bitmapIndex <= maxBitmaps; bitmapIndex++ {
for bitmapIndex := 2; bitmapIndex <= f.options.MaxBitmaps; bitmapIndex++ {

// are there fields for this (bitmapIndex) bitmap?
bitmapStart := (bitmapIndex-1)*64 + 2 // we skip firt bit as it's for the next bitmap
Expand Down
148 changes: 139 additions & 9 deletions field/composite.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,15 @@ import (
"fmt"
"reflect"
"regexp"
"strconv"

"github.com/moov-io/iso8583/padding"
"github.com/moov-io/iso8583/sort"
"github.com/moov-io/iso8583/utils"
)

const compositeBitmapKey = "0"

var _ Field = (*Composite)(nil)
var _ json.Marshaler = (*Composite)(nil)
var _ json.Unmarshaler = (*Composite)(nil)
Expand Down Expand Up @@ -58,6 +61,8 @@ var _ json.Unmarshaler = (*Composite)(nil)
type Composite struct {
spec *Spec

bitmap *Bitmap

orderedSpecFieldTags []string

// stores all fields according to the spec
Expand Down Expand Up @@ -115,7 +120,10 @@ func (f *Composite) SetSpec(spec *Spec) {
panic(err)
}
f.spec = spec
f.orderedSpecFieldTags = orderedKeys(spec.Subfields, spec.Tag.Sort)

if spec.Tag != nil && spec.Tag.Sort != nil {
f.orderedSpecFieldTags = orderedKeys(spec.Subfields, spec.Tag.Sort)
}
}

func (f *Composite) Unmarshal(v interface{}) error {
Expand Down Expand Up @@ -290,6 +298,18 @@ func (f *Composite) Bytes() ([]byte, error) {
return f.pack()
}

// Bitmap returns an instantiation of Bitmap using the BitmapSpec from the spec attribute.
func (f *Composite) Bitmap() *Bitmap {
if f.bitmap == nil {
if field, ok := f.subfields[compositeBitmapKey]; ok {
f.bitmap = field.(*Bitmap)
}

}

return f.bitmap
}

// String iterates over the receiver's subfields, packs them and converts the
// result to a string. The result does not incorporate the encoded aggregate
// length of the subfields in the prefix.
Expand Down Expand Up @@ -342,6 +362,53 @@ func (f *Composite) UnmarshalJSON(b []byte) error {
}

func (f *Composite) pack() ([]byte, error) {
if f.Bitmap() != nil {
return f.packSubfieldsByBitmap()
}

return f.packSubfieldsByTag()
}

func (f *Composite) packSubfieldsByBitmap() ([]byte, error) {
packed := []byte{}
f.Bitmap().Reset()

ids, err := f.packableFieldIDs()
if err != nil {
return nil, fmt.Errorf("failed to pack message: %w", err)
}

for _, id := range ids {
// index 0 is for bitmap
// regular field number startd from index 1
idInt, err := strconv.Atoi(id)
if err != nil {
return nil, fmt.Errorf("failed to pack message: %w", err)
}

if idInt < 1 {
continue
}
f.Bitmap().Set(idInt)
}

// pack fields
for _, i := range ids {
field, ok := f.subfields[i]
if !ok {
return nil, fmt.Errorf("failed to pack field %v: no specification found", i)
}
packedField, err := field.Pack()
if err != nil {
return nil, fmt.Errorf("failed to pack field %v (%s): %w", i, field.Spec().Description, err)
}
packed = append(packed, packedField...)
}

return packed, nil
}

func (f *Composite) packSubfieldsByTag() ([]byte, error) {
packed := []byte{}
for _, tag := range f.orderedSpecFieldTags {
field, ok := f.subfields[tag]
Expand Down Expand Up @@ -377,6 +444,9 @@ func (f *Composite) pack() ([]byte, error) {
}

func (f *Composite) unpack(data []byte, isVariableLength bool) (int, error) {
if f.Bitmap() != nil {
return f.unpackSubfieldsByBitmap(data)
}
if f.spec.Tag.Enc != nil {
return f.unpackSubfieldsByTag(data)
}
Expand Down Expand Up @@ -442,21 +512,63 @@ func (f *Composite) unpackSubfieldsByTag(data []byte) (int, error) {
return offset, nil
}

func (f *Composite) unpackSubfieldsByBitmap(data []byte) (int, error) {
var off int

// reset fields that were set
f.setSubfields = map[string]struct{}{}

// unpack Bitmap
read, err := f.Bitmap().Unpack(data[off:])
if err != nil {
return 0, fmt.Errorf("failed to unpack bitmap: %w", err)
}

off += read

for i := 1; i <= f.Bitmap().Len(); i++ {
if f.Bitmap().IsSet(i) {
iStr := fmt.Sprintf("%v", i)

fl, ok := f.subfields[iStr]
if !ok {
return 0, fmt.Errorf("failed to unpack field %v: no specification found", iStr)
}

read, err = fl.Unpack(data[off:])
if err != nil {
return 0, fmt.Errorf("failed to unpack field %v (%s): %w", iStr, fl.Spec().Description, err)
}

f.setSubfields[iStr] = struct{}{}

off += read
}
}

return off, nil
}

func validateCompositeSpec(spec *Spec) error {
if spec.Tag == nil || spec.Tag.Sort == nil {
return fmt.Errorf("Composite spec requires a Tag.Sort function to be defined")
var bm *Bitmap
if value, ok := spec.Subfields[compositeBitmapKey]; ok {
bm = value.(*Bitmap)
}
if spec.Pad != nil && spec.Pad != padding.None {
return fmt.Errorf("Composite spec only supports nil or None spec padding values")

if (spec.Tag == nil || spec.Tag.Sort == nil) && bm == nil {
return fmt.Errorf("Composite spec requires a Tag.Sort function or a BitmapSpec to be defined")
}
if spec.Enc != nil {
return fmt.Errorf("Composite spec only supports a nil Enc value")
if spec.Tag != nil && spec.Tag.Sort != nil && bm != nil {
return fmt.Errorf("Composite spec doesn't support Tag.Sort function when a BitmapSpec is defined")
}
if spec.Tag != nil && spec.Tag.Enc == nil && spec.Tag.Length > 0 {
return fmt.Errorf("Composite spec requires a Tag.Enc to be defined if Tag.Length > 0")
}
if spec.Tag.Sort == nil {
return fmt.Errorf("Composite spec requires a Tag.Sort function to be defined")
if spec.Pad != nil && spec.Pad != padding.None {
return fmt.Errorf("Composite spec only supports nil or None spec padding values")
}
if spec.Enc != nil {
return fmt.Errorf("Composite spec only supports a nil Enc value")
}
return nil
}
Expand Down Expand Up @@ -488,3 +600,21 @@ func getFieldIndexOrTag(field reflect.StructField) (string, error) {

return "", nil
}

func (f *Composite) packableFieldIDs() ([]string, error) {
// Index 0 represent bitmap which is always populated.
populatedFieldIDs := []string{compositeBitmapKey}

for id := range f.setSubfields {
// represents the bitmap
if id == compositeBitmapKey {
continue
}

populatedFieldIDs = append(populatedFieldIDs, id)
}

sort.StringsByInt(populatedFieldIDs)

return populatedFieldIDs, nil
}
8 changes: 4 additions & 4 deletions field/composite_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -979,8 +979,8 @@ func TestCompositePanicsOnSpecValidationFailures(t *testing.T) {
spec *Spec
}{
{
desc: "panics on nil Tag being defined in spec",
err: "Composite spec requires a Tag.Sort function to be defined",
desc: "panics on nil Tag and no Bitmap being defined in spec",
err: "Composite spec requires a Tag.Sort function or a BitmapSpec to be defined",
spec: &Spec{
Length: 6,
Pref: prefix.ASCII.Fixed,
Expand All @@ -989,8 +989,8 @@ func TestCompositePanicsOnSpecValidationFailures(t *testing.T) {
},
},
{
desc: "panics on nil Tag.Sort being defined in spec",
err: "Composite spec requires a Tag.Sort function to be defined",
desc: "panics on nil Tag and no Bitmap being defined in spec",
err: "Composite spec requires a Tag.Sort function or a BitmapSpec to be defined",
spec: &Spec{
Length: 6,
Pref: prefix.ASCII.Fixed,
Expand Down
3 changes: 3 additions & 0 deletions field/spec.go
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,9 @@ type Spec struct {
// Subfields defines the subfields held within the field. Only
// applicable to composite field types.
Subfields map[string]Field
// BitmapOptions could be used on a bitmap field, and is optional.
// Contains options that modifies the behaviour of the bitmap.
BitmapOptions *BitmapOptions
}

func NewSpec(length int, desc string, enc encoding.Encoder, pref prefix.Prefixer) *Spec {
Expand Down

0 comments on commit b0e41ef

Please sign in to comment.