<!DOCTYPE html>

<html lang="en" xmlns="http://www.w3.org/1999/xhtml" xml:lang="en">
<head profile="http://www.w3.org/2006/03/hcard http://dublincore.org/documents/2008/08/04/dc-html/">

  <title>OpenID Connect Account Porting</title>

  <style type="text/css" title="Xml2Rfc (sans serif)">
  /*<![CDATA[*/
          a {
          text-decoration: none;
          }
      /* info code from SantaKlauss at http://www.madaboutstyle.com/tooltip2.html */
      a.info {
          /* This is the key. */
          position: relative;
          z-index: 24;
          text-decoration: none;
      }
      a.info:hover {
          z-index: 25;
          color: #FFF; background-color: #900;
      }
      a.info span { display: none; }
      a.info:hover span.info {
          /* The span will display just on :hover state. */
          display: block;
          position: absolute;
          font-size: smaller;
          top: 2em; left: -5em; width: 15em;
          padding: 2px; border: 1px solid #333;
          color: #900; background-color: #EEE;
          text-align: left;
      }
          a.smpl {
          color: black;
          }
          a:hover {
          text-decoration: underline;
          }
          a:active {
          text-decoration: underline;
          }
          address {
          margin-top: 1em;
          margin-left: 2em;
          font-style: normal;
          }
          body {
          color: black;
          font-family: verdana, helvetica, arial, sans-serif;
          font-size: 10pt;
          max-width: 55em;
          
          }
          cite {
          font-style: normal;
          }
          dd {
          margin-right: 2em;
          }
          dl {
          margin-left: 2em;
          }
        
          ul.empty {
          list-style-type: none;
          }
          ul.empty li {
          margin-top: .5em;
          }
          dl p {
          margin-left: 0em;
          }
          dt {
          margin-top: .5em;
          }
          h1 {
          font-size: 14pt;
          line-height: 21pt;
          page-break-after: avoid;
          }
          h1.np {
          page-break-before: always;
          }
          h1 a {
          color: #333333;
          }
          h2 {
          font-size: 12pt;
          line-height: 15pt;
          page-break-after: avoid;
          }
          h3, h4, h5, h6 {
          font-size: 10pt;
          page-break-after: avoid;
          }
          h2 a, h3 a, h4 a, h5 a, h6 a {
          color: black;
          }
          img {
          margin-left: 3em;
          }
          li {
          margin-left: 2em;
          margin-right: 2em;
          }
          ol {
          margin-left: 2em;
          margin-right: 2em;
          }
          ol p {
          margin-left: 0em;
          }
          p {
          margin-left: 2em;
          margin-right: 2em;
          }
          pre {
          margin-left: 3em;
          background-color: lightyellow;
          padding: .25em;
          }
          pre.text2 {
          border-style: dotted;
          border-width: 1px;
          background-color: #f0f0f0;
          width: 69em;
          }
          pre.inline {
          background-color: white;
          padding: 0em;
          }
          pre.text {
          border-style: dotted;
          border-width: 1px;
          background-color: #f8f8f8;
          width: 69em;
          }
          pre.drawing {
          border-style: solid;
          border-width: 1px;
          background-color: #f8f8f8;
          padding: 2em;
          }
          table {
          margin-left: 2em;
          }
          table.tt {
          vertical-align: top;
          }
          table.full {
          border-style: outset;
          border-width: 1px;
          }
          table.headers {
          border-style: outset;
          border-width: 1px;
          }
          table.tt td {
          vertical-align: top;
          }
          table.full td {
          border-style: inset;
          border-width: 1px;
          }
          table.tt th {
          vertical-align: top;
          }
          table.full th {
          border-style: inset;
          border-width: 1px;
          }
          table.headers th {
          border-style: none none inset none;
          border-width: 1px;
          }
          table.left {
          margin-right: auto;
          }
          table.right {
          margin-left: auto;
          }
          table.center {
          margin-left: auto;
          margin-right: auto;
          }
          caption {
          caption-side: bottom;
          font-weight: bold;
          font-size: 9pt;
          margin-top: .5em;
          }
        
          table.header {
          border-spacing: 1px;
          width: 95%;
          font-size: 10pt;
          color: white;
          }
          td.top {
          vertical-align: top;
          }
          td.topnowrap {
          vertical-align: top;
          white-space: nowrap; 
          }
          table.header td {
          background-color: gray;
          width: 50%;
          }
          table.header a {
          color: white;
          }
          td.reference {
          vertical-align: top;
          white-space: nowrap;
          padding-right: 1em;
          }
          thead {
          display:table-header-group;
          }
          ul.toc, ul.toc ul {
          list-style: none;
          margin-left: 1.5em;
          margin-right: 0em;
          padding-left: 0em;
          }
          ul.toc li {
          line-height: 150%;
          font-weight: bold;
          font-size: 10pt;
          margin-left: 0em;
          margin-right: 0em;
          }
          ul.toc li li {
          line-height: normal;
          font-weight: normal;
          font-size: 9pt;
          margin-left: 0em;
          margin-right: 0em;
          }
          li.excluded {
          font-size: 0pt;
          }
          ul p {
          margin-left: 0em;
          }
        
          .comment {
          background-color: yellow;
          }
          .center {
          text-align: center;
          }
          .error {
          color: red;
          font-style: italic;
          font-weight: bold;
          }
          .figure {
          font-weight: bold;
          text-align: center;
          font-size: 9pt;
          }
          .filename {
          color: #333333;
          font-weight: bold;
          font-size: 12pt;
          line-height: 21pt;
          text-align: center;
          }
          .fn {
          font-weight: bold;
          }
          .hidden {
          display: none;
          }
          .left {
          text-align: left;
          }
          .right {
          text-align: right;
          }
          .title {
          color: #990000;
          font-size: 18pt;
          line-height: 18pt;
          font-weight: bold;
          text-align: center;
          margin-top: 36pt;
          }
          .vcardline {
          display: block;
          }
          .warning {
          font-size: 14pt;
          background-color: yellow;
          }
        
        
          @media print {
          .noprint {
                display: none;
          }
        
          a {
                color: black;
                text-decoration: none;
          }
        
          table.header {
                width: 90%;
          }
        
          td.header {
                width: 50%;
                color: black;
                background-color: white;
                vertical-align: top;
                font-size: 12pt;
          }
        
          ul.toc a::after {
                content: leader('.') target-counter(attr(href), page);
          }
        
          ul.ind li li a {
                content: target-counter(attr(href), page);
          }
        
          .print2col {
                column-count: 2;
                -moz-column-count: 2;
                column-fill: auto;
          }
          }
        
          @page {
          @top-left {
                   content: "Internet-Draft"; 
          } 
          @top-right {
                   content: "August 2016"; 
          } 
          @top-center {
                   content: "OIDC Porting";
          } 
          @bottom-left {
                   content: "Manger"; 
          } 
          @bottom-center {
                   content: "Expires Febuary 2017"; 
          } 
          @bottom-right {
                   content: "[Page " counter(page) "]"; 
          } 
          }
        
          @page:first { 
                @top-left {
                  content: normal;
                }
                @top-right {
                  content: normal;
                }
                @top-center {
                  content: normal;
                }
          }
  /*]]>*/
  </style>

