1 /*
2  * Copyright (C) 2010 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.nfc;
18 
19 import android.os.Parcel;
20 import android.os.Parcelable;
21 
22 import java.nio.ByteBuffer;
23 import java.util.Arrays;
24 
25 
26 /**
27  * Represents an immutable NDEF Message.
28  * <p>
29  * NDEF (NFC Data Exchange Format) is a light-weight binary format,
30  * used to encapsulate typed data. It is specified by the NFC Forum,
31  * for transmission and storage with NFC, however it is transport agnostic.
32  * <p>
33  * NDEF defines messages and records. An NDEF Record contains
34  * typed data, such as MIME-type media, a URI, or a custom
35  * application payload. An NDEF Message is a container for
36  * one or more NDEF Records.
37  * <p>
38  * When an Android device receives an NDEF Message
39  * (for example by reading an NFC tag) it processes it through
40  * a dispatch mechanism to determine an activity to launch.
41  * The type of the <em>first</em> record in the message has
42  * special importance for message dispatch, so design this record
43  * carefully.
44  * <p>
45  * Use {@link #NdefMessage(byte[])} to construct an NDEF Message from
46  * binary data, or {@link #NdefMessage(NdefRecord[])} to
47  * construct from one or more {@link NdefRecord}s.
48  * <p class="note">
49  * {@link NdefMessage} and {@link NdefRecord} implementations are
50  * always available, even on Android devices that do not have NFC hardware.
51  * <p class="note">
52  * {@link NdefRecord}s are intended to be immutable (and thread-safe),
53  * however they may contain mutable fields. So take care not to modify
54  * mutable fields passed into constructors, or modify mutable fields
55  * obtained by getter methods, unless such modification is explicitly
56  * marked as safe.
57  *
58  * @see NfcAdapter#ACTION_NDEF_DISCOVERED
59  * @see NdefRecord
60  */
61 public final class NdefMessage implements Parcelable {
62     private final NdefRecord[] mRecords;
63 
64     /**
65      * Construct an NDEF Message by parsing raw bytes.<p>
66      * Strict validation of the NDEF binary structure is performed:
67      * there must be at least one record, every record flag must
68      * be correct, and the total length of the message must match
69      * the length of the input data.<p>
70      * This parser can handle chunked records, and converts them
71      * into logical {@link NdefRecord}s within the message.<p>
72      * Once the input data has been parsed to one or more logical
73      * records, basic validation of the tnf, type, id, and payload fields
74      * of each record is performed, as per the documentation on
75      * on {@link NdefRecord#NdefRecord(short, byte[], byte[], byte[])}<p>
76      * If either strict validation of the binary format fails, or
77      * basic validation during record construction fails, a
78      * {@link FormatException} is thrown<p>
79      * Deep inspection of the type, id and payload fields of
80      * each record is not performed, so it is possible to parse input
81      * that has a valid binary format and confirms to the basic
82      * validation requirements of
83      * {@link NdefRecord#NdefRecord(short, byte[], byte[], byte[])},
84      * but fails more strict requirements as specified by the
85      * NFC Forum.
86      *
87      * <p class="note">
88      * It is safe to re-use the data byte array after construction:
89      * this constructor will make an internal copy of all necessary fields.
90      *
91      * @param data raw bytes to parse
92      * @throws FormatException if the data cannot be parsed
93      */
NdefMessage(byte[] data)94     public NdefMessage(byte[] data) throws FormatException {
95         if (data == null) throw new NullPointerException("data is null");
96         ByteBuffer buffer = ByteBuffer.wrap(data);
97 
98         mRecords = NdefRecord.parse(buffer, false);
99 
100         if (buffer.remaining() > 0) {
101             throw new FormatException("trailing data");
102         }
103     }
104 
105     /**
106      * Construct an NDEF Message from one or more NDEF Records.
107      *
108      * @param record first record (mandatory)
109      * @param records additional records (optional)
110      */
NdefMessage(NdefRecord record, NdefRecord ... records)111     public NdefMessage(NdefRecord record, NdefRecord ... records) {
112         // validate
113         if (record == null) throw new NullPointerException("record cannot be null");
114 
115         for (NdefRecord r : records) {
116             if (r == null) {
117                 throw new NullPointerException("record cannot be null");
118             }
119         }
120 
121         mRecords = new NdefRecord[1 + records.length];
122         mRecords[0] = record;
123         System.arraycopy(records, 0, mRecords, 1, records.length);
124     }
125 
126     /**
127      * Construct an NDEF Message from one or more NDEF Records.
128      *
129      * @param records one or more records
130      */
NdefMessage(NdefRecord[] records)131     public NdefMessage(NdefRecord[] records) {
132         // validate
133         if (records.length < 1) {
134             throw new IllegalArgumentException("must have at least one record");
135         }
136         for (NdefRecord r : records) {
137             if (r == null) {
138                 throw new NullPointerException("records cannot contain null");
139             }
140         }
141 
142         mRecords = records;
143     }
144 
145     /**
146      * Get the NDEF Records inside this NDEF Message.<p>
147      * An {@link NdefMessage} always has one or more NDEF Records: so the
148      * following code to retrieve the first record is always safe
149      * (no need to check for null or array length >= 1):
150      * <pre>
151      * NdefRecord firstRecord = ndefMessage.getRecords()[0];
152      * </pre>
153      *
154      * @return array of one or more NDEF records.
155      */
getRecords()156     public NdefRecord[] getRecords() {
157         return mRecords;
158     }
159 
160     /**
161      * Return the length of this NDEF Message if it is written to a byte array
162      * with {@link #toByteArray}.<p>
163      * An NDEF Message can be formatted to bytes in different ways
164      * depending on chunking, SR, and ID flags, so the length returned
165      * by this method may not be equal to the length of the original
166      * byte array used to construct this NDEF Message. However it will
167      * always be equal to the length of the byte array produced by
168      * {@link #toByteArray}.
169      *
170      * @return length of this NDEF Message when written to bytes with {@link #toByteArray}
171      * @see #toByteArray
172      */
getByteArrayLength()173     public int getByteArrayLength() {
174         int length = 0;
175         for (NdefRecord r : mRecords) {
176             length += r.getByteLength();
177         }
178         return length;
179     }
180 
181     /**
182      * Return this NDEF Message as raw bytes.<p>
183      * The NDEF Message is formatted as per the NDEF 1.0 specification,
184      * and the byte array is suitable for network transmission or storage
185      * in an NFC Forum NDEF compatible tag.<p>
186      * This method will not chunk any records, and will always use the
187      * short record (SR) format and omit the identifier field when possible.
188      *
189      * @return NDEF Message in binary format
190      * @see #getByteArrayLength()
191      */
toByteArray()192     public byte[] toByteArray() {
193         int length = getByteArrayLength();
194         ByteBuffer buffer = ByteBuffer.allocate(length);
195 
196         for (int i=0; i<mRecords.length; i++) {
197             boolean mb = (i == 0);  // first record
198             boolean me = (i == mRecords.length - 1);  // last record
199             mRecords[i].writeToByteBuffer(buffer, mb, me);
200         }
201 
202         return buffer.array();
203     }
204 
205     @Override
describeContents()206     public int describeContents() {
207         return 0;
208     }
209 
210     @Override
writeToParcel(Parcel dest, int flags)211     public void writeToParcel(Parcel dest, int flags) {
212         dest.writeInt(mRecords.length);
213         dest.writeTypedArray(mRecords, flags);
214     }
215 
216     public static final @android.annotation.NonNull Parcelable.Creator<NdefMessage> CREATOR =
217             new Parcelable.Creator<NdefMessage>() {
218         @Override
219         public NdefMessage createFromParcel(Parcel in) {
220             int recordsLength = in.readInt();
221             NdefRecord[] records = new NdefRecord[recordsLength];
222             in.readTypedArray(records, NdefRecord.CREATOR);
223             return new NdefMessage(records);
224         }
225         @Override
226         public NdefMessage[] newArray(int size) {
227             return new NdefMessage[size];
228         }
229     };
230 
231     @Override
hashCode()232     public int hashCode() {
233         return Arrays.hashCode(mRecords);
234     }
235 
236     /**
237      * Returns true if the specified NDEF Message contains
238      * identical NDEF Records.
239      */
240     @Override
equals(Object obj)241     public boolean equals(Object obj) {
242         if (this == obj) return true;
243         if (obj == null) return false;
244         if (getClass() != obj.getClass()) return false;
245         NdefMessage other = (NdefMessage) obj;
246         return Arrays.equals(mRecords, other.mRecords);
247     }
248 
249     @Override
toString()250     public String toString() {
251         return "NdefMessage " + Arrays.toString(mRecords);
252     }
253 }