Source: parse-siwe.js

  1. /**
  2. * @file src/parse-siwe.js
  3. * @author Lowell D. Thomas <ldt@sabnf.com>
  4. */
  5. import { Parser } from './parser.js';
  6. import { default as Grammar } from './grammar.js';
  7. import { cb } from './callbacks.js';
  8. import { isERC55, toERC55 } from './keccak256.js';
  9. export { parseSiweMessage, isUri, siweObjectToString };
  10. /**
  11. * Parses an [ERC-4361: Sign-In with Ethereum](https://eips.ethereum.org/EIPS/eip-4361)
  12. * message to an object with the message components.
  13. *
  14. * @param {string} msg the ERC-4361 message
  15. * @param {string} erc55 controls [ERC-55](https://eips.ethereum.org/EIPS/eip-55) processing of the message address<br>
  16. * - 'validate' - (default) parser fails if address is not in ERC-55 encoding
  17. * - 'convert' - parser will convert the address to ERC-55 encoding
  18. * - 'other' - (actually, any value other than 'validate', 'convert' or undefined) parser ignores ERC-55 encoding
  19. * @returns An siwe message object or throws exception with instructive message on format error.<br>
  20. * e.g. message
  21. * ````
  22. example.com:80 wants you to sign in with your Ethereum account:
  23. 0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2
  24. Valid statement
  25. URI: https://example.com/login
  26. Version: 1
  27. Chain ID: 123456789
  28. Nonce: 32891756
  29. Issued At: 2021-09-30T16:25:24Z
  30. Request ID: someRequestId
  31. Resources:
  32. - ftp://myftpsite.com/
  33. - https://example.com/my-web2-claim.json
  34. * ````
  35. * returns object, obj:
  36. * ````
  37. obj.scheme = undefined
  38. obj.domain = 'example.com:80'
  39. obj.address = '0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2'
  40. obj.statement = 'Valid statement'
  41. obj.uri = 'https://example.com/login'
  42. obj.version = 1
  43. obj.chainId = 123456789
  44. obj.nonce = '32891756'
  45. obj.issuedAt = '2021-09-30T16:25:24Z'
  46. obj.expirationTime = undefined
  47. obj.notBefore = undefined
  48. obj.requestId = 'someRequestId'
  49. obj.resources = [ 'ftp://myftpsite.com/', 'https://example.com/my-web2-claim.json' ]
  50. * ````
  51. */
  52. function parseSiweMessage(msg, erc55 = 'validate') {
  53. const fname = 'parseSiweMessage: ';
  54. if (typeof msg !== 'string') {
  55. throw new Error(`${fname} invalid input msg: message must be of type string`);
  56. }
  57. const p = new Parser();
  58. const g = new Grammar();
  59. // set the parser's callback functions
  60. p.callbacks['ffscheme'] = cb.ffscheme;
  61. p.callbacks['fdomain'] = cb.fdomain;
  62. p.callbacks['faddress'] = cb.faddress;
  63. p.callbacks['fstatement'] = cb.fstatement;
  64. p.callbacks['furi'] = cb.furi;
  65. p.callbacks['fnonce'] = cb.fnonce;
  66. p.callbacks['fversion'] = cb.fversion;
  67. p.callbacks['fchain-id'] = cb.fchainId;
  68. p.callbacks['fissued-at'] = cb.fissuedAt;
  69. p.callbacks['fexpiration-time'] = cb.fexpirationTime;
  70. p.callbacks['fnot-before'] = cb.fnotBefore;
  71. p.callbacks['frequest-id'] = cb.frequestId;
  72. p.callbacks['fresources'] = cb.fresources;
  73. p.callbacks['fresource'] = cb.fresource;
  74. p.callbacks['empty-statement'] = cb.emptyStatement;
  75. p.callbacks['no-statement'] = cb.noStatement;
  76. p.callbacks['actual-statement'] = cb.actualStatment;
  77. p.callbacks['pre-uri'] = cb.preUri;
  78. p.callbacks['pre-version'] = cb.preVersion;
  79. p.callbacks['pre-chain-id'] = cb.preChainId;
  80. p.callbacks['pre-nonce'] = cb.preNonce;
  81. p.callbacks['pre-issued-at'] = cb.preIssuedAt;
  82. // URI callbacks
  83. p.callbacks['uri'] = cb.URI;
  84. p.callbacks['scheme'] = cb.scheme;
  85. p.callbacks['userinfo-at'] = cb.userinfo;
  86. p.callbacks['host'] = cb.host;
  87. p.callbacks['IP-literal'] = cb.ipLiteral;
  88. p.callbacks['port'] = cb.port;
  89. p.callbacks['path-abempty'] = cb.pathAbempty;
  90. p.callbacks['path-absolute'] = cb.pathAbsolute;
  91. p.callbacks['path-rootless'] = cb.pathRootless;
  92. p.callbacks['path-empty'] = cb.pathEmpty;
  93. p.callbacks['query'] = cb.query;
  94. p.callbacks['fragment'] = cb.fragment;
  95. p.callbacks['IPv4address'] = cb.ipv4;
  96. p.callbacks['nodcolon'] = cb.nodcolon;
  97. p.callbacks['dcolon'] = cb.dcolon;
  98. p.callbacks['h16'] = cb.h16;
  99. p.callbacks['h16c'] = cb.h16;
  100. p.callbacks['h16n'] = cb.h16;
  101. p.callbacks['h16cn'] = cb.h16;
  102. p.callbacks['dec-octet'] = cb.decOctet;
  103. p.callbacks['dec-digit'] = cb.decDigit;
  104. // BEGIN: some parser helper functions
  105. function validateDateTime(time, name = 'unknown') {
  106. const ret = p.parse(g, 'date-time', time);
  107. if (!ret.success || isNaN(Date.parse(time))) {
  108. throw new Error(`${fname}invalid date time: ${name} not date time string format: ${time}`);
  109. }
  110. }
  111. function validInt(i) {
  112. const valid = parseInt(i, 10);
  113. if (isNaN(valid)) {
  114. throw new Error(`${fname}invalid integer: not a number: ${i}`);
  115. } else if (valid === Infinity) {
  116. throw new Error(`${fname}invalid integer: Infinity: ${i}`);
  117. }
  118. return valid;
  119. }
  120. // Validates an RFC 3986 URI.
  121. // returns true if the URI is valid, false otherwise
  122. function isUri(URI) {
  123. ret = p.parse(g, 'uri', URI, {});
  124. return ret.success;
  125. }
  126. // END: some parser helper functions
  127. // first pass parse of the message
  128. // capture all the message parts, then validate them one-by-one later
  129. let data = { error: false };
  130. let ret;
  131. ret = p.parse(g, 'siwe-first-pass', msg, data);
  132. if (data.error) {
  133. throw new Error(`${fname}invalid siwe message: ${data.error}`);
  134. }
  135. if (!ret.success) {
  136. throw new Error(
  137. `${fname}invalid siwe message: carefully check message syntax, especially after required "Issued At: "\n${JSON.stringify(
  138. ret
  139. )}`
  140. );
  141. }
  142. if (data.fscheme) {
  143. // validate the scheme
  144. ret = p.parse(g, 'scheme', data.fscheme, {});
  145. if (!ret.success) {
  146. throw new Error(`${fname}invalid scheme: ${data.fscheme}`);
  147. }
  148. }
  149. // validate the domain
  150. ret = p.parse(g, 'authority', data.fdomain, {});
  151. if (!ret.success) {
  152. throw new Error(`${fname}invalid domain: ${data.fdomain}`);
  153. }
  154. // validate the address
  155. ret = p.parse(g, 'address', data.faddress);
  156. if (!ret.success) {
  157. throw new Error(`${fname}invalid address: ${data.faddress}`);
  158. }
  159. if (erc55 == 'validate') {
  160. // address must be ERC55 format
  161. if (!isERC55(data.faddress)) {
  162. throw new Error(
  163. `${fname}invalid ERC-55 format address: 'validate' specified, MUST be ERC55-conformant: ${data.faddress}`
  164. );
  165. }
  166. } else if (erc55 === 'convert') {
  167. // convert address to ERC55 format
  168. data.faddress = toERC55(data.faddress);
  169. }
  170. // else for any other value of erc55 no further action on the address is taken
  171. // validate the statement
  172. if (data.fstatement !== undefined && data.fstatement !== '') {
  173. ret = p.parse(g, 'statement', data.fstatement);
  174. if (!ret.success) {
  175. throw new Error(`${fname}invalid statement: ${data.fstatement}`);
  176. }
  177. }
  178. // validate the URI
  179. if (!isUri(data.furi)) {
  180. throw new Error(`${fname}invalid URI: ${data.furi}`);
  181. }
  182. // validate the version
  183. ret = p.parse(g, 'version', data.fversion);
  184. if (!ret.success) {
  185. throw new Error(`${fname}invalid Version: ${data.fversion}`);
  186. }
  187. data.fversion = validInt(data.fversion);
  188. // validate the chain-id
  189. ret = p.parse(g, 'chain-id', data.fchainId);
  190. if (!ret.success) {
  191. throw new Error(`${fname}invalid Chain ID: ${data.fchainId}`);
  192. }
  193. data.fchainId = validInt(data.fchainId);
  194. // validate nonce
  195. ret = p.parse(g, 'nonce', data.fnonce);
  196. if (!ret.success) {
  197. throw new Error(`${fname}invalid Nonce: ${data.fnonce}`);
  198. }
  199. // validate the date times
  200. validateDateTime(data.fissuedAt, 'Issued At');
  201. if (data.fexpirationTime) {
  202. validateDateTime(data.fexpirationTime, 'Expiration Time');
  203. }
  204. if (data.fnotBefore) {
  205. validateDateTime(data.fnotBefore, 'Not Before');
  206. }
  207. // validate request-id
  208. if (data.frequestId !== undefined && data.frequestId !== '') {
  209. ret = p.parse(g, 'request-id', data.frequestId);
  210. if (!ret.success) {
  211. throw new Error(`${fname}invalid Request ID: i${data.frequestId}`);
  212. }
  213. }
  214. // validate all resource URIs, if any
  215. if (data.fresources && data.fresources.length) {
  216. for (let i = 0; i < data.fresources.length; i++) {
  217. if (!isUri(data.fresources[i])) {
  218. throw new Error(`${fname}invalid resource URI [${i}]: ${data.fresources[i]}`);
  219. }
  220. }
  221. }
  222. // by now all first-pass values (e.g. fscheme) have been validated
  223. const o = {};
  224. o.scheme = data.fscheme;
  225. o.domain = data.fdomain;
  226. o.address = data.faddress;
  227. o.statement = data.fstatement;
  228. o.uri = data.furi;
  229. o.version = data.fversion;
  230. o.chainId = data.fchainId;
  231. o.nonce = data.fnonce;
  232. o.issuedAt = data.fissuedAt;
  233. o.expirationTime = data.fexpirationTime;
  234. o.notBefore = data.fnotBefore;
  235. o.requestId = data.frequestId;
  236. o.resources = undefined;
  237. if (data.fresources) {
  238. o.resources = [];
  239. for (let i = 0; i < data.fresources.length; i++) {
  240. o.resources.push(data.fresources[i]);
  241. }
  242. }
  243. return o;
  244. }
  245. /**
  246. * Parses a Uniform Resource Identifier (URI) defined in [RFC 3986](https://www.rfc-editor.org/rfc/rfc3986).
  247. *
  248. * @param {string} URI the URI to parse
  249. * @returns `false` if the URI is invalid, otherwise a URI object. e.g.<br>
  250. * ````
  251. * const obj = isUri('uri://user:pass@example.com:123/one/two.three?q1=a1&q2=a2#body');
  252. * ````
  253. * returns:
  254. * ````
  255. * obj.scheme = 'uri'
  256. * obj.userinfo = 'user:pass'
  257. * obj.host = 'example.com',
  258. * obj.port = 123,
  259. * obj.path = '/one/two.three',
  260. * obj.query = 'q1=a1&q2=a2',
  261. * obj.fragment = 'body'
  262. * ````
  263. */
  264. function isUri(URI) {
  265. const p = new Parser();
  266. const g = new Grammar();
  267. const uriData = {};
  268. p.callbacks['uri'] = cb.URI;
  269. p.callbacks['scheme'] = cb.scheme;
  270. p.callbacks['userinfo-at'] = cb.userinfo;
  271. p.callbacks['host'] = cb.host;
  272. p.callbacks['IP-literal'] = cb.ipLiteral;
  273. p.callbacks['port'] = cb.port;
  274. p.callbacks['path-abempty'] = cb.pathAbempty;
  275. p.callbacks['path-absolute'] = cb.pathAbsolute;
  276. p.callbacks['path-rootless'] = cb.pathRootless;
  277. p.callbacks['path-empty'] = cb.pathEmpty;
  278. p.callbacks['query'] = cb.query;
  279. p.callbacks['fragment'] = cb.fragment;
  280. p.callbacks['IPv4address'] = cb.ipv4;
  281. p.callbacks['nodcolon'] = cb.nodcolon;
  282. p.callbacks['dcolon'] = cb.dcolon;
  283. p.callbacks['h16'] = cb.h16;
  284. p.callbacks['h16c'] = cb.h16;
  285. p.callbacks['h16n'] = cb.h16;
  286. p.callbacks['h16cn'] = cb.h16;
  287. p.callbacks['dec-octet'] = cb.decOctet;
  288. p.callbacks['dec-digit'] = cb.decDigit;
  289. const ret = p.parse(g, 'uri', URI, uriData);
  290. if (!ret.success) {
  291. return false;
  292. }
  293. const uriObject = {
  294. uri: uriData['uri'],
  295. scheme: uriData['scheme'],
  296. userinfo: uriData['userinfo'],
  297. host: uriData['host'],
  298. port: uriData['port'],
  299. path: uriData['path'],
  300. query: uriData['query'],
  301. fragment: uriData['fragment'],
  302. };
  303. return uriObject;
  304. }
  305. /**
  306. * Stringify an [ERC-4361: Sign-In with Ethereum](https://eips.ethereum.org/EIPS/eip-4361) (siwe) object.
  307. * Note that this function does no validation of the input object.
  308. * It simply returns a string that includes the valid parts of the object, if any.
  309. * For validation, use {@link parseSiweMessage}.
  310. *
  311. * @param {object} o an siwe message object (see {@link parseSiweMessage} )
  312. * @returns A stringified version of the object suitable as input to {@link parseSiweMessage}.
  313. *
  314. * For example, to validate an siwe object <br>*(ignore backslash, JSDoc can't handle closed bracket character without it)*:
  315. * ````
  316. * try{
  317. * parseSiweMessage(siweObjectToString(siweObject));
  318. * console.log('siweObject is valid');
  319. * \}catch(e){
  320. * console.log('siweObject is not valid: ' + e.message);
  321. * \}
  322. * ````
  323. */
  324. function siweObjectToString(o) {
  325. let str = '';
  326. if (typeof o !== 'object') {
  327. return str;
  328. }
  329. if (o.scheme && o.scheme !== '') {
  330. str += `${o.scheme}://`;
  331. }
  332. if (o.domain && o.domain !== '') {
  333. str += o.domain;
  334. }
  335. str += ' wants you to sign in with your Ethereum account:\n';
  336. if (o.address && o.address !== '') {
  337. str += `${o.address}\n`;
  338. }
  339. str += '\n';
  340. if (o.statement !== undefined) {
  341. str += `${o.statement}\n`;
  342. }
  343. str += '\n';
  344. if (o.uri && o.uri !== '') {
  345. str += `URI: ${o.uri}\n`;
  346. }
  347. if (o['version'] && o['version'] !== '') {
  348. str += `Version: ${o['version']}\n`;
  349. }
  350. if (o['chainId'] && o['chainId'] !== '') {
  351. str += `Chain ID: ${o['chainId']}\n`;
  352. }
  353. if (o['nonce'] && o['nonce'] !== '') {
  354. str += `Nonce: ${o['nonce']}\n`;
  355. }
  356. if (o['issuedAt'] && o['issuedAt'] !== '') {
  357. str += `Issued At: ${o['issuedAt']}`;
  358. }
  359. if (o['expirationTime'] && o['expirationTime'] !== '') {
  360. str += `\nExpiration Time: ${o['expirationTime']}`;
  361. }
  362. if (o['notBefore'] && o['notBefore'] !== '') {
  363. str += `\nNot Before: ${o['notBefore']}`;
  364. }
  365. if (o['requestId'] !== undefined) {
  366. str += `\nRequest ID: ${o['requestId']}`;
  367. }
  368. if (o['resources']) {
  369. str += `\nResources:`;
  370. if (Array.isArray(o.resources)) {
  371. for (let i = 0; i < o.resources.length; i++) {
  372. str += `\n- ${o.resources[i]}`;
  373. }
  374. }
  375. }
  376. return str;
  377. }