</head>

<body>

<table class="header">
  <tbody>
        <tr>
      <td class="left"></td>
      <td class="right">J. Manger</td>
    </tr>
    <tr>
      <td class="left"></td>
      <td class="right">Telstra</td>
    </tr>
    <tr>
      <td class="left"></td>
      <td class="right">August 10, 2016</td>
    </tr>
  </tbody>
</table>

<p class="title">OpenID Connect Account Porting<br />
<span class="filename">draft-account-porting-00</span></p>
  

<h1 id="rfc.abstract">Abstract</a></h1>

<p>This document specifies mechanisms to support a user porting from one OpenID Connect Provider to another, such that relying parties can automatically recognize and verify the change.</p>

<hr class="noprint" />


<h1 class="np" id="rfc.toc">Table of Contents</h1>

<ul class="toc">
  <li>1.   <a href="#rfc.section.1">Introduction</a></li>
  <ul>
    <li>1.1.   <a href="#rfc.section.1.1">Requirements Notation, Conventions, and Terminology</a></li>
  </ul>
  <li>2.   <a href="#rfc.section.2">Discovery</a></li>
  <li>3.   <a href="#rfc.section.3">Porting data API</a></li>
  <li>4.   <a href="#rfc.section.4">Indicating a previous identity: "<tt>aka</tt>" member</a></li>
  <li>5.   <a href="#rfc.section.5">Porting check API</a></li>
  <li>6.   <a href="#rfc.section.6">Security Considerations</a></li>
  <ul>
    <li>6.1.   <a href="#rfc.section.6.1">Impersonation of legitimate user's old OP account</a></li>
    <li>6.2.   <a href="#rfc.section.6.2">Trick user into login with attacker's account</a></li>
    <li>6.3.   <a href="#rfc.section.6.3">Impersonate old OP/replay migration data</a></li>
    <li>6.4.   <a href="#rfc.section.6.4">Bad OP</a></li>
    <li>6.5.   <a href="#rfc.section.6.5">Inject migration data during RP account migration</a></li>
    <li>6.6.   <a href="#rfc.section.6.6">Further Considerations</a></li>
  </ul>
  <li>7.   <a href="#rfc.section.7">Privacy Considerations</a></li>
  <li>8.   <a href="#rfc.section.8">IANA Considerations</a></li>
  <li>9.   <a href="#rfc.references">References</a></li>
  <ul>
    <li>9.1.   <a href="#rfc.references.1">Normative References</a></li>
    <li>9.2.   <a href="#rfc.references.2">Informative References</a></li>
  </ul>
  <li>Appendix A.   <a href="#rfc.appendix.A">Full example</a></li>
  <li>Appendix B.   <a href="#rfc.appendix.B">Acknowledgements</a></li>
  <li>Appendix C.   <a href="#rfc.appendix.C">Notices</a></li>
  <li>Appendix D.   <a href="#rfc.appendix.D">Document history</a></li>
  <li><a href="#rfc.authors">Author's Address</a></li>
