code.dart 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. import 'package:ente_auth/utils/totp_util.dart';
  2. class Code {
  3. static const defaultDigits = 6;
  4. static const defaultPeriod = 30;
  5. int? generatedID;
  6. final String account;
  7. final String issuer;
  8. final int digits;
  9. final int period;
  10. final String secret;
  11. final Algorithm algorithm;
  12. final Type type;
  13. final String rawData;
  14. bool? hasSynced;
  15. Code(
  16. this.account,
  17. this.issuer,
  18. this.digits,
  19. this.period,
  20. this.secret,
  21. this.algorithm,
  22. this.type,
  23. this.rawData, {
  24. this.generatedID,
  25. });
  26. Code copyWith({
  27. String? account,
  28. String? issuer,
  29. int? digits,
  30. int? period,
  31. String? secret,
  32. Algorithm? algorithm,
  33. Type? type,
  34. }) {
  35. final String updateAccount = account ?? this.account;
  36. final String updateIssuer = issuer ?? this.issuer;
  37. final int updatedDigits = digits ?? this.digits;
  38. final int updatePeriod = period ?? this.period;
  39. final String updatedSecret = secret ?? this.secret;
  40. final Algorithm updatedAlgo = algorithm ?? this.algorithm;
  41. final Type updatedType = type ?? this.type;
  42. return Code(
  43. updateAccount,
  44. updateIssuer,
  45. updatedDigits,
  46. updatePeriod,
  47. updatedSecret,
  48. updatedAlgo,
  49. updatedType,
  50. "otpauth://${updatedType.name}/" +
  51. updateIssuer +
  52. ":" +
  53. updateAccount +
  54. "?algorithm=${updatedAlgo.name}&digits=$updatedDigits&issuer=" +
  55. updateIssuer +
  56. "&period=$updatePeriod&secret=" +
  57. updatedSecret,
  58. generatedID: generatedID,
  59. );
  60. }
  61. static Code fromAccountAndSecret(
  62. String account,
  63. String issuer,
  64. String secret,
  65. ) {
  66. return Code(
  67. account,
  68. issuer,
  69. defaultDigits,
  70. defaultPeriod,
  71. secret,
  72. Algorithm.sha1,
  73. Type.totp,
  74. "otpauth://totp/" +
  75. issuer +
  76. ":" +
  77. account +
  78. "?algorithm=SHA1&digits=6&issuer=" +
  79. issuer +
  80. "&period=30&secret=" +
  81. secret,
  82. );
  83. }
  84. static Code fromRawData(String rawData) {
  85. Uri uri = Uri.parse(rawData);
  86. return Code(
  87. _getAccount(uri),
  88. _getIssuer(uri),
  89. _getDigits(uri),
  90. _getPeriod(uri),
  91. getSanitizedSecret(uri.queryParameters['secret']!),
  92. _getAlgorithm(uri),
  93. _getType(uri),
  94. rawData,
  95. );
  96. }
  97. static String _getAccount(Uri uri) {
  98. try {
  99. String path = Uri.decodeComponent(uri.path);
  100. if (path.startsWith("/")) {
  101. path = path.substring(1, path.length);
  102. }
  103. // Parse account name from documented auth URI
  104. // otpauth://totp/ACCOUNT?secret=SUPERSECRET&issuer=SERVICE
  105. if (uri.queryParameters.containsKey("issuer") && !path.contains(":")) {
  106. return path;
  107. }
  108. return path.split(':')[1];
  109. } catch (e) {
  110. return "";
  111. }
  112. }
  113. static String _getIssuer(Uri uri) {
  114. try {
  115. if (uri.queryParameters.containsKey("issuer")) {
  116. String issuerName = uri.queryParameters['issuer']!;
  117. // Handle issuer name with period
  118. // See https://github.com/ente-io/auth/pull/77
  119. if (issuerName.contains("period=")) {
  120. return issuerName.substring(0, issuerName.indexOf("period="));
  121. }
  122. return issuerName;
  123. }
  124. final String path = Uri.decodeComponent(uri.path);
  125. return path.split(':')[0].substring(1);
  126. } catch (e) {
  127. return "";
  128. }
  129. }
  130. static int _getDigits(Uri uri) {
  131. try {
  132. return int.parse(uri.queryParameters['digits']!);
  133. } catch (e) {
  134. return defaultDigits;
  135. }
  136. }
  137. static int _getPeriod(Uri uri) {
  138. try {
  139. return int.parse(uri.queryParameters['period']!);
  140. } catch (e) {
  141. return defaultPeriod;
  142. }
  143. }
  144. static Algorithm _getAlgorithm(Uri uri) {
  145. try {
  146. final algorithm =
  147. uri.queryParameters['algorithm'].toString().toLowerCase();
  148. if (algorithm == "sha256") {
  149. return Algorithm.sha256;
  150. } else if (algorithm == "sha512") {
  151. return Algorithm.sha512;
  152. }
  153. } catch (e) {
  154. // nothing
  155. }
  156. return Algorithm.sha1;
  157. }
  158. static Type _getType(Uri uri) {
  159. if (uri.host == "totp") {
  160. return Type.totp;
  161. } else if (uri.host == "hotp") {
  162. return Type.hotp;
  163. }
  164. throw UnsupportedError("Unsupported format with host ${uri.host}");
  165. }
  166. @override
  167. bool operator ==(Object other) {
  168. if (identical(this, other)) return true;
  169. return other is Code &&
  170. other.account == account &&
  171. other.issuer == issuer &&
  172. other.digits == digits &&
  173. other.period == period &&
  174. other.secret == secret &&
  175. other.type == type &&
  176. other.rawData == rawData;
  177. }
  178. @override
  179. int get hashCode {
  180. return account.hashCode ^
  181. issuer.hashCode ^
  182. digits.hashCode ^
  183. period.hashCode ^
  184. secret.hashCode ^
  185. type.hashCode ^
  186. rawData.hashCode;
  187. }
  188. }
  189. enum Type {
  190. totp,
  191. hotp,
  192. }
  193. enum Algorithm {
  194. sha1,
  195. sha256,
  196. sha512,
  197. }