blob: 9f3116886fd2ffc999359eefa0caef17697f9eea [file] [log] [blame]
Harald Weltec781ab82021-05-23 11:50:19 +02001#!/usr/bin/env python3
2
3# RESTful HTTP service for performing authentication against USIM cards
4#
Harald Welte04897d52022-07-23 14:07:00 +02005# (C) 2021-2022 by Harald Welte <laforge@osmocom.org>
Harald Weltec781ab82021-05-23 11:50:19 +02006#
7# This program is free software: you can redistribute it and/or modify
8# it under the terms of the GNU General Public License as published by
9# the Free Software Foundation, either version 2 of the License, or
10# (at your option) any later version.
11#
12# This program is distributed in the hope that it will be useful,
13# but WITHOUT ANY WARRANTY; without even the implied warranty of
14# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15# GNU General Public License for more details.
16#
17# You should have received a copy of the GNU General Public License
18# along with this program. If not, see <http://www.gnu.org/licenses/>.
19
20import json
21import sys
22import argparse
23
Harald Welte3f3b45a2022-07-23 13:44:20 +020024from klein import Klein
Harald Weltec781ab82021-05-23 11:50:19 +020025
26from pySim.transport import ApduTracer
27from pySim.transport.pcsc import PcscSimLink
28from pySim.commands import SimCardCommands
Philipp Maier91b379a2023-08-16 11:43:19 +020029from pySim.cards import UiccCardBase
Harald Weltec781ab82021-05-23 11:50:19 +020030from pySim.exceptions import *
31
32class ApduPrintTracer(ApduTracer):
33 def trace_response(self, cmd, sw, resp):
34 #print("CMD: %s -> RSP: %s %s" % (cmd, sw, resp))
35 pass
36
Harald Welte33f8da82021-11-03 12:14:50 +010037def connect_to_card(slot_nr:int):
38 tp = PcscSimLink(slot_nr, apdu_tracer=ApduPrintTracer())
39 tp.connect()
40
41 scc = SimCardCommands(tp)
Philipp Maier91b379a2023-08-16 11:43:19 +020042 card = UiccCardBase(scc)
Harald Welte33f8da82021-11-03 12:14:50 +010043
44 # this should be part of UsimCard, but FairewavesSIM breaks with that :/
45 scc.cla_byte = "00"
46 scc.sel_ctrl = "0004"
47
48 card.read_aids()
Philipp Maier71a3fb82023-08-16 11:31:31 +020049
50 # ensure that MF is selected when we are done.
51 card._scc.select_file('3f00')
Harald Welte33f8da82021-11-03 12:14:50 +010052
53 return tp, scc, card
54
Harald Welte3f3b45a2022-07-23 13:44:20 +020055class ApiError:
Harald Welte04897d52022-07-23 14:07:00 +020056 def __init__(self, msg:str, sw=None):
Harald Welte3f3b45a2022-07-23 13:44:20 +020057 self.msg = msg
Harald Welte04897d52022-07-23 14:07:00 +020058 self.sw = sw
Harald Welte3f3b45a2022-07-23 13:44:20 +020059
60 def __str__(self):
Harald Welte04897d52022-07-23 14:07:00 +020061 d = {'error': {'message':self.msg}}
62 if self.sw:
63 d['error']['status_word'] = self.sw
64 return json.dumps(d)
Harald Welte3f3b45a2022-07-23 13:44:20 +020065
66
Harald Welte6f8cf9b2022-07-06 16:28:56 +020067def set_headers(request):
68 request.setHeader('Content-Type', 'application/json')
Harald Welte33f8da82021-11-03 12:14:50 +010069
Harald Welte3f3b45a2022-07-23 13:44:20 +020070class SimRestServer:
71 app = Klein()
Harald Weltec781ab82021-05-23 11:50:19 +020072
Harald Welte3f3b45a2022-07-23 13:44:20 +020073 @app.handle_errors(NoCardError)
74 def no_card_error(self, request, failure):
75 set_headers(request)
Harald Weltec781ab82021-05-23 11:50:19 +020076 request.setResponseCode(410)
Harald Welte3f3b45a2022-07-23 13:44:20 +020077 return str(ApiError("No SIM card inserted in slot"))
Harald Weltec781ab82021-05-23 11:50:19 +020078
Harald Welte3f3b45a2022-07-23 13:44:20 +020079 @app.handle_errors(ReaderError)
80 def reader_error(self, request, failure):
81 set_headers(request)
82 request.setResponseCode(404)
83 return str(ApiError("Reader Error: Specified SIM Slot doesn't exist"))
84
85 @app.handle_errors(ProtocolError)
86 def protocol_error(self, request, failure):
87 set_headers(request)
88 request.setResponseCode(500)
Harald Welte04897d52022-07-23 14:07:00 +020089 return str(ApiError("Protocol Error: %s" % failure.value))
Harald Welte3f3b45a2022-07-23 13:44:20 +020090
91 @app.handle_errors(SwMatchError)
92 def sw_match_error(self, request, failure):
93 set_headers(request)
94 request.setResponseCode(500)
Harald Welte04897d52022-07-23 14:07:00 +020095 sw = failure.value.sw_actual
96 if sw == '9862':
97 return str(ApiError("Card Authentication Error - Incorrect MAC", sw))
98 elif sw == '6982':
99 return str(ApiError("Security Status not satisfied - Card PIN enabled?", sw))
100 else:
101 return str(ApiError("Card Communication Error %s" % failure.value), sw)
Harald Welte3f3b45a2022-07-23 13:44:20 +0200102
103
104 @app.route('/sim-auth-api/v1/slot/<int:slot>')
105 def auth(self, request, slot):
106 """REST API endpoint for performing authentication against a USIM.
107 Expects a JSON body containing RAND and AUTN.
108 Returns a JSON body containing RES, CK, IK and Kc."""
109 try:
110 # there are two hex-string JSON parameters in the body: rand and autn
111 content = json.loads(request.content.read())
112 rand = content['rand']
113 autn = content['autn']
114 except:
115 set_headers(request)
116 request.setResponseCode(400)
117 return str(ApiError("Malformed Request"))
118
119 tp, scc, card = connect_to_card(slot)
120
Harald Weltec781ab82021-05-23 11:50:19 +0200121 card.select_adf_by_aid(adf='usim')
122 res, sw = scc.authenticate(rand, autn)
Harald Weltec781ab82021-05-23 11:50:19 +0200123
Harald Welte3f3b45a2022-07-23 13:44:20 +0200124 tp.disconnect()
Harald Weltec781ab82021-05-23 11:50:19 +0200125
Harald Welte3f3b45a2022-07-23 13:44:20 +0200126 set_headers(request)
127 return json.dumps(res, indent=4)
Harald Weltec781ab82021-05-23 11:50:19 +0200128
Harald Welte3f3b45a2022-07-23 13:44:20 +0200129 @app.route('/sim-info-api/v1/slot/<int:slot>')
130 def info(self, request, slot):
131 """REST API endpoint for obtaining information about an USIM.
132 Expects empty body in request.
133 Returns a JSON body containing ICCID, IMSI."""
Harald Welte33f8da82021-11-03 12:14:50 +0100134
Harald Welte33f8da82021-11-03 12:14:50 +0100135 tp, scc, card = connect_to_card(slot)
Harald Welte33f8da82021-11-03 12:14:50 +0100136
Harald Welte33f8da82021-11-03 12:14:50 +0100137 card.select_adf_by_aid(adf='usim')
138 iccid, sw = card.read_iccid()
139 imsi, sw = card.read_imsi()
140 res = {"imsi": imsi, "iccid": iccid }
Harald Welte33f8da82021-11-03 12:14:50 +0100141
Harald Welte3f3b45a2022-07-23 13:44:20 +0200142 tp.disconnect()
Harald Welte33f8da82021-11-03 12:14:50 +0100143
Harald Welte3f3b45a2022-07-23 13:44:20 +0200144 set_headers(request)
145 return json.dumps(res, indent=4)
Harald Welte33f8da82021-11-03 12:14:50 +0100146
147
Harald Weltec781ab82021-05-23 11:50:19 +0200148def main(argv):
149 parser = argparse.ArgumentParser()
150 parser.add_argument("-H", "--host", help="Host/IP to bind HTTP to", default="localhost")
151 parser.add_argument("-p", "--port", help="TCP port to bind HTTP to", default=8000)
152 #parser.add_argument("-v", "--verbose", help="increase output verbosity", action='count', default=0)
153
154 args = parser.parse_args()
155
Harald Welte3f3b45a2022-07-23 13:44:20 +0200156 srr = SimRestServer()
157 srr.app.run(args.host, args.port)
Harald Weltec781ab82021-05-23 11:50:19 +0200158
159if __name__ == "__main__":
160 main(sys.argv)