</ul>


<h1 id="rfc.section.1">1. <a id="Introduction">Introduction</a></h1>

<p>OpenID Connect is a federated identity protocol that allows a user to leverage his or her authentication to an OpenID Connect Provider (OP) to login to Relying Parties (RPs). An RP can (and often will) accept logins via many OPs. The primary identifier for a user at an RP is the combination of a Subject Id ("<tt>sub</tt>" value) and OP identifier ("<tt>iss</tt>" value). The "<tt>sub</tt>" value is only defined to be unambiguous within the context of an "<tt>iss</tt>" value [<a href="#OIDC">OIDC</a> &#xA7;5.7. Claim Stability and Uniqueness]. This design deliberately isolates each OP's user population so one OP cannot adversely affect the security of another OP. One consequence, however, is that a user porting from one OP to another appears to an RP as two unrelated users.</p>

<p>This document extends OpenID Connect to enable an RP to recognize when a user logging in via a New OP is the same user who has previously logged in via an Old OP. That is, it enables porting between OPs to be handled automatically by RPs. Portability is achieved by defining three processes: 1) how the New OP obtains details about the user from the Old OP; 2) how the Old OP is identified to the RP in an id_token issued by the New OP; and 3) how the RP can confirm with the Old OP that the port is legitimate. The first process occurs when a user ports between two OPs. The other two processes occur for each RP the first time the user logs into that RP after the porting.</p>

<p>OpenID Connect account porting is motivated by the GSMA Mobile Connect scheme, in which mobile network operators collectively offer federated identity services [<a href="#MobileConnect">MobileConnect</a>]. Most jurisdictions allow users to port their mobile phone number between network operators, and a reasonable expectation of those users will be that their Mobile Connect account (provided by those same operators) can also be ported at the same time. There is a significant difference between phone number and account porting, however. Other parties are generally unaware when number porting occurs; they simply keep using the same phone number to communicate with the user. In contrast, RPs are aware when an OpenID Connect account is ported as they interact with a New OP.</p>

<p>Section 2 defines metadata that indicates an OP supports porting, which can be obtained via OpenID Discovery [<a href="#Discovery">Discovery</a>]. Section 3 specifies an API offered by an Old OP that a New OP uses to obtain porting-related details about a user. This API is protected by an OAuth 2.0 exchange to ensure the user consents to releasing the details to the New OP. Section 4 specifies how a New OP can identify the Old OP in an id_token presented to an RP. Section 5 describes how an RP can confirm the porting information with the Old OP. Annex A provides a example of a complete porting flow.</p>


<h1 id="rfc.section.1.1">1.1. <a id="rnc">Requirements Notation, Conventions, and Terminology</a></h1>

<p>The key words "MUST", "MUST NOT", "REQUIRED", "SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in <a href="#RFC2119">[RFC2119]</a>.</p>

<p>Throughout this document, values are quoted to indicate that they are to be taken literally in protocol messages.</p>

<p>Terms such as "OpenID Provider" (OP), "Relying Party" (RP), "Subject Id" (sub), and "Sector Id" are defined in [<a href="#OIDC">OIDC</a>]. This document uses the terms "Old OP" and "New OP" to indicate an OpenID Provider that supports users porting out and porting in respectively. An OP can support both roles (and hopefully most will).</p>


<h1 id="rfc.section.2">2. <a id="discovery">Discovery</a></h1>

<p>OpenID Discovery offers details about the features supported by an OP [<a href="#Discovery">Discovery</a>]. The following members MUST be included in the metadata for an Old OP that supports users porting out:</p>

<dl>

  <dt>port_data_endpoint</dt>
  <dd style="margin-left: 8">Base URL for the Porting data API described in section 3 that a New OP can use to obtain details about an account that is being ported. The URL MUST use the <tt>https</tt> scheme.</dd>

  <dt>port_check_endpoint</dt>
  <dd style="margin-left: 8">Base URL for the Porting check API described in section 5 that an RP can use to confirm that an account has been ported to a specific New OP. The URL MUST use the <tt>https</tt> scheme.</dd>

</dl>

