write-tweet.ts 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. import { Component } from "@angular/core";
  2. import {
  3. IonicPage,
  4. NavController,
  5. NavParams,
  6. LoadingController,
  7. AlertController
  8. } from "ionic-angular";
  9. import {
  10. FormBuilder,
  11. Validators,
  12. FormGroup,
  13. ValidatorFn,
  14. AbstractControl
  15. } from "@angular/forms";
  16. import { TwitterApiProvider } from "../../providers/twitter-api/twitter-api";
  17. import { Storage } from "@ionic/storage";
  18. import { P2pStorageIpfsProvider } from "../../providers/p2p-storage-ipfs/p2p-storage-ipfs";
  19. import { P2pDatabaseGunProvider } from "../../providers/p2p-database-gun/p2p-database-gun";
  20. import { PgpKeyServerProvider } from "../../providers/pgp-key-server/pgp-key-server";
  21. import twittertext from "twitter-text";
  22. import { CryptoProvider } from "../../providers/crypto/crypto";
  23. import * as openpgp from 'openpgp';
  24. @IonicPage()
  25. @Component({
  26. selector: "page-write-tweet",
  27. templateUrl: "write-tweet.html"
  28. })
  29. export class WriteTweetPage {
  30. tweet: FormGroup;
  31. retweetId: string;
  32. replyToStatusId: string;
  33. retweet;
  34. replyTweet;
  35. openpgp;
  36. privateKey;
  37. publicKey;
  38. pk: any[] = [];
  39. passp = 'super long and hard to guess secret';
  40. hkp = new openpgp.HKP('https://sks-keyservers.net/');
  41. constructor(
  42. public navCtrl: NavController,
  43. public navParams: NavParams,
  44. private formBuilder: FormBuilder,
  45. private twitter: TwitterApiProvider,
  46. private loadingCtrl: LoadingController,
  47. private storage: Storage,
  48. private ipfs: P2pStorageIpfsProvider,
  49. private gun: P2pDatabaseGunProvider,
  50. private cryptoUtils: CryptoProvider,
  51. private opnpgp: PgpKeyServerProvider,
  52. private alertCtrl: AlertController
  53. ) {
  54. this.retweetId = this.navParams.get("tweetId");
  55. this.replyToStatusId = this.navParams.get("replyToStatus");
  56. this.tweet = this.formBuilder.group({
  57. text: [""],
  58. p2p: [false]
  59. });
  60. this.addValidators();
  61. }
  62. private async addValidators() {
  63. const triggerWords = await this.storage.get("keywords");
  64. const validators = [
  65. Validators.maxLength(140),
  66. this.containsTriggerWord(triggerWords)
  67. ];
  68. this.tweet.controls["text"].setValidators(validators);
  69. }
  70. private containsTriggerWord(triggerWords: string): ValidatorFn {
  71. return (control: AbstractControl): {
  72. [key: string]: any
  73. } | null => {
  74. if (triggerWords) {
  75. const regexList = triggerWords
  76. .toLowerCase()
  77. .split(", ")
  78. .join("|");
  79. const regex = new RegExp(regexList);
  80. const containsTriggerWord = regex.test(control.value.toLowerCase());
  81. return containsTriggerWord ? { containsTriggerWord: { value: control.value } } :
  82. null;
  83. } else {
  84. return null;
  85. }
  86. };
  87. }
  88. async ionViewDidLoad() {
  89. if (this.retweetId) {
  90. this.retweet = await this.twitter.fetchTweet(this.retweetId);
  91. }
  92. if (this.replyToStatusId) {
  93. this.replyTweet = await this.twitter.fetchTweet(this.replyToStatusId);
  94. }
  95. }
  96. get tweetCharProgress() {
  97. const progress = 1 - this.tweet.value["text"].length / 140;
  98. const radius = 8;
  99. const circumference = Math.PI * radius * 2;
  100. return progress * circumference;
  101. }
  102. get showTrigger(): boolean {
  103. return (
  104. this.tweet &&
  105. this.tweet.controls &&
  106. this.tweet.controls.text &&
  107. this.tweet.controls.text.errors &&
  108. this.tweet.controls.text.errors["containsTriggerWord"] &&
  109. !this.tweet.controls.p2p.value
  110. );
  111. }
  112. showTriggerInfo() {
  113. this.alertCtrl
  114. .create({
  115. title: "Watch Out!",
  116. message: "Your tweet contains words you have previously defined to only share securely via P2P. Currently P2P mode is not selected.",
  117. buttons: ["OK"]
  118. })
  119. .present();
  120. }
  121. async submitTweet() {
  122. const loading = this.loadingCtrl.create();
  123. loading.present();
  124. if (this.tweet.value.p2p) {
  125. loading.setContent("Validate keys...");
  126. if (
  127. (await this.cryptoUtils.isPrivateKeySet()) &&
  128. (await this.cryptoUtils.isPublicKeyPublished())
  129. ) {
  130. loading.setContent("Publish private tweet...");
  131. await this.tweetPrivate();
  132. } else {
  133. loading.dismiss();
  134. const alert = this.alertCtrl.create({
  135. title: "Oooops...",
  136. message: "Please verify that you have set a private and public key in the settings and that your latest public key was published."
  137. });
  138. alert.present();
  139. return;
  140. }
  141. } else {
  142. loading.setContent("Publish on Twitter...");
  143. await this.twitter.tweet(
  144. this.tweet.value["text"],
  145. this.retweet,
  146. this.replyToStatusId
  147. );
  148. }
  149. loading.dismiss();
  150. this.navCtrl.pop();
  151. }
  152. private async tweetPrivate() {
  153. const tweet = await this.buildPrivateTweet();
  154. const privateKey = await this.storage.get("privateKey");
  155. //encrypting for self
  156. let email = await this.storage.get("email");
  157. await this.opnpgp.lookupKeys(email);
  158. //encrypt the tweet with multiple keys
  159. let encryptedTweet = await this.opnpgp.encrypt(JSON.stringify(tweet));
  160. this.storeIPFS(encryptedTweet, tweet)
  161. return encryptedTweet;
  162. }
  163. private async storeIPFS(result, tweet){
  164. const res = await this.ipfs.storeTweet(result);
  165. this.gun.storeLastTweetHashForUser(
  166. tweet.user_id,
  167. res["Hash"],
  168. tweet.created_at
  169. );
  170. this.gun.publishHashtags(tweet.entities.hashtags);
  171. }
  172. private async buildPrivateTweet() {
  173. const status = this.tweet.value["text"].trim();
  174. const entities = await this.getEntities(status);
  175. return {
  176. full_text: status,
  177. user_id: await this.storage.get("userId"),
  178. created_at: Date.now(),
  179. private_tweet: true,
  180. in_reply_to_status_id: this.replyToStatusId,
  181. quoted_status_id: this.retweetId,
  182. display_text_range: [0, status.length],
  183. entities: entities
  184. };
  185. }
  186. private async getEntities(status: string) {
  187. return {
  188. hashtags: twittertext.extractHashtagsWithIndices(status),
  189. urls: twittertext.extractUrlsWithIndices(status),
  190. user_mentions: await this.getMentions(status)
  191. };
  192. }
  193. private async getMentions(status: string) {
  194. // extract mentions
  195. const entities = twittertext.extractMentionsWithIndices(status);
  196. // add user_id
  197. const entitiesWithPromises = entities.map(async mention => {
  198. try {
  199. const user = await this.twitter.fetchUserFromScreenName(
  200. mention.screenName
  201. );
  202. mention["id_str"] = user[0]["id_str"];
  203. mention["screen_name"] = mention.screenName;
  204. delete mention.screenName;
  205. } catch (err) {
  206. console.error(
  207. "There is no user signed up to twitter with username: " +
  208. mention.screenName
  209. );
  210. }
  211. return mention;
  212. });
  213. // filter for valid users and return
  214. return (await Promise.all(entitiesWithPromises)).filter(el =>
  215. el.hasOwnProperty("id_str")
  216. );
  217. }
  218. }