Revision: 4
Copyright 2019, Readium Foundation. All Rights Reserved.
Latest version:
This version:
This section is informative
In order to deliver a Publication to a User with specific rights and restrictions, a Content Provider may want to encrypt the Publication’s Resources and associate them with a specific license.
This specification, Licensed Content Protection (LCP), defines a standard License Document and an Encryption Profile for encrypting resources in different publication formats (especially EPUB), and for securely delivering decryption keys to Reading Systems via licenses tailored to specific Users. It also defines a simple passphrase-based authentication method for Reading Systems to access encrypted resources and verify a license.
This specification adopts terms defined in the EPUB 3 family of specifications. Important terms used include:
- Publication
- A logical document entity consisting of a set of interrelated resources, packaged in an EPUB Container.
- Publication Resource (or Resource)
- A resource which contains content or instructions that contribute to the logic and rendering of the EPUB Publication.
- Package Document
- A Publication Resource carrying meta information about the EPUB Publication, provides a manifest of resources and defines the default reading order.
- EPUB Container (or Container)
- The ZIP-based packaging and distribution format for EPUB Publications defined in [OCF ZIP Container].
- EPUB Reading System (or Reading System)
- A system that processes EPUB Publications for presentation to a user in a manner conformant with the EPUB 3 specification.
- Codec
- Codec refers to content types that have intrinsic binary format qualities, such as video and audio media types which are already designed for optimum compression, or which provide optimized streaming capabilities.
- Non-Codec
- Non-Codec refers to content types that benefit from compression due to the nature of their internal data structure, such as file formats based on character strings (for example, HTML, CSS, etc.).
The following terms are defined by this specification:
- Protected Publication
- A Publication that has been protected according to this specification.
- License Authority
- The entity which operates the Readium LCP ecosystem.
- License Document
- Document that contains references to the various keys, links to related external resources, rights and restrictions that are applied to the Protected Publication, and user information.
- Content Key
- Symmetric key used to encrypt the resources of the Protected Publication. In the License Document, this Content Key will be encrypted using the User Key.
- User
- An individual that consumes a Publication using a Reading System.
- User Passphrase
- A string of text entered by the user that is used to generate the User Key.
- User Key
- A hash of the User Passphrase that is used to decrypt the Content Key and selected user information fields.
- Encryption Profile
- A set of encryption algorithms used in a specific Protected Publication and associated Licence Document.
- Content Provider (or Provider)
- An entity that delivers LCP licenses for Protected Publications to Users.
- Provider Certificate
- A certificate that is included in the License Document to identify the Content Provider and validate the signature of the License Document.
- Root Certificate
- A certificate that is embedded in the Reading System in order to confirm that the Provider Certificate is valid.
This section is informative
In order to encrypt Resources and expose its Encryption Profile, LCP relies on two different files:
-
License Document (
META-INF/license.lcpl
) -
META-INF/encryption.xml
Both of these files are contained inside the EPUB Container, although the License Document can be transmitted initially outside the Container. Following the [OCF] specification, META-INF/encryption.xml
identifies the Publication Resources that are protected and points to the Content Key needed to decrypt them. This Content Key is located inside the License Document and is itself encrypted using the third element of LCP, the User Key. The User Key is generated by taking a hash of a User Passphrase. The User Key is used to decrypt the Content Key, which in turn is used to decrypt the Publication Resources.
The License Document may also contain links to external resources, information identifying the User, and information about what rights are conveyed to the User and which are not. Rights information may include things like the time for which the license is valid, whether the book may be printed or copied, etc. Finally, the License Document always includes a digital signature to prevent modification of any of its components.
Figure 1: Relationships among the various components of LCP
To protect a Publication, the Provider follows these steps:
-
Generate a unique Content Key for the Publication.
-
Store this Content Key for future use in licensing the Publication.
-
Encrypt each protected Resource using that key, after compression if applicable.
-
Add these protected Resources to the Container, replacing the unprotected versions.
-
Create a
META-INF/encryption.xml
document (as described in section 2.2) that includes an EncryptedData element for each protected Resource that contains:a. An EncryptionMethod element that lists the algorithm used
b. A KeyInfo element with a RetrievalMethod child that points to the Content Key in the License Document
c. A CipherData element that identifies the protected Resource
-
Add
META-INF/encryption.xml
to the Container.
The Publication is now protected (i.e., has become a Protected Publication) and is ready for licensing to one or more Users.
After a User requests a Protected Publication, the following steps are followed by the Provider to license the Protected Publication:
-
Generate the User Key by hashing the User Passphrase (as described in section 4.2). It is assumed that the User and their User Passphrase are already known to the Provider.
-
Encrypt the Content Key for the Protected Publication using the User Key.
-
Create a License Document (
META-INF/license.lcpl
) with the following contents (as described in section 3):a. A unique ID for this license
b. The date the license was issued
c. The URI that identifies the Content Provider
d. The encrypted Content Key
e. Information about the User Key
f. Links to additional information stored outside of the Protected Publication and License Document (optional)
g. Information on specific rights being granted to the User (optional)
h. Information identifying the User (optional). Some of the fields in this section may be encrypted using the User Key.
-
Generate a digital signature for the License Document data and add to the License Document.
There are then two different methods to deliver the License Document and Protected Publication to the User:
-
License Document included inside Protected Publication: The Provider adds the License Document to the Protected Publication’s Container and delivers this to the User.
-
License Document delivered separately: The Provider includes a link from the License Document to the Protected Publication, and then delivers just the License Document to the User. The Reading System processing the License Document will retrieve the Protected Publication and add the License Document to the Container of this Protected Publication.
Whichever method is used, the Reading System will be presented with an EPUB Container that includes the Protected Publication and the License Document.
In order to decrypt and render a Protected Publication, the User’s Reading System will follow these steps:
-
Verify the signature for the License Document.
-
Get the User Key (if already stored) or generate it by hashing the User Passphrase.
-
Decrypt the Content Key using the User Key.
-
Decrypt the protected Resources using the Content Key.
The keywords must, must not, required, shall, shall not, should, should not, recommended, may, and optional in this document are to be interpreted as described in [RFC2119].
All sections of this specification are normative except where identified by the informative status label "This section is informative". The application of informative status to sections and appendices applies to all child content and subsections they may contain.
All examples in this specification are informative.
This section is informative
LCP follows the [OCF] specification to identify encrypted Resources within the EPUB Container. Specifically, LCP uses META-INF/encryption.xml
to indicate which Resources listed in the Package Document are encrypted and to reference the Content Key.
In order to make sure that Protected Publications remain identifiable by Reading Systems, a certain number of Resources are prohibited from being encrypted.
LCP inherits the following list of files prohibited from being encrypted from the [OCF] specification:
-
mimetype
-
META-INF/container.xml
-
META-INF/encryption.xml
-
META-INF/manifest.xml
-
META-INF/metadata.xml
-
META-INF/rights.xml
-
META-INF/signatures.xml
-
EPUB
rootfiles
(the Package Document for any rendition)
In addition, this specification defines that the following files must not be encrypted:
-
META-INF/license.lcpl
-
Navigation Documents referenced in any Package Document from the Publication (all Publication Resources listed in the Publication manifest with the "nav" property)
-
NCX documents referenced in any Package Document from the Publication (all Publication Resources listed in the Publication manifest with the media type "application/x-dtbncx+xml")
-
Cover images (all Publication Resources listed in the Publication manifest with the "cover-image" property)
As defined in the [OCF] specification, all encrypted Publication Resources must be identified in the well-known file META-INF/encryption.xml
using [XML-ENC].
Publication Resources with Non-Codec content types should be compressed and the Deflate compression algorithm must be used. This practice ensures that file entries stored in the Container have a smaller size.
Resources with Codec content types should be stored without compression. In such case, compression would introduce unnecessary processing overhead at production time (especially with large resource files) and would impact audio/video playback performance at consumption time.
In Publications protected using LCP, there are additional requirements for determining the length of the full resource ahead of media playback, especially when using partial content requests:
Streams of data that are compressed before they are encrypted must provide additional EncryptionProperties
metadata to specify the size of the initial resource (i.e., before compression and encryption), as per the Compression XML element defined in OCF, Compression and Encryption.
Streams of data that are not compressed before they are encrypted should provide such additional EncryptionProperties
metadata.
In Publications protected using LCP, the following XML syntax identifies the key used to encrypt Resources (the LCP Content Key):
-
The
ds:KeyInfo
element must point to the Content Key using theds:RetrievalMethod
element. -
The
URI
attribute ofds:RetrievalMethod
must use a value of "license.lcpl#/encryption/content_key
" to point to the encrypted Content Key stored in the License Document. This URI follows the [JSON Pointer] specification. -
The
Type
attribute must use a value of "http://readium.org/2014/01/lcp#EncryptedContentKey
" to identify the target of the URI as an encrypted Content Key.
Example1: An image.jpeg
resource is declared as encrypted using AES within encryption.xml
.
<encryption
xmlns ="urn:oasis:names:tc:opendocument:xmlns:container"
xmlns:enc="http://www.w3.org/2001/04/xmlenc#"
xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
<enc:EncryptedData Id="ED1">
<enc:EncryptionMethod Algorithm="http://www.w3.org/2001/04/xmlenc#aes256-cbc"/>
<ds:KeyInfo>
<ds:RetrievalMethod URI="license.lcpl#/encryption/content_key"
Type="http://readium.org/2014/01/lcp#EncryptedContentKey"/>
</ds:KeyInfo>
<enc:CipherData>
<enc:CipherReference URI="image.jpeg"/>
</enc:CipherData>
<enc:EncryptionProperties>
<enc:EncryptionProperty xmlns:ns="http://www.idpf.org/2016/encryption#compression">
<ns:Compression Method="0" OriginalLength="450100"/>
</enc:EncryptionProperty>
</enc:EncryptionProperties>
</enc:EncryptedData>
</encryption>
This section is informative
While META-INF/encryption.xml
describes how the Resources are encrypted and where the encrypted Content Key is located, all of the other relevant information for LCP is stored in the License Document.
This specification defines the License Document’s syntax, location in the Container, media type, file extension and processing model.
A License Document must meet all of the following criteria:
Document properties
-
It must meet the conformance constraints for JSON documents as defined in [JSON].
-
It must be encoded using UTF-8.
File properties
-
Its filename must use the file extension
.lcpl
. -
Its MIME media type is
application/vnd.readium.lcp.license.v1.0+json
-
Its location in the Container must be
META-INF/license.lcpl
The License Document must contain the following name/value pairs:
Name | Value | Format |
---|---|---|
id |
Unique identifier for the License | String |
issued |
Date and time when the license was first issued | ISO 8601 |
provider |
Unique identifier for the Provider | URI |
In addition, the License Document may contain the following name/value pair:
Name | Value | Format |
---|---|---|
updated |
Date and time when the license was last updated | ISO 8601 |
In addition to Core License information, the License Document must contain an encryption
object with the following name/value pair:
Name | Value | Format |
---|---|---|
profile |
Identifies the Encryption Profile used by this LCP-protected Publication | URI |
The encryption
object must also contain the following two objects: content_key
and user_key
.
The encryption/content_key
object contains the Content Key (encrypted using the User Key) used to encrypt the Publication Resources. It must contain the following name/value pairs:
Name | Value | Format |
---|---|---|
encrypted_value |
Encrypted Content Key | Base 64 encoded octet sequence |
algorithm |
Algorithm used to encrypt the Content Key, identified using the URIs defined in [XML-ENC]. This must match the Content Key encryption algorithm named in the Encryption Profile identified in encryption/profile . |
URI |
The encryption/user_key
object contains information regarding the User Key used to encrypt the Content Key. It must contain the following name/value pairs:
Name | Value | Format |
---|---|---|
text_hint |
A hint to be displayed to the User to help them remember the User Passphrase | String |
algorithm |
Algorithm used to generate the User Key from the User Passphrase, identified using the URIs defined in [XML-ENC]. This must match the User Key hash algorithm named in the Encryption Profile identified in encryption/profile . |
URI |
key_check |
The value of the License Document’s id field, encrypted using the User Key and the same algorithm identified for Content Key encryption in encryption/content_key/algorithm . This is used to verify that the Reading System has the correct User Key. |
Base 64 encoded octet sequence |
Example 2: Encryption information for a License Document that uses the Basic Encryption Profile for LCP 1.0.
{
"id": "ef15e740-697f-11e3-949a-0800200c9a66",
"issued": "2013-11-04T01:08:15+01:00",
"updated": "2014-02-21T09:44:17+01:00",
"provider": "https://www.imaginaryebookretailer.com",
"encryption": {
"profile": "http://readium.org/lcp/basic-profile",
"content_key": {
"encrypted_value": "/k8RpXqf4E2WEunCp76E8PjhS051NXwAXeTD1ioazYxCRGvHLAck/KQ3cCh5JxDmCK0nRLyAxs1X0aA3z55boQ==",
"algorithm": "http://www.w3.org/2001/04/xmlenc#aes256-cbc"
},
"user_key": {
"text_hint": "Enter your email address",
"algorithm": "http://www.w3.org/2001/04/xmlenc#sha256",
"key_check": "jJEjUDipHK3OjGt6kFq7dcOLZuicQFUYwQ+TYkAIWKm6Xv6kpHFhF7LOkUK/Owww"
}
},
"links": "...",
"rights": "...",
"signature": "..."
}
A License Document must also contain a links
object. This is used to associate the License Document with resources that are not locally available.
Each Link Object nested in links
contains the link URI, a link relationship and an optional set of other link properties.
Each Link Object supports the following keys:
Name | Value | Format | Required? |
---|---|---|---|
href |
Location of the linked resource | URI or URI Template | Yes |
rel |
Link relationship to the document | List of well-known relation values, URIs for extensions | Yes |
title |
Title of the link | String | No |
type |
Expected MIME media type value for the external resources | MIME media type | No, but highly recommended |
templated |
Indicates that the href is a URI Template | Boolean | No, default value is "false" |
profile |
Expected profile used to identify the external resource | URI | No |
length |
Content length in octets | Integer | No |
hash |
SHA-256 hash of the resource | Base 64 encoded octet sequence | No |
Templated URIs follow the [URI-Template] specification.
This specification introduces the following link relationships for each Link Object:
Relation | Semantics | Required? |
---|---|---|
hint |
Location where a Reading System can redirect a User looking for additional information about the User Passphrase | Yes |
publication |
Location where the Publication associated with the License Document can be downloaded | Yes |
self |
As defined in the IANA registry of link relations: "Conveys an identifier for the link's context." | No |
support |
Support resources for the user (either a website, an email or a telephone number) | No |
In addition to these link relationships, this specification introduces the LCP Link Relations Registry, in which all link relations defined in this specification are referenced.
Link relationships may also be extended for vendor-specific applications. Such links must use a URI instead of a string to identify their link relationships.
Example 3: A License Document that points to a publication, contains the location of a hint about its User Passphrase and uses an extension to provide authentication and recommendation services.
{
"id": "ef15e740-697f-11e3-949a-0800200c9a66",
"issued": "2013-11-04T01:08:15+01:00",
"updated": "2014-02-21T09:44:17+01:00",
"provider": "https://www.imaginaryebookretailer.com",
"encryption": "...",
"links": [
{ "rel": "publication",
"href": "https://www.example.com/file.epub",
"type": "application/epub+zip",
"length": "264929",
"hash": "8b752f93e5e73a3efff1c706c1c2e267dffc6ec01c382cbe2a6ca9bd57cc8378"
},
{ "rel": "hint",
"href": "https://www.example.com/passphraseHint?user_id=1234",
"type": "text/html"
},
{ "rel": "support",
"href": "mailto:support@example.org"
},
{ "rel": "support",
"href": "tel:1800836482"
},
{ "rel": "support",
"href": "https://example.com/support",
"type": "text/html"
},
{ "rel": "https://mylcpextension.com/authentication",
"href": "https://www.example.com/authenticateMe",
"title": "Authentication service",
"type": "application/vnd.myextension.authentication+json"
},
{ "rel": "https://mylcpextension.com/book_recommendations",
"href": "https://www.example.com/recommended/1",
"type": "text/html"
},
{ "rel": "https://mylcpextension.com/book_recommendations",
"href": "https://www.example.com/recommended/1.opds",
"type": "application/atom+xml; profile=opds-catalog; kind=acquisition"}
],
"rights": "...",
"signature": "..."
}
The License Document may also express a series of rights using the rights
object. The rights
object may include the following fields:
Name | Value | Format | Default |
---|---|---|---|
print |
Maximum number of pages that can be printed over the lifetime of the license. | Integer | Unlimited |
copy |
Maximum number of characters that can be copied to the clipboard over the lifetime of the license. | Integer | Unlimited |
start |
Date and time when the license begins. | ISO 8601 | None (perpetual license) |
end |
Date and time when the license ends. | ISO 8601 | None (perpetual license) |
All name/value pairs using an integer as their data type (print and copy for this specification) must not use leading zeroes in values (e.g., "9", not “09”).
For the print
right, a page is defined as follows:
-
The page as defined in the Publication, if it is pre-paginated (fixed layout) OR
-
The page as defined by the page-list nav element of the EPUB Navigation Document, if this exists OR
-
1024 Unicode characters for all other cases
The copy
right only covers the ability to copy to the clipboard and is limited to text (no images, etc.).
In addition to these rights, this specification introduces the LCP Rights Registry, in which all rights defined in this specification are referenced.
The rights
object may be extended to include any number of implementor-specific rights. Each extension right must be identified using a URI controlled by the implementor.
Example 4: this License Document defines the following rights: print up to ten pages and copy up to 10000 characters characters for the lifetime of the license; the license has an expiration date. There is also a vendor extension granting the right to tweet parts of this book.
{
"id": "ef15e740-697f-11e3-949a-0800200c9a66",
"issued": "2013-11-04T01:08:15+01:00",
"updated": "2014-02-21T09:44:17+01:00",
"provider": "https://www.imaginaryebookretailer.com",
"encryption": "...",
"links": "...",
"rights": {
"print": 10,
"copy": 10000,
"start": "2013-11-04T01:08:15+01:00",
"end": "2013-11-25T01:08:15+01:00",
"https://www.imaginaryebookretailer.com/lcp/rights/tweet": true
},
"signature": "..."
}
The License Document may embed information about the user using the user
object. The user
object includes the following fields:
Name | Value | Format | Required? |
---|---|---|---|
id |
Unique identifier for the User at a specific Provider | String | No, but highly recommended |
email |
The User’s e-mail address | String | No |
name |
The User’s name | String | No |
encrypted |
A list of which user object values are encrypted in this License Document | Array of one or more strings matching the above names or an extension | Yes, if encryption is used for any field |
In addition to these user information, this specification introduces the LCP User Fields Registry, in which all user fields defined in this specification are referenced.
As with rights, the user
object may be extended to include any number of implementor-specific fields. Each extension field must be identified by a URI controlled by the implementor.
To protect private User data, any of these fields may be encrypted, except for the encrypted
field, which must remain in plain text. If encrypted, the field values must be encrypted using the User Key and the same encryption algorithm identified in the encryption/content_key
object. The names of all encrypted fields must be listed in the encrypted
array.
Example 5: License Document where an identifier, a provider and an email are provided for the user. There is also an extension to indicate the user’s preferred language. The email is encrypted.
{
"id": "ef15e740-697f-11e3-949a-0800200c9a66",
"issued": "2013-11-04T01:08:15+01:00",
"updated": "2014-02-21T09:44:17+01:00",
"provider": "https://www.imaginaryebookretailer.com",
"encryption": "...",
"links": "...",
"rights": "...",
"user": {
"id": "d9f298a7-7f34-49e7-8aae-4378ecb1d597",
"email": "EnCt2b8c6d2afd94ae4ed201b27049d8ce1afe31a90ceb8c6d2afd94ae4ed201b2704RjkaXRveAAarHwdlID1KCIwEmS",
"encrypted": ["email"],
"https://www.imaginaryebookretailer.com/lcp/user/language": "tlh"
},
"signature": "..."
}
As described in 5. Signature and Public Key Infrastructure, the License Document includes a digital signature to validate that it has not been altered. The License Document must include information about the signature using the signature
object. The signature
object must include the following fields:
Name | Value | Format |
---|---|---|
algorithm |
Algorithm used to calculate the signature, identified using the URIs given in [XML-SIG]. This must match the signature algorithm named in the Encryption Profile identified in encryption/profile |
URI |
certificate |
The Provider Certificate: an X509 certificate used by the Content Provider | Base 64 encoded DER certificate |
value |
Value of the signature | Base 64 encoded octet sequence |
For more information on how the signature and the certificate should be calculated, encoded and processed, see 5. Signature and Public Key Infrastructure.
Example 6: Signature based on the basic LCP encryption profile.
{
"id": "ef15e740-697f-11e3-949a-0800200c9a66",
"issued": "2013-11-04T01:08:15+01:00",
"updated": "2014-02-21T09:44:17+01:00",
"provider": "https://www.imaginaryebookretailer.com",
"encryption": "...",
"links": "...",
"rights": "...",
"user": "...",
"signature": {
"algorithm": "http://www.w3.org/2001/04/xmldsig-more#rsa-sha256",
"certificate": "MIIDEjCCAfoCCQDwMOjkYYOjPjANBgkqhkiG9w0BAQUFADBLMQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTETMBEGA1UEBxMKRXZlcnl3aGVyZTESMBAGA1UEAxMJbG9jYWxob3N0MB4XDTE0MDEwMjIxMjYxNloXDTE1MDEwMjIxMjYxNlowSzELMAkGA1UEBhMCVVMxEzARBgNVBAgTCkNhbGlmb3JuaWExEzARBgNVBAcTCkV2ZXJ5d2hlcmUxEjAQBgNVBAMTCWxvY2FsaG9zdDCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBAOpCRECG7icpf0H37kuAM7s42oqggBoikoTpo5yapy+s5eFSp8HSqwhIYgZ4SghNLkj3e652SALav7chyZ2vWvitZycY+aq50n5UTTxDvdwsC5ZNeTycuzVWZALKGhV7VUPEhtWZNm0gruntronNa8l2WS0aF7P5SbhJ65SDQGprFFaYOSyN6550P3kqaAO7tDddcA1cmuIIDRf8tOIIeMkBFk1Qf+lh+3uRP2wztOTECSMRxX/hIkCe5DRFDK2MuDUyc/iY8IbY0hMFFGw5J7MWOwZLBOaZHX+Lf5lOYByPbMH78O0dda6T+tLYAVzsmJdHJFtaRguCaJVtSXKQUAMCAwEAATANBgkqhkiG9w0BAQUFAAOCAQEAi9HIM+FMfqXsRUY0rGxLlw403f3YtAG/ohzt5i8DKiKUG3YAnwRbL/VzXLZaHru7XBC40wmKefKqoA0RHyNEddXgtY/aXzOlfTvp+xirop+D4DwJIbaj8/wHKWYGBucA/VgGY7JeSYYTUSuz2RoYtjPNRELIXN8A+D+nkJ3dxdFQ6jFfVfahN3nCIgRqRIOt1KaNI39CShccCaWJ5DeSASLXLPcEjrTi/pyDzC4kLF0VjHYlKT7lq5RkMO6GeC+7YFvJtAyssM2nqunA2lUgyQHb1q4Ih/dcYOACubtBwW0ITpHz8N7eO+r1dtH/BF4yxeWl6p5kGLvuPXNU21ThgA==",
"value": "q/3IInic9c/EaJHyG1Kkqk5v1zlJNsiQBmxz4lykhyD3dA2jg2ZzrOenYU9GxP/xhe5H5Kt2WaJ/hnt8+GWrEx1QOwnNEij5CmIpZ63yRNKnFS5rSRnDMYmQT/fkUYco7BUi7MPPU6OFf4+kaToNWl8m/ZlMxDcS3BZnVhSEKzUNQn1f2y3sUcXjes7wHbImDc6dRthbL/E+assh5HEqakrDuA4lM8XNfukEYQJnivqhqMLOGM33RnS5nZKrPPK/c2F/vGjJffSrlX3W3Jlds0/MZ6wtVeKIugR06c56V6+qKsnMLAQJaeOxxBXmbFdAEyplP9irn4D9tQZKqbbMIw=="
}
}
This section is informative
In order for any symmetrically encrypted message to work, there must be an agreed-upon key shared by the sender and receiver. In LCP, this is the User Key. The Provider must have access to the User Key in order to secure the Content Key within the License Document. The Reading System must also have the User Key in order to decrypt the Content Key in the License Document.
LCP uses a passphrase model for sharing the User Key: in a simple implementation, when the Reading System receives a new License Document, it prompts the User for a passphrase to access the Content Key. LCP defines the User Key as a hash of this User Passphrase. This passphrase can be anything at all: a User-defined password, a Provider-defined password, an e-mail address, a library card number, etc.
The User Passphrase must be a UTF-8 encoded string. There are no restrictions on the length or content of the User Passphrase. There are no requirements for how it is to be created.
The User Key is the result of applying a hashing function on the User Passphrase, using the hash algorithm provided in the encryption/user_key
object in the License Document. Processing of any kind, including whitespace escaping or normalization, shall not be done on the User Passphrase before hashing.
In order to facilitate the entry of the passphrase by the User, the LCP License Document supports two ways for the Reading System to provide passphrase prompts or hints to the User:
-
The
user_key/text_hint
field is a simple prompt that can be displayed to the User when entering the User Passphrase (e.g., "Enter your Imaginary Book Retailer password"). -
The
links
object withrel=hint
points to a location that provides passphrase hints or other assistance.
The content of user_key/text_hint
and of the resource pointed to by the links
object with rel=hint
should be human-readable, directed to the User, and should help the User enter their passphrase.
In order to simplify the process for accessing Protected Publications, the Provider should use the same User Key for all licenses issued to the same User.
The security of LCP depends in large part on the security of the User Key and User Passphrase. Therefore, special care should be taken to secure these throughout the licensing workflow:
-
Passphrases should be sufficiently complex to prevent brute-force attacks.
-
Passphrases must not be transmitted in plaintext.
-
If the Provider needs to share User Key information among multiple systems, they should transmit User Keys and not User Passphrases, and should transmit this information over secure channels.
-
User Passphrases must not be stored by the Reading System; instead it should only store User Keys.
This section is informative
Given the importance of the precise expression of various objects in the License Document, it is critical that the Reading System be able to verify that the content of the License Document is authentic and has not been altered. This is done via a digital signature that is verified via a Public Key Infrastructure as defined in [X509].
Calculating a signature is done on a byte stream, which is unique, while the License Document is a JSON document where multiple representations might lead to the same structure. Thus, to ensure a stable signature between the Reading System and the Content Provider, some transformations must be applied prior to signing the Document and verifying the signature.
The steps required from the Provider to sign the License Document are:
-
The contents of the License Document (minus the signature object) are put in a canonical form: alphabetized with non-significant whitespace removed (see Section 5.3)
-
The signature value for the canonical form of the License Document is calculated following the algorithm identified in the Encryption Profile. This typically will involve taking a hash of the data and encrypting it using the private key.
-
The signature value and the Provider Certificate are added to the License Document in the
signature
object, as described in Section 3.8.
The steps required of the Reading System to validate the signature are:
-
Obtain the Root Certificate from the License Authority. This certificate is embedded in the Reading System to validate all Provider Certificates.
-
Using this Root Certificate, validate the Provider Certificate that is included in the
signature
object. This prevents another entity who is not the Provider from forging License Documents. -
Strip the
signature
object from the License Document. -
Put the remainder of the License Document contents into canonical form.
-
Calculate the hash of the canonical License Document data using the signature algorithm identified in the Encryption Profile.
-
Decrypt the signature value given in the signature object using the public key given in the Provider Certificate and confirm that it matches the calculated hash. This confirms that the contents of the License Document have not been altered in transit.
Because the Provider Certificate is included in the License Document and the Root Certificate is embedded in the Reading System, the entire validation process can take place without any Internet connectivity.
To make sure that the Provider Certificate has not been revoked, the Reading System also checks a Certificate Revocation List maintained by the Readium Foundation. In order to facilitate offline reading, the Reading System does not need to check the revocation list every time it processes a License Document. It is only necessary that it updates the list regularly when an Internet connection is available (e.g., every time it downloads a new License or book).
Content Providers must have a Certificate in the [X509] v3 format issued and signed by the License Authority using the Root Certificate: this is referred to here as the Provider Certificate. The subject of the Provider Certificate should represent the Content Provider.
Content Providers must distribute their Provider Certificate in any License Document they issue in the signature/certificate
field. They also must use the private key paired with their Provider Certificate’s public key to sign the License Document. For a License Document to be considered valid, the Provider Certificate must have been valid at the time the License Document was issued (as indicated by the issued
field), and the Provider Certificate must not have been revoked.
Reading Systems must obtain the Root Certificate in the [X509] v3 format from the License Authority, and should keep it up to date. It must be embedded in the Reading System for offline use.
The canonical form of the License Document is used when calculating and validating the signature. To create the canonical form of the License Document, the following serialization rules must be followed:
-
Since the
signature
object of the License Document is the product of the calculation, it must be removed. -
All JSON object properties (i.e. key/value pairs) in the License Document must be lexicographically sorted by Unicode Code Point. Note that this rule is recursive, so that members are sorted at all levels of object nesting.
-
Within arrays, the order of elements must not be altered.
-
Numbers must not include insignificant leading or trailing zeroes. Numbers that include a fraction part (non-integers) must be expressed as a number, fraction, and exponent (normalized scientific notation) using an upper-case "E".
-
Strings must use escaping only for those characters for which it is required by [JSON]: backslash (\), double-quotation mark ("), and control characters (U+0000 through U+001F). When escaping control characters, the hexadecimal digits must be upper case.
-
Non-significant whitespace (as defined in [JSON]) must be removed. Whitespace found within strings must be kept.
This section is informative
For this example, we'll use a License Document which contains a link (hint) and basic user information.
{
"id": "ef15e740-697f-11e3-949a-0800200c9a66",
"issued": "2013-11-04T01:08:15+01:00",
"updated": "2014-02-21T09:44:17+01:00",
"provider": "https://www.imaginaryebookretailer.com",
"encryption": {
"profile": "http://readium.org/lcp/basic-profile",
"content_key": {
"encrypted_value": "/k8RpXqf4E2WEunCp76E8PjhS051NXwAXeTD1ioazYxCRGvHLAck/KQ3cCh5JxDmCK0nRLyAxs1X0aA3z55boQ==",
"algorithm": "http://www.w3.org/2001/04/xmlenc#aes256-cbc"
},
"user_key": {
"text_hint": "Enter your email address",
"algorithm": "http://www.w3.org/2001/04/xmlenc#sha256",
"key_check": "jJEjUDipHK3OjGt6kFq7dcOLZuicQFUYwQ+TYkAIWKm6Xv6kpHFhF7LOkUK/Owww"
}
},
"links": [
{ "rel": "hint", "href": "https://www.imaginaryebookretailer.com/lcp/hint", "type": "text/html"}
],
"user": { "id": "d9f298a7-7f34-49e7-8aae-4378ecb1d597"}
}
First of all, let's sort this document. Every [JSON] object needs to be sorted:
{
"encryption": {
"content_key": {
"algorithm": "http://www.w3.org/2001/04/xmlenc#aes256-cbc",
"encrypted_value": "/k8RpXqf4E2WEunCp76E8PjhS051NXwAXeTD1ioazYxCRGvHLAck/KQ3cCh5JxDmCK0nRLyAxs1X0aA3z55boQ=="
},
"profile": "http://readium.org/lcp/basic-profile",
"user_key": {
"algorithm": "http://www.w3.org/2001/04/xmlenc#sha256",
"key_check": "jJEjUDipHK3OjGt6kFq7dcOLZuicQFUYwQ+TYkAIWKm6Xv6kpHFhF7LOkUK/Owww",
"text_hint": "Enter your email address"
}
},
"id": "ef15e740-697f-11e3-949a-0800200c9a66",
"issued": "2013-11-04T01:08:15+01:00",
"links": [
{
"rel": "hint",
"href": "https://www.imaginaryebookretailer.com/lcp/hint",
"type": "text/html"
}
],
"provider": "https://www.imaginaryebookretailer.com",
"updated": "2014-02-21T09:44:17+01:00",
"user": {"id": "d9f298a7-7f34-49e7-8aae-4378ecb1d597"}
}
Now that our document is sorted, we can strip all whitespaces and end of lines:
{"encryption":{"content_key":{"algorithm":"http://www.w3.org/2001/04/xmlenc#aes256-cbc","encrypted_value":"/k8RpXqf4E2WEunCp76E8PjhS051NXwAXeTD1ioazYxCRGvHLAck/KQ3cCh5JxDmCK0nRLyAxs1X0aA3z55boQ=="},"profile":"http://readium.org/lcp/basic-profile","user_key":{"algorithm":"http://www.w3.org/2001/04/xmlenc#sha256","key_check":"jJEjUDipHK3OjGt6kFq7dcOLZuicQFUYwQ+TYkAIWKm6Xv6kpHFhF7LOkUK/Owww","text_hint":"Enter your email address"}},"id":"ef15e740-697f-11e3-949a-0800200c9a66","issued":"2013-11-04T01:08:15+01:00","links":[{"rel":"hint","href":"https://www.imaginaryebookretailer.com/lcp/hint","type":"text/html"}],"provider":"https://www.imaginaryebookretailer.com","updated":"2014-02-21T09:44:17+01:00","user":{"id":"d9f298a7-7f34-49e7-8aae-4378ecb1d597"}}
In order to sign a License Document, the Content Provider must go through the following steps in order:
-
The Content Provider must create the canonical form of the License Document following the rules given in 5.3. Canonical form of the License Document.
-
The Content Provider must calculate the signature using this canonical form of the License Document and using the algorithm described in the
algorithm
field with the private key of the Provider Certificate. -
The signature must be embedded in the
value
field of thesignature
object using Base 64 encoding. -
The Provider Certificate used to validate the signature must be inserted in the
certificate
field. It must use the DER notation and be encoded using Base 64.
This section is informative
Given the License Document in its canonical form
{"encryption":{"content_key":{"algorithm":"http://www.w3.org/2001/04/xmlenc#aes256-cbc","encrypted_value":"/k8RpXqf4E2WEunCp76E8PjhS051NXwAXeTD1ioazYxCRGvHLAck/KQ3cCh5JxDmCK0nRLyAxs1X0aA3z55boQ=="},"profile":"http://readium.org/lcp/basic-profile","user_key":{"algorithm":"http://www.w3.org/2001/04/xmlenc#sha256","key_check":"jJEjUDipHK3OjGt6kFq7dcOLZuicQFUYwQ+TYkAIWKm6Xv6kpHFhF7LOkUK/Owww","text_hint":"Enter your email address"}},"id":"ef15e740-697f-11e3-949a-0800200c9a66","issued":"2013-11-04T01:08:15+01:00","links":[{"rel":"hint","href":"https://www.imaginaryebookretailer.com/lcp/hint","type":"text/html"}],"provider":"https://www.imaginaryebookretailer.com","updated":"2014-02-21T09:44:17+01:00","user":{"id":"d9f298a7-7f34-49e7-8aae-4378ecb1d597"}}
Using the signature algorithm required by the Encryption Profile, a Content Provider must first hash the License Document, giving the following byte sequence, represented here in hexadecimal:
23c68442c7214ba294ddd1a2902756e9fe575116a88f36e55baf94590a90c2ad
This SHA-256 form must then be signed using the Content Provider’s Private Key, giving the following result, in Base 64:
q/3IInic9c/EaJHyG1Kkqk5v1zlJNsiQBmxz4lykhyD3dA2jg2ZzrOenYU9GxP/xhe5H5Kt2WaJ/hnt8+GWrEx1QOwnNEij5CmIpZ63yRNKnFS5rSRnDMYmQT/fkUYco7BUi7MPPU6OFf4+kaToNWl8m/ZlMxDcS3BZnVhSEKzUNQn1f2y3sUcXjes7wHbImDc6dRthbL/E+assh5HEqakrDuA4lM8XNfukEYQJnivqhqMLOGM33RnS5nZKrPPK/c2F/vGjJffSrlX3W3Jlds0/MZ6wtVeKIugR06c56V6+qKsnMLAQJaeOxxBXmbFdAEyplP9irn4D9tQZKqbbMIw==
With this signature and the certificate, a valid license will be created as e.g.:
{
"issued":"2013-11-04T01:08:15+01:00",
"updated":"2014-02-21T09:44:17+01:00",
"encryption": {
"content_key": {
"algorithm": "http://www.w3.org/2001/04/xmlenc#aes256-cbc",
"encrypted_value": "/k8RpXqf4E2WEunCp76E8PjhS051NXwAXeTD1ioazYxCRGvHLAck/KQ3cCh5JxDmCK0nRLyAxs1X0aA3z55boQ=="
},
"profile": "http://readium.org/lcp/basic-profile",
"user_key": {
"algorithm": "http://www.w3.org/2001/04/xmlenc#sha256",
"key_check":"jJEjUDipHK3OjGt6kFq7dcOLZuicQFUYwQ+TYkAIWKm6Xv6kpHFhF7LOkUK/Owww",
"text_hint": "Enter your email address"
}
},
"id": "ef15e740-697f-11e3-949a-0800200c9a66",
"links": [
{ "rel": "hint",
"href": "https://www.imaginaryebookretailer.com/lcp/hint",
"type": "text/html"
}
],
"user": {"id": "d9f298a7-7f34-49e7-8aae-4378ecb1d597"},
"signature": {
"algorithm": "http://www.w3.org/2001/04/xmldsig-more#rsa-sha256",
"certificate": "MIIDEjCCAfoCCQDwMOjkYYOjPjANBgkqhkiG9w0BAQUFADBLMQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTETMBEGA1UEBxMKRXZlcnl3aGVyZTESMBAGA1UEAxMJbG9jYWxob3N0MB4XDTE0MDEwMjIxMjYxNloXDTE1MDEwMjIxMjYxNlowSzELMAkGA1UEBhMCVVMxEzARBgNVBAgTCkNhbGlmb3JuaWExEzARBgNVBAcTCkV2ZXJ5d2hlcmUxEjAQBgNVBAMTCWxvY2FsaG9zdDCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBAOpCRECG7icpf0H37kuAM7s42oqggBoikoTpo5yapy+s5eFSp8HSqwhIYgZ4SghNLkj3e652SALav7chyZ2vWvitZycY+aq50n5UTTxDvdwsC5ZNeTycuzVWZALKGhV7VUPEhtWZNm0gruntronNa8l2WS0aF7P5SbhJ65SDQGprFFaYOSyN6550P3kqaAO7tDddcA1cmuIIDRf8tOIIeMkBFk1Qf+lh+3uRP2wztOTECSMRxX/hIkCe5DRFDK2MuDUyc/iY8IbY0hMFFGw5J7MWOwZLBOaZHX+Lf5lOYByPbMH78O0dda6T+tLYAVzsmJdHJFtaRguCaJVtSXKQUAMCAwEAATANBgkqhkiG9w0BAQUFAAOCAQEAi9HIM+FMfqXsRUY0rGxLlw403f3YtAG/ohzt5i8DKiKUG3YAnwRbL/VzXLZaHru7XBC40wmKefKqoA0RHyNEddXgtY/aXzOlfTvp+xirop+D4DwJIbaj8/wHKWYGBucA/VgGY7JeSYYTUSuz2RoYtjPNRELIXN8A+D+nkJ3dxdFQ6jFfVfahN3nCIgRqRIOt1KaNI39CShccCaWJ5DeSASLXLPcEjrTi/pyDzC4kLF0VjHYlKT7lq5RkMO6GeC+7YFvJtAyssM2nqunA2lUgyQHb1q4Ih/dcYOACubtBwW0ITpHz8N7eO+r1dtH/BF4yxeWl6p5kGLvuPXNU21ThgA==",
"value": "q/3IInic9c/EaJHyG1Kkqk5v1zlJNsiQBmxz4lykhyD3dA2jg2ZzrOenYU9GxP/xhe5H5Kt2WaJ/hnt8+GWrEx1QOwnNEij5CmIpZ63yRNKnFS5rSRnDMYmQT/fkUYco7BUi7MPPU6OFf4+kaToNWl8m/ZlMxDcS3BZnVhSEKzUNQn1f2y3sUcXjes7wHbImDc6dRthbL/E+assh5HEqakrDuA4lM8XNfukEYQJnivqhqMLOGM33RnS5nZKrPPK/c2F/vGjJffSrlX3W3Jlds0/MZ6wtVeKIugR06c56V6+qKsnMLAQJaeOxxBXmbFdAEyplP9irn4D9tQZKqbbMIw=="
}
}
-
The Reading System must check the signature of the Provider Certificate using the Root Certificate it embeds.
-
If a network connection is available, the Reading System must periodically update its Certificate Revocation List, as defined in [X509].
-
The Reading System must check that the Certificate was not revoked, as defined in [X509].
-
The Reading System must check that the Certificate was not expired when the License Document was last updated.
In order to validate the signature, the following steps must be followed in order:
-
The Reading System must extract and remove the signature from the License Document.
-
The Reading System must calculate the canonical form of the License Document following the rules expressed in 5.3. Canonical form of the License Document.
-
The Reading System must recalculate the signature as defined in 5.4. Generating the signature.
-
The Reading System must verify that the calculated signature value is consistent with the one previously extracted from the License Document.
This section is informative
LCP is entirely based on standard encryption algorithms, as defined in [XML-ENC] and [XML-SIG]. In order to maintain maximum flexibility, no specific algorithms are mandated by this specification. Instead, the design of both encryption.xml
and the License Document allow for the identification of encryption algorithms to be discovered by Reading Systems when presented with a Protected Publication.
In order to simplify the discovery process, LCP defines the notion of Encryption Profile, which is the set of encryption algorithms used in a specific Protected Publication and associated Licence Document. Reading Systems that implement the algorithms identified in the Encryption Profile will be able to decrypt Protected Publications encoded using such Encryption Profile. The identification of the Encryption Profile in the License Document eases the discovery of these requirements by Reading Systems.
This specification defines the Basic Encryption Profile 1.0, composed from a set of associated algorithms extracted from [XML-ENC] or [XML-SIG]. The Basic Encryption Profile is for test only, as it does not provide the level of obfuscation required by a reliable protection mechanism.
Other Encryption Profiles are (or will be) defined for use in production; these profiles are referenced in the LCP Encryption Profiles Registry.
All Encryption Profiles must identify algorithms for the following targets:
-
Publication Resources
-
Content Key and User fields (if encrypted)
-
User Passphrase
-
Signature
All algorithms used in an Encryption Profile should be defined in [XML-ENC] or [XML-SIG].
All Encryption Profiles must use a URI to identify themselves in profile
(contained in the encryption
object of the License Document).
The Basic Encryption Profile 1.0 is identified in the encryption
object of the License Document using the URL http://readium.org/lcp/basic-profile
as value of the profile
attribute .
The following algorithms are associated to the Basic Encryption Profile 1.0:
Encryption target | Algorithm (name) | Algorithm (URI) | Identified in |
---|---|---|---|
Publication Resources | AES 256 bits CBC | http://www.w3.org/2001/04/xmlenc#aes256-cbc | encryption.xml |
Content Key, User fields (if encrypted) | AES 256 bits CBC | http://www.w3.org/2001/04/xmlenc#aes256-cbc | License Document |
User Passphrase | SHA-256 | http://www.w3.org/2001/04/xmlenc#sha256 | License Document |
Signature | RSA with SHA-256 | http://www.w3.org/2001/04/xmldsig-more#rsa-sha256 | License Document |
Reading Systems can detect that a Publication is protected using LCP by either of these means:
-
The presence of the License Document (
META-INF/license.lcpl
) -
The presence of a
ds:KeyInfo/ds:RetrievalMethod
element inMETA-INF/encryption.xml
that references an LCP Content Key (i.e.,Type
attribute value is "http://readium.org/2014/01/lcp#EncryptedContentKey
" )
If encryption.xml
references an LCP Content Key but the License Document is missing, the Reading System should report this to the User.
In processing a License Document, Reading Systems must ignore all name/value pairs they do not understand.
Reading Systems must not store unencrypted versions of the Content Key and/or encrypted user fields.
Reading Systems must:
-
Validate the syntax and completeness of the License Document.
-
Validate the signature as defined in 5.5. Validating the certificate and signature.
When License Documents are delivered independently of the Protected Publication, Reading Systems must download the Protected Publication at the URL given in links/publication/href
.
A Reading System that will make the Protected Publication file accessible to the User must add the License Document to the downloaded Protected Publication at META-INF/license.lcpl
.
Reading Systems should verify the integrity of the downloaded Protected Publication, if a hash is provided.
A Reading System should report any failure to acquire the Protected Publication to the user.
Reading Systems must:
- Show the text hint and URL when prompting the User for their User Passphrase.
Reading Systems should:
-
Store the User Key, but in a secured manner.
-
Try previously stored User Keys before prompting the User to enter their User Passphrase for a new Protected Publication.
Reading Systems may:
-
Use an alternate technique to discover and exchange the User Key in the background.
-
Associate specific User Keys with specific Providers and User IDs in order to optimize the discovery of the correct User Key to process a License Document.
Reading Systems must not:
- Store the User Passphrase, only the User Key.
Reading Systems must:
-
Validate the Signature and Provider Certificate as described in 5.5. Validating the certificate and signature.
-
Update the Certification Revocation List on a regular basis.
Reading Systems should:
- Be able to update their Root Certificate.
Reading Systems may:
- Choose to open a Publication even if their own Root Certificate is deprecated.
Reading Systems must not:
-
Open a Publication with an invalid Signature or Provider Certificate.
-
Block a User from opening a Publication if a network connection is unavailable or the Certificate Revocation List can't be reached.
Reading Systems must:
- Respect all rights limitations given in the License Document.
Reading Systems may:
- Delete a Protected Publication if its License expired.
Reading Systems must not:
- Store unencrypted Publication Resources.
- [EPUB] EPUB 3.2.
- [JSON] The application/json Media Type for JavaScript Object Notation (JSON).
- [JSON Pointer] JavaScript Object Notation (JSON) Pointer.
- [OCF] Open Container Format 3.2.
- [RFC2119] Key words for use in RFCs to Indicate Requirement Levels.
- [URI-Template] URI Template.
- [X509] Internet X.509 Public Key Infrastructure Certificate and Certificate Revocation List (CRL) Profile.
- [XML-ENC] XML Encryption Syntax and Processing Version 1.1.
- [XML-SIG] XML Signature Syntax and Processing (Second Edition).
- LCP Link Relations Registry
- LCP Rights Registry
- LCP User Fields Registry
- LCP Encryption Profiles Registry
A JSON Schema for LCP 1.0 is available under version control at https://github.com/readium/lcp-specs/tree/master/schema
For the purpose of validating a license, use the following JSON Schema resource: https://readium.org/lcp-specs/schema/license.schema.json