<p>Below is a sample metadata request and respone.</p>

<pre>GET /.well-known/openid-configuration HTTP/1.1
Host: oldop.example.net</pre>

<pre>HTTP/1.1 200 OK
Content-Type: application/json

{
  "issuer": "https://oldop.example.net/",
  "authorization_endpoint": "https://oldop.example.net/connect/authorize",
  "token_endpoint": "https://oldop.example.net/connect/token",
  "port_data_endpoint": "https://oldop.example.net/connect/port_data",
  "port_check_endpoint": "https://oldop.example.net/connect/port_check",
  &#x2026;
}</pre>


<h1 id="rfc.section.3">3. <a id="port_data">Porting data API</a></h1>

<p>The Porting data API (offered by an Old OP and consumed by a New OP) provides a port_token for each RP with whom a specific user has federated. The New OP passes the relevant port_token to each RP when the user subsequently logs into that RP [<a href="#aka">section 4</a>]. The RP, in turn, can use the port_token with the Porting check API [<a href="#port_check">section 5</a>] to confirm the port with the Old OP.</p>

<p>A separate port_token is required for each Subject Id ("<tt>sub</tt>" value) that is still in use. An Old OP that provides the same "<tt>sub</tt>" value to every RP only needs to provide a single port_token. Whereas an Old OP that uses pairwise Subject Ids needs to provide a separate port_token for each group of related RPs, that is for each Sector Id. See section 8 "Subject Identifier Types" in [<a href="#OIDC">OIDC</a>] for an explanation of Sector Ids.</p>

<p>Access to porting data requires the consent of the user to whom it relates. The Old OP MUST support the OAuth 2.0 authorization code flow for accessing the API [RFC6749]. The Old OP MUST support Bearer access tokens [<a href="RFC6450">RFC6450</a>]. The New OP MUST include the scope "<tt>porting_data</tt>" when requesting access.</p>

<p>The API provides a single read-only resource per user. It can be read with a <tt>GET</tt> request. The path "<tt>/me</tt>" appended to the base URL (from the "<tt>port_data_endpoint</tt>" member in the Old OP discovery metadata) indicates the resource for the user associated with the access token used to make the request.</p>

<p>The response is a JSON object with "<tt>pairwise</tt>" and/or "<tt>public</tt>" members. The "<tt>pairwise</tt>" member value is a JSON object whose member names are Sector Ids of RPs with whom the user has federated. The value of each Sector Id member (and of the top-level "<tt>public</tt>" member) can be either a string (the port_token value), or a JSON object with a "<tt>port_token</tt>" member with a string value. The latter syntax allows extensibility; the former is a shortcut. A New OP MUST support both syntaxes for obtaining port_token values.</p>

<p>Any unrecognized members MUST be ignored (such as "<tt>extra_stuff</tt>" in the example below).</p>

<p>An Old OP MUST choose port_token values to have sufficient entropy that they cannot be guessed by other parties. It is NOT RECOMMENDED that the Subject Id ("<tt>sub</tt>" value) be used as a port_token as that might tempt RP software to process a port_token from a New OP without confirming it with the Old OP.</p>

<p>Below is a sample porting data request and respone. The user has used the Old OP to federate to 3 RPs (or groups of related RPs) using pairwise subject ids.</p>

<pre>GET /connect/port_data/me HTTP/1.1
Host: oldop.example.net
Authorization: Bearer E3yyDiR5_i8CFCVDo3h8T5qgKpAdu8XkGZBv81vn428</pre>

<pre>HTTP/1.1 200 OK
Content-Type: application/json

{
  "pairwise": {
    "rp.example.org": "c8nGXBcdTK0YYeVf3Bnjd4NuLIU_ICR3qUMkrzO_4ho",
    "music.example": "aZ3RxSlgsHx_8xgWCkheNc034KIZgYhAGuIGCu_jok4",
    "shop.example.com": {
      "port_token": "3O9YHawMDXLpKb-FVjQ1_qSS9R9wbwb0TWbUxLvqAAI",
      "extra_stuff": 34
    }
  }
}</pre>

<p>Below is a sample respone from an Old OP that identifies the user with the same subject identifier to all RPs.</p>

<pre>HTTP/1.1 200 OK
Content-Type: application/json

{
  "public": "4vkhLCrpTm1Gidg5H6MZfwt6HnzNDTZJB8cNMFD0834"
}</pre>

<p>If "<tt>pairwise</tt>" and "<tt>public</tt>" members are both present, the port_token from "<tt>public</tt>" SHOULD only be used with RPs whose Sector Id is not present in "<tt>pairwise</tt>".</p>

<p>An Old OP will generally know via other processes when and to whom a user is porting. In such circumstances, the API SHOULD reject requests about a user not known to be in the process of porting, and SHOULD reject requests from any OP other than the known destination. Only OPs explicitly trusted by the Old OP SHOULD be allowed to use the API (eg to use the "<tt>porting_data</tt>" scope) as the API is privacy-sensitive since it allows the caller to correlate pairwise Subject Ids.</p>


<h1 id="rfc.section.4">4. <a id="aka">Indicating a previous identity: "<tt>aka</tt>" member</a></h1>

<p>This document defines the "<tt>aka</tt>" (also known as) member that a New OP can include in an id_token delivered to an RP to indicate another identifier for the same user. The "<tt>aka</tt>" value is a JSON object with "<tt>port_token</tt>" and "<tt>iss</tt>" members, both with string values. "<tt>iss</tt>" identifies the Old OP, while "<tt>port_token</tt>" can be used at the Old OP's Porting check API [<a href="#port_check">section 5</a>] to confirm the port and obtain the Old OP's "<tt>sub</tt>" value for the user.</p>

<p>On receiving an id_token, a RP uses "<tt>iss</tt>" & "<tt>sub</tt>" to lookup the user's account in the RP's systems. If no such account is found, but the id_token contains an "<tt>aka</tt>" member, then the RP MUST process the "<tt>aka</tt>" value to determine if the RP knows the user via federation from an Old OP. Processing involves: calling the Porting check API for the Old OP with the given "<tt>port_token</tt>"; looking for an existing account at the RP matching the resulting Old OP's "<tt>sub</tt>" value; and linking that account to the New OP's "<tt>sub</tt>" value.</p>

<p>Below is a sample payload of an id_token.</p>

<pre>{
  "iss": "https://newop.example.net/",
  "sub": "JRrr09BzhZ6BJ9t0yD8DzyZjg7ziB3a40jeoUvZkUgw",
  "aud": "s6BhdRkqt3"
  "iat": 1471238000
  "exp": 1471238300,
  "at_hash": "mrxuWF8TvnILKqFpZgEYCQ",
  "aka": {
    "iss": "https://oldop.example.net/",
    "port_token": "c8nGXBcdTK0YYeVf3Bnjd4NuLIU_ICR3qUMkrzO_4ho"
  }
}</pre>


<h1 id="rfc.section.5">5. <a id="port_check">Porting check API</a></h1>

<p>The Porting check API is offered by an Old OP to RPs so they can confirm that a user did port to a specific New OP. The API converts a port_token to a Subject Id ("<tt>sub</tt>" value), which the RP needs to identify the ported user's existing account at the RP.</p>

<p>An RP makes a <tt>GET</tt> request to the API endpoint (from the "<tt>port_check_endpoint</tt>" member in the Old OP discovery metadata), adding query parameters identifying the New OP ("<tt>iss</tt>"), the RP ("<tt>sector_id</tt>"), and a porting action ("<tt>port_token</tt>"). A successful response (eg with a status code of 200) conveys a JSON object with a "<tt>sub</tt>" member and optionally an "<tt>aka</tt>" member. A successful response means the identified user at the Old OP has ported to the given New OP. Otherwise, if the port_token is invalid, or it is not associated with the given sector_id, or the port was not to the identified New OP, then an unsuccessful HTTP response status code MUST be returned (eg 400 Bad request).</p>

<p>The "<tt>sector_id</tt>" value is ignored by the Old OP if it uses a single public "<tt>sub</tt>" value for a user at any RP, but an RP cannot assume that is the case so the RP MUST still include the parameter in the request.</p>

<p>The port_token effectively acts as a bearer token for authorizing the request to the Porting check API. No further access control is required. That is, the API is not expected to be further protected by, say, an OAuth 2.0 flow to authenticate the RP.</p>

<p>The "<tt>aka</tt>" value has the same syntax and semantics as when it appears in an id_token [<a href="#aka">section 4</a>]. In a porting check response it indicates that the user ported to the Old OP from an even older OP. When the "<tt>sub</tt>" value (in combination with the Old OP's "<tt>iss</tt>" value) is not recognized by the RP, the RP SHOULD process the "<tt>aka</tt>" value to determine if the RP knows the user via federation from the even older OP. An RP MUST NOT recursively follow "<tt>aka</tt>" values indefinitely; a limit of a few recursions is likely to be sufficient for valid use cases.</p>

<p>Below is a sample request with a successful response. [The long request line is wrapped over 5 lines just for display purposes.] The RP would lookup <tt>sub=kY6N6&#x2026;</tt> & <tt>iss=https://oldop&#x2026;</tt> in its systems to find the existing account that had ported to the New OP.</p>

<pre>GET /connect/port_check?
  iss=https://newop.example.net/&
  sector_id=rp.example.org&
  port_token=c8nGXBcdTK0YYeVf3Bnjd4NuLIU_ICR3qUMkrzO_4ho
   HTTP/1.1
Host: oldop.example.net
</pre>

<pre>HTTP/1.1 200 OK
Content-Type: application/json

{
  "sub": "kY6N64H86MBTho9I-JlRnpvaLOwAb5m3yYta6pq1VFM"
}</pre>

<p>Below is a sample resonse that identifies a previous port. If the RP did not recognize <tt>sub=kY6N6&#x2026;</tt> & <tt>iss=https://oldop&#x2026;</tt> it can process the "<tt>aka</tt>" value to see if that leads back to an account known to the RP.</p>

<pre>HTTP/1.1 200 OK
Content-Type: application/json

{
  "sub": "kY6N64H86MBTho9I-JlRnpvaLOwAb5m3yYta6pq1VFM",
  "aka": {
    "iss": "https://anotherop.example.net/",
    "port_token": "QSfmp2V30cNqKShhyOjQEwDaQzLNhji5w5nD-EZq3ok"
  }
}</pre>

<p>Below is a sample unsuccessful response (which happens to use the Problem Details format specified in [<a href="#RFC7807">RFC7807</a>]).</p>

<pre>HTTP/1.1 400 Bad request
Content-Type: application/problem+json

{
  "type": "https://oldop.example.net/probs/porting_rp",
  "title": "Port_token presented by the wrong RP"
}</pre>


<h1 id="rfc.section.6">6. <a id="security_considerations">Security Considerations</a></h1>

<p>Attackers goal: get access to the victims RP accounts either by taking over the respective OP or RP account.  This section analysis the protocol flow along the lines of the migration process for threat vectors. Note: this list is a starting point and not meant to be comprehensive (yet).</p>

<h2 id="rfc.section.6.1">6.1. Impersonation of legitimate user's old OP account</h2>
<p>If the attacker can get hold of the authorization grant issued by the old OP for access to the account migration API, the attacker can impersonate the legitimate user in the migration process at the new OP.  </p>

<ul>
  <li>Prevent leakage of authorization grant at old OP (mix up, referrer header, log files, ...)</li>
  <li>Prevent injection at new OP (TBD - state, id_token, nonce, pkce, token binding, ...)</li>
  <li>Audience/purpose restrict access tokens on old OP side, in order to strictly limit access tokens, which can be used for account migration. Combined with a strict limitation (whitelisting) of all clients, which are allowed to request access tokens for account migration shall reduce attack surface.</li>
  <li>idea: use migration specific credential, which is directly entered during migration process at new OP (in turn potentially imposes the risk of men in the middle/phishing)</li>
  <li>old OP should notify user of account migration process via different communication channel.</li>
</ul>

<h2 id="rfc.section.6.2">6.2. Trick user into login with attacker's account</h2>
<p>The attacker could try to cause the legitimate user to login with her new OP with the attacker's account with this new OP. That would cause the new OP to associate the migration data with the attackers account, which in turn allows the attacker to migrate all RP accounts of the legitimate user to his account with the new OP.  </p>

<ul>
  <li>Prevent XSRF, session fixation</li>
  <li>Explicitly show user identity data of both user accounts, which she is about to tie together and ask for consent.</li>
  <li>Send information back to old OP about new OP identity? It could inform user of the migration, which just happened (too).</li>
</ul>

<h2 id="rfc.section.6.3">6.3. Impersonate old OP/replay migration data</h2>
<p id="rfc.section.6.3.p.1">If the attacker gets hold of valid migration data of an account at OP1 and "migrates" the account data to his account with OP1. Scenario: The attacker logs into the new OP with her OP account and sends the new OP to the OP under her control. This OP returns migration data obtained from a legitimate OP in a previous step.  </p>

<ul>
  <li>audience restriction: verify subject of migration data (migrated_to must be iss of new OP)</li>
  <li>detection: use discovery to match account migration endpoint to endpoint of legitimate OP</li>
</ul>

<h2 id="rfc.section.6.4">6.4. Bad OP</h2>
<p>Attacker could try to trick victim into (unintentionally) migration her OP account to a OP under the attacker's control </p>

<ul>
  <li>white listing or blacklisting of OPs for which account migration is supported</li>
  <li>clear wording and consent during release of the account migration authorization grant.</li>
  <li>no SSO and no auto consent: require re-authentication and explicit consent + specific migration credential (just to make the process less convenient and make the victim think about what she is up to.).</li>
  <li>old OP should notify user of account migration process via different communication channel.</li>
</ul>

<h2 id="rfc.section.6.5">6.5. Inject migration data during RP account migration</h2>
<p>Attacker gets hold of valid migration data and injects it into the login response. This would only work for implicit grant, not for grant type code.  </p>

<ul>
  <li>use grant type code</li>
  <li>migration data is included in ID Token, so it is signed by the new OP and cannot be modfied by the attacker.</li>
</ul>

<h2 id="rfc.section.6.6">6.6. Further Considerations</h2>
<p>Note: the new OP could also accidentally assign the account migration data to the wrong user account in its database. Care must be taken that the respective functions are thoroughly tested with every change.</p>


<h1 id="rfc.section.7">7. <a id="privacy_considerations">Privacy Considerations</a></h1>

<p>TBD</p>


<h1 id="rfc.section.8">8. <a id="IANA">IANA Considerations</a></h1>

<p>This document registers the "<tt>aka</tt>" claim in the IANA "JSON Web Token Claims" registry [<a href="#IANA.JWT.Claims">IANA.JWT.Claims</a>].</p>

<ul>
  <li>Claim Name: "<tt>aka</tt>"</li>
  <li>Claim Description: Also Known As</li>
  <li>Change Controller: OpenID Foundation MODRNA working group <<a href="https://openid.net/wg/mobile/">openid.net/wg/mobile/</a>></li>
  <li>Specification Document(s): <a href="#aka">Section 4</a> of [[ this specification ]]</li>
</ul>


<h1 id="rfc.references">9. References</h1>

<h2 id="rfc.references.1">9.1. Normative References</h2>

<table>
  <tbody>
    <tr>
      <td class="reference">
        <b id="OpenID.Core">[OIDC]</b>
      </td>
      <td class="top"><a title="Nomura Research Institute, Ltd.">Sakimura, N.</a>, <a title="Ping Identity">Bradley, J.</a>, <a title="Microsoft">Jones, M.</a>, <a title="Google">de Medeiros, B.</a>, <a title="Salesforce">Mortimore, C.</a> and <a title="Illumila">E. Jay</a>, "<a href="https://openid.net/specs/openid-connect-core-1_0.html">OpenID Connect Core 1.0</a>", December 2013.</td>
    </tr>
    <tr>
      <td class="reference">
        <b id="OpenID.Discovery">[Discovery]</b>
      </td>
      <td class="top"><a title="Nomura Research Institute, Ltd.">Sakimura, N.</a>, <a title="Ping Identity">Bradley, J.</a>, <a title="Microsoft">Jones, M.</a> and <a title="Illumila">E. Jay</a>, "<a href="https://openid.net/specs/openid-connect-discovery-1_0.html">OpenID Connect Discovery 1.0</a>", August 2015.</td>
    </tr>
    <tr>
      <td class="reference">
        <b id="RFC2119">[RFC2119]</b>
      </td>
      <td class="top"><a>Bradner, S.</a>, "<a href="https://tools.ietf.org/html/rfc2119">Key words for use in RFCs to Indicate Requirement Levels</a>", BCP 14, RFC 2119, DOI 10.17487/RFC2119, March 1997.</td>
    </tr>
    <tr>
      <td class="reference">
        <b id="RFC7230">[RFC7230]</b>
      </td>
      <td class="top">Fielding, R. and Reschke, J., "<a href="https://tools.ietf.org/html/rfc7230">Hypertext Transfer Protocol (HTTP/1.1): Message Syntax and Routing</a>", RFC 7230, June 2014.</td>
    </tr>
    <tr>
      <td class="reference">
        <b id="RFC5246">[RFC5246]</b>
      </td>
      <td class="top"><a>Dierks, T.</a> and <a>E. Rescorla</a>, "<a href="https://tools.ietf.org/html/rfc5246">The Transport Layer Security (TLS) Protocol Version 1.2</a>", RFC 5246, DOI 10.17487/RFC5246, August 2008.</td>
    </tr>
    <tr>
      <td class="reference">
        <b id="RFC6749">[RFC6749]</b>
      </td>
      <td class="top"><a>Hardt, D.</a>, "<a href="https://tools.ietf.org/html/rfc6749">The OAuth 2.0 Authorization Framework</a>", RFC 6749, DOI 10.17487/RFC6749, October 2012.</td>
    </tr>
    <tr>
      <td class="reference">
        <b id="RFC6750">[RFC6750]</b>
      </td>
      <td class="top"><a>Jones, M.</a> and <a>D. Hardt</a>, "<a href="https://tools.ietf.org/html/rfc6750">The OAuth 2.0 Authorization Framework: Bearer Token Usage</a>", RFC 6750, DOI 10.17487/RFC6750, October 2012.</td>
    </tr>
    <tr>
      <td class="reference">
        <b id="IANA.JWT.Claims">[IANA.JWT.Claims]</b>
      </td>
      <td class="top">IANA, "JSON Web Token Claims", <<a href="https://www.iana.org/assignments/jwt/">https://www.iana.org/assignments/jwt/</a>></td>
    </tr>

  </tbody>
</table>


<h2 id="rfc.references.2">9.2. Informative References</h2>

<table>
  <tbody>
    <tr>
      <td class="reference">
        <b id="MobileConnect">[MobileConnect]</b>
      </td>
      <td class="top">GSMA, "Mobile Connect", <<a href="https://mobileconnect.io/">mobileconnect.io</a>>.</td>
    </tr>
    <tr>
      <td class="reference">
        <b id="OpenID20.Migration">[OpenID20.Migration]</b>
      </td>
      <td class="top"><a title="Nomura Research Institute, Ltd.">Sakimura, N.</a>, <a title="Ping Identity">Bradley, J.</a>, <a title="Google">Agarwal, N.</a> and <a title="Illumila">E. Jay</a>, "<a href="https://openid.net/specs/openid-connect-migration-1_0.html">OpenID 2.0 to OpenID Connect Migration 1.0</a>", April 2015.</td>
    </tr>
  </tbody>
</table>


<h1 id="rfc.appendix.A">Appendix A. <a id="example">Full example</a></h1>

<p>This example involves a user (Alice), an RP, and 3 OPs (OP1, OP2, and OP3). Alice originally federated to the RP via OP1; ported to OP2; then (shortly afterwards) ported to OP3. When Alice logs in to RP via OP3 for the first time she is not recognized directly from OP3's id_token. But by following the "<tt>aka</tt>" trail, the RP automatically establishes the link to an id from OP1 that the RP recognizes as Alice.</p>

<p>Many details are elided. Dotted arrows represent HTTP redirects.</p>

<img alt="Porting example sequence diagram" src=""/>

<h1 id="rfc.appendix.B">Appendix B. <a id="Acknowledgements">Acknowledgements</a></h1>

<p>The following people in particular have contributed to the development of this specification.</p>

<ul class="empty">
  <li>Torsten Lodderstedt (t.lodderstedt@telekom.de), Deutsche Telekom</li>
  <li>Jörg Connotte (j.connotte@telekom.de), Deutsche Telekom</li>
  <li>John Bradley (ve7jtb@ve7jtb.com), Ping Identity</li>
  <li>Axel Nennker (Axel.Nennker@telekom.de), Deutsche Telekom</li>
  <li>Arne Georg Gleditsch (argggh@telenordigital.com), Telenor Digital AS</li>
</ul>


<h1 id="rfc.appendix.C">Appendix C. <a id="notices">Notices</a></h1>

<p>Copyright (c) 2016 The OpenID Foundation.</p>

<p>The OpenID Foundation (OIDF) grants to any Contributor, developer, implementer, or other interested party a non-exclusive, royalty free, worldwide copyright license to reproduce, prepare derivative works from, distribute, perform and display, this Implementers Draft or Final Specification solely for the purposes of (i) developing specifications, and (ii) implementing Implementers Drafts and Final Specifications based on such documents, provided that attribution be made to the OIDF as the source of the material, but that such attribution does not indicate an endorsement by the OIDF.</p>

<p>The technology described in this specification was made available from contributions from various sources, including members of the OpenID Foundation and others. Although the OpenID Foundation has taken steps to help ensure that the technology is available for distribution, it takes no position regarding the validity or scope of any intellectual property or other rights that might be claimed to pertain to the implementation or use of the technology described in this specification or the extent to which any license under such rights might or might not be available; neither does it represent that it has made any independent effort to identify any such rights. The OpenID Foundation and the contributors to this specification make no (and hereby expressly disclaim any) warranties (express, implied, or otherwise), including implied warranties of merchantability, non-infringement, fitness for a particular purpose, or title, related to this specification, and the entire risk as to implementing this specification is assumed by the implementer. The OpenID Intellectual Property Rights policy requires contributors to offer a patent promise not to assert certain patent claims against other contributors and against implementers. The OpenID Foundation invites any interested party to bring to its attention any copyrights, patents, patent applications, or other proprietary rights that may cover technology that may be required to practice this specification.</p>


<h1 id="rfc.appendix.D">Appendix D. <a id="history">Document history</a></h1>

<p>[[ To be removed from the final specification ]]</p>

<p>-00 </p>

<ul>
  <li>Initial draft</li>
  <li>Proposed as an alternative approach draft-account-migration-02</li>
</ul>


<h1 id="rfc.authors"><a id="rfc.authors">Author's Address</a></h1>

<div class="avoidbreak">
  <address class="vcard">
        <span class="vcardline">
          <span class="fn">James Manger</span> 
        </span>
        <span class="org vcardline">Telstra</span>
        <span class="adr">
          
          <span class="vcardline">
                <span class="locality"></span> 
                <span class="region"></span>
                <span class="code"></span>
          </span>
          <span class="country-name vcardline"></span>
        </span>
        <span class="vcardline">EMail: <a href="mailto:james.h.manger@team.telstra.com">james.h.manger@team.telstra.com</a></span>

  </address>
</div>

</body>
</html>