2018-09-14 13:30:16 +02:00
|
|
|
package apdu
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/binary"
|
|
|
|
"errors"
|
2018-09-27 15:29:07 +02:00
|
|
|
"fmt"
|
2018-09-14 13:30:16 +02:00
|
|
|
)
|
|
|
|
|
2018-09-27 15:29:07 +02:00
|
|
|
const (
|
2018-10-05 13:03:58 +02:00
|
|
|
// SwOK is returned from smartcards as a positive response code.
|
2018-10-02 16:52:50 +02:00
|
|
|
SwOK = 0x9000
|
2018-09-27 15:29:07 +02:00
|
|
|
)
|
|
|
|
|
2018-10-05 13:03:58 +02:00
|
|
|
// ErrBadResponse defines an error conaining the returned Sw code and a description message.
|
2018-09-27 15:29:07 +02:00
|
|
|
type ErrBadResponse struct {
|
2019-03-06 14:31:40 +01:00
|
|
|
Sw uint16
|
2018-09-27 15:29:07 +02:00
|
|
|
message string
|
|
|
|
}
|
|
|
|
|
2018-10-05 13:03:58 +02:00
|
|
|
// NewErrBadResponse returns a ErrBadResponse with the specified sw and message values.
|
2018-09-27 15:29:07 +02:00
|
|
|
func NewErrBadResponse(sw uint16, message string) *ErrBadResponse {
|
|
|
|
return &ErrBadResponse{
|
2019-03-06 14:31:40 +01:00
|
|
|
Sw: sw,
|
2018-09-27 15:29:07 +02:00
|
|
|
message: message,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-05 13:03:58 +02:00
|
|
|
// Error implements the error interface.
|
2018-09-27 15:29:07 +02:00
|
|
|
func (e *ErrBadResponse) Error() string {
|
2019-03-06 14:31:40 +01:00
|
|
|
return fmt.Sprintf("bad response %x: %s", e.Sw, e.message)
|
2018-09-27 15:29:07 +02:00
|
|
|
}
|
2018-09-25 17:48:21 +02:00
|
|
|
|
2018-10-05 13:03:58 +02:00
|
|
|
// Response represents a struct containing the smartcard response fields.
|
2018-09-14 13:30:16 +02:00
|
|
|
type Response struct {
|
|
|
|
Data []byte
|
|
|
|
Sw1 uint8
|
|
|
|
Sw2 uint8
|
|
|
|
Sw uint16
|
|
|
|
}
|
|
|
|
|
2018-10-05 13:03:58 +02:00
|
|
|
// ErrBadRawResponse is an error returned by ParseResponse in case the response data is not long enough.
|
2018-09-14 13:30:16 +02:00
|
|
|
var ErrBadRawResponse = errors.New("response data must be at least 2 bytes")
|
|
|
|
|
2018-10-05 13:03:58 +02:00
|
|
|
// ParseResponse parses a raw response and return a Response.
|
2018-09-25 17:44:25 +02:00
|
|
|
func ParseResponse(data []byte) (*Response, error) {
|
|
|
|
r := &Response{}
|
|
|
|
return r, r.deserialize(data)
|
|
|
|
}
|
|
|
|
|
2018-09-14 13:30:16 +02:00
|
|
|
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
|
|
|
|
}
|
2018-09-25 17:48:21 +02:00
|
|
|
|
2018-10-05 13:03:58 +02:00
|
|
|
// IsOK returns true if the response Sw code is 0x9000.
|
2018-09-25 17:48:21 +02:00
|
|
|
func (r *Response) IsOK() bool {
|
2018-09-27 15:29:07 +02:00
|
|
|
return r.Sw == SwOK
|
2018-09-25 17:48:21 +02:00
|
|
|
}
|