serviceProviders/openpgp.js

  1. /*
  2. Copyright 2024 Yarmo Mackenbach
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. /**
  14. * OpenPGP service provider ({@link https://docs.keyoxide.org/service-providers/openpgp/|Keyoxide docs})
  15. * @module serviceProviders/openpgp
  16. * @example
  17. * import { ServiceProviderDefinitions } from 'doipjs';
  18. * const sp = ServiceProviderDefinitions.data.openpgp.processURI('openpgp4fpr:ABC123DEF456');
  19. */
  20. import * as E from '../enums.js'
  21. import { ServiceProvider } from '../serviceProvider.js'
  22. export const reURI = /^(.*)/
  23. const reURIHkp = /^openpgp4fpr:(?:0x)?([a-zA-Z0-9.\-_]*)/
  24. const reURIWkdDirect = /^https:\/\/(.*)\/.well-known\/openpgpkey\/hu\/([a-zA-Z0-9]*)(?:\?l=(.*))?/
  25. const reURIWkdAdvanced = /^https:\/\/(openpgpkey.*)\/.well-known\/openpgpkey\/(.*)\/hu\/([a-zA-Z0-9]*)(?:\?l=(.*))?/
  26. /**
  27. * @function
  28. * @param {string} uri - Claim URI to process
  29. * @returns {ServiceProvider} The service provider information based on the claim URI
  30. */
  31. export function processURI (uri) {
  32. let reURI = null
  33. let mode = null
  34. let match = null
  35. if (reURIHkp.test(uri)) {
  36. reURI = reURIHkp
  37. mode = E.OpenPgpQueryProtocol.HKP
  38. match = uri.match(reURI)
  39. }
  40. if (!mode && reURIWkdAdvanced.test(uri)) {
  41. reURI = reURIWkdAdvanced
  42. mode = E.OpenPgpQueryProtocol.WKD
  43. match = uri.match(reURI)
  44. }
  45. if (!mode && reURIWkdDirect.test(uri)) {
  46. reURI = reURIWkdDirect
  47. mode = E.OpenPgpQueryProtocol.WKD
  48. match = uri.match(reURI)
  49. }
  50. let output = null
  51. switch (mode) {
  52. case E.OpenPgpQueryProtocol.HKP:
  53. output = new ServiceProvider({
  54. about: {
  55. id: 'openpgp',
  56. name: 'OpenPGP'
  57. },
  58. profile: {
  59. display: `openpgp4fpr:${match[1]}`,
  60. uri: `https://keys.openpgp.org/search?q=${match[1]}`,
  61. qr: null
  62. },
  63. claim: {
  64. uriRegularExpression: reURI.toString(),
  65. uriIsAmbiguous: false
  66. },
  67. proof: {
  68. request: {
  69. uri: `https://keys.openpgp.org/vks/v1/by-fingerprint/${match[1].toUpperCase()}`,
  70. fetcher: E.Fetcher.OPENPGP,
  71. accessRestriction: E.ProofAccessRestriction.NONE,
  72. data: {
  73. url: `https://keys.openpgp.org/vks/v1/by-fingerprint/${match[1].toUpperCase()}`,
  74. protocol: E.OpenPgpQueryProtocol.HKP
  75. }
  76. },
  77. response: {
  78. format: E.ProofFormat.JSON
  79. },
  80. target: [{
  81. format: E.ClaimFormat.URI,
  82. encoding: E.EntityEncodingFormat.PLAIN,
  83. relation: E.ClaimRelation.EQUALS,
  84. path: ['notations', 'proof@ariadne.id']
  85. }]
  86. }
  87. })
  88. break
  89. case E.OpenPgpQueryProtocol.WKD:
  90. output = new ServiceProvider({
  91. about: {
  92. id: 'openpgp',
  93. name: 'OpenPGP'
  94. },
  95. profile: {
  96. display: 'unknown fingerprint',
  97. uri,
  98. qr: null
  99. },
  100. claim: {
  101. uriRegularExpression: reURI.toString(),
  102. uriIsAmbiguous: false
  103. },
  104. proof: {
  105. request: {
  106. uri,
  107. fetcher: E.Fetcher.OPENPGP,
  108. accessRestriction: E.ProofAccessRestriction.NONE,
  109. data: {
  110. url: uri,
  111. protocol: E.OpenPgpQueryProtocol.WKD
  112. }
  113. },
  114. response: {
  115. format: E.ProofFormat.JSON
  116. },
  117. target: [{
  118. format: E.ClaimFormat.URI,
  119. encoding: E.EntityEncodingFormat.PLAIN,
  120. relation: E.ClaimRelation.EQUALS,
  121. path: ['notations', 'proof@ariadne.id']
  122. }]
  123. }
  124. })
  125. break
  126. }
  127. return output
  128. }
  129. export const tests = [
  130. {
  131. uri: 'openpgp4fpr:123456789',
  132. shouldMatch: true
  133. },
  134. {
  135. uri: 'openpgp4fpr:abcdef123',
  136. shouldMatch: true
  137. },
  138. {
  139. uri: 'https://openpgpkey.domain.tld/.well-known/openpgpkey/domain.tld/hu/123abc456def?l=name',
  140. shouldMatch: true
  141. },
  142. {
  143. uri: 'https://openpgpkey.domain.tld/.well-known/openpgpkey/domain.tld/hu/123abc456def',
  144. shouldMatch: true
  145. },
  146. {
  147. uri: 'https://domain.tld/.well-known/openpgpkey/hu/123abc456def?l=name',
  148. shouldMatch: true
  149. },
  150. {
  151. uri: 'https://domain.tld/.well-known/openpgpkey/hu/123abc456def',
  152. shouldMatch: true
  153. },
  154. // The following will not pass .processURI, but reURI currently accepts anything
  155. {
  156. uri: 'https://domain.tld',
  157. shouldMatch: true
  158. },
  159. {
  160. uri: 'https://openpgpkey.domain.tld/.well-known/openpgpkey/hu/123abc456def?l=name',
  161. shouldMatch: true
  162. },
  163. {
  164. uri: 'https://domain.tld/.well-known/openpgpkey/123abc456def?l=name',
  165. shouldMatch: true
  166. }
  167. ]