A LibNFC and MFOC NodeJS wrapper to read/write data from/to ISO/IEC 14443 A compliant tags. This repository can also be seen as a playground for a better understanding of Mifare technologies.
I am not an expert in RFID nor NFC technologies but really enjoyed deep diving inside. I would be glad you to correct me if something is wrong.
Let's take a tour.
Before being able to use this library, you need to install both libNFC
and mfoc
libraries (use brew on OSX) :
brew install libnfc mfoc
Moreover, you'll need a NFC reader to be able to play with Mifare chipsets. I'm personnaly using Identive SCM SCL3711 but it doesn't really matter. Only make sure that you're using one that is part of the offical LibNFC list. If you want to go deeper, I would advise you to take a look at the great work of Johnatan Westhues who developed Proxmark which is definitely the best audit tool for NFC/RFID technologies with also a highly active community.
Mifare is a communication protocol that partially relies on ISO 14443 standards. In case of Mifare Classic, only layers 1 and 2 are taking advantage of the standards. Transmission and communication protocols, respectively 4th and 3rd layers, rely on the NXP CRYPTO-1 protocol that has been broken in 2008 by Karsten Nohl and Henryk Plötz
Contactless devices can most of the time either be considered as RFID or NFC devices. The differences between these two types mainly rely on the frequency. RFID devices (which stands for Radio Frequency Identification) are Low Frequency devices (operating @ 125kHz up to 134,2 kHz). On the other hand, NFC (Near Field Contact) devices are considered as High Frequency devices (13,56MHz). Contactless cards can be found quite everywhere and it might be useful for security reasons to be aware of their opportunities but also of their weaknesses.
Better than making mistakes during the explanation of all the protocols, I better invite you to take a look at some great documents that have been written by far more experts about the subject :
- Original lecture by Karsten Nohl and Henryk Plötz @ CCC
- Ciphertext-only Cryptanalysis on Hardened Mifare Classic Cards
- Hacking Mifare Classic Cards
- APDU commands for communication
Each Mifare Classic card can be splitted in 16 sectors of data. Each sector has two authentication keys : key A and key B. Both of it are contained into the last memory block of each sector according to the following structure :
Key A and key B can have read or read-write access.
The main issue with CRYPTO-1 protocol is that from one given key, you can easily (thanks to the great work of Nethemba about Offline Attack of Mifare cards which has been implemented into the mfoc library) deduce all the others. Since a lot of chipsets resellers use default keys for production, a single portion of mutual keys can help you breaking all the others. I've been looking for building a consistent database for a while. Here is part of it. These keys are in the keys.txt
file at the root of the project. Other can be found on internet, but keep in mind that it might be illegal to publish it. Most of the keys will need you to brute force a tag. If you have the opportunity to get a few tags, you'll be able to exponentially increase your ability to unlock all the others. That said, there are two ways I am aware of to brute force a tag :
- mfcuk for Mifare Classic Universal Toolkit (never managed to make it work and might take a few time)
- proxmark3 as said before that can provide you much more tools than any other library. If you don't mind spending $300, go for it. Surprisingly powerful.
There is a last thing you need to be aware of : the specific sector 0. Out of 16 other ones which content can be overwritten, the sector 0 contains a few unique informations that can not be erased :
- UID
- BCC
- SAK
- ATAQ
- Manufacturer Data
The UID of the tag is basically what identifies it to the reader. When you're looking for cloning a tag, it's not only about cloning the content of all the other sectors but also being able to give both the original and the clone the same "name". Common tags that you can find on the market won't allow you to do so since their sector 0 is read only. But some chinese clones will. Moreover there are two types of chinese cards (probably more, a bit hard to know) :
- Basic chinese Mifare 1K that provide an APDU command for unlocking sector 0 and being able to write data. If ACK is received, then you can send the new data to the tag (and reset it's UID)
- 2nd generation chinese Mifare 1K, that removed this APDU command and allow you to immediately write data into sector 0
The nfc-mfclassic
command from the libnfc
library will not allow you to write the sector 0 with a 2nd generation card. However you can do it with a Proxmark or also fork the libnfc library to remove the specific case for sector 0.
For getting started, instanciate a new LockSmith
. The only parameter that can be provided to it's constructor is an object that might contain the following keys :
-
keys
, additional keys that should be added to the default ones for authenticating to the tags. -
workspace
, the workspace that will be used for storing dumps -
defaultKeysPath
, path to a file that contains the default keys that should be used for authentication. If none is provided, the rootkeys.txt
file will be used. I recommend not to set it. Prefer usingkeys
for adding values
import LockSmith from 'mifareJS';
const lockSmith = new LockSmith({
keys: ['a5b6c7d8', 'a1b2c3d4'],
workspace: '/mifare',
});
import LockSmith from 'mifareJS';
const lockSmith = new LockSmith();
await lockSmith.dump('tag.mfd');
import LockSmith from 'mifareJS';
const dump = await LockSmith.readHexFile('./tag.mfd', true); // The second parameter can be set to false if you don't want the hex content to be displayed in standard output
We'll here guide you on the way cloning a tag. Let's say you do have a chinese empty tag and your own, which you would like to duplicate. Let's call their dump chinese.mfd
and original.mfd
. The last parameter of the clone
methods lets you tell if we should unlock sector 0 (Chinese Mifare 1K) or not (Classic tag).
import LockSmith from 'mifareJS';
const lockSmith = new LockSmith();
// Place the chinese tag on the reader
await lockSmith.dump('chinese.mfd');
// Place the original tag on the reader
await lockSmith.dump('original.mfd');
// Then write the content of the original onto the chinese one
await LockSmith.clone('original.mfd', 'chinese.mfd', true);