74 lines
1.3 KiB
Go
74 lines
1.3 KiB
Go
package apdu
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/binary"
|
|
"errors"
|
|
"fmt"
|
|
)
|
|
|
|
const (
|
|
SwOK = 0x9000
|
|
SwSecurityConditionNotSatisfied = 0x6982
|
|
SwAuthenticationMethodBlocked = 0x6983
|
|
)
|
|
|
|
type ErrBadResponse struct {
|
|
sw uint16
|
|
message string
|
|
}
|
|
|
|
func NewErrBadResponse(sw uint16, message string) *ErrBadResponse {
|
|
return &ErrBadResponse{
|
|
sw: sw,
|
|
message: message,
|
|
}
|
|
}
|
|
|
|
func (e *ErrBadResponse) Error() string {
|
|
return fmt.Sprintf("bad response %x: %s", e.sw, e.message)
|
|
}
|
|
|
|
type Response struct {
|
|
Data []byte
|
|
Sw1 uint8
|
|
Sw2 uint8
|
|
Sw uint16
|
|
}
|
|
|
|
var ErrBadRawResponse = errors.New("response data must be at least 2 bytes")
|
|
|
|
func ParseResponse(data []byte) (*Response, error) {
|
|
r := &Response{}
|
|
return r, r.deserialize(data)
|
|
}
|
|
|
|
func (r *Response) deserialize(data []byte) error {
|
|
if len(data) < 2 {
|
|
return ErrBadRawResponse
|
|
}
|
|
|
|
r.Data = make([]byte, len(data)-2)
|
|
buf := bytes.NewReader(data)
|
|
|
|
if err := binary.Read(buf, binary.BigEndian, &r.Data); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := binary.Read(buf, binary.BigEndian, &r.Sw1); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := binary.Read(buf, binary.BigEndian, &r.Sw2); err != nil {
|
|
return err
|
|
}
|
|
|
|
r.Sw = (uint16(r.Sw1) << 8) | uint16(r.Sw2)
|
|
|
|
return nil
|
|
}
|
|
|
|
func (r *Response) IsOK() bool {
|
|
return r.Sw == SwOK
|
|
}
|