सुरक्षित ब्राउझर विस्तार लिहित आहे

सुरक्षित ब्राउझर विस्तार लिहित आहे

सामान्य "क्लायंट-सर्व्हर" आर्किटेक्चरच्या विपरीत, विकेंद्रित ऍप्लिकेशन्सचे वैशिष्ट्य आहे:

  • वापरकर्ता लॉगिन आणि पासवर्डसह डेटाबेस संचयित करण्याची आवश्यकता नाही. प्रवेश माहिती केवळ वापरकर्त्यांद्वारेच संग्रहित केली जाते आणि त्यांच्या सत्यतेची पुष्टी प्रोटोकॉल स्तरावर होते.
  • सर्व्हर वापरण्याची गरज नाही. ऍप्लिकेशन लॉजिक ब्लॉकचेन नेटवर्कवर कार्यान्वित केले जाऊ शकते, जिथे आवश्यक प्रमाणात डेटा संग्रहित करणे शक्य आहे.

वापरकर्ता की साठी 2 तुलनेने सुरक्षित स्टोरेज आहेत - हार्डवेअर वॉलेट आणि ब्राउझर विस्तार. हार्डवेअर वॉलेट्स बहुतेक अत्यंत सुरक्षित असतात, परंतु वापरण्यास कठीण असतात आणि ते विनामूल्य नसतात, परंतु ब्राउझर विस्तार हे सुरक्षितता आणि वापर सुलभतेचे परिपूर्ण संयोजन आहेत आणि अंतिम वापरकर्त्यांसाठी पूर्णपणे विनामूल्य देखील असू शकतात.

हे सर्व विचारात घेऊन, व्यवहार आणि स्वाक्षरींसह कार्य करण्यासाठी एक साधी API प्रदान करून विकेंद्रित ऍप्लिकेशन्सचा विकास सुलभ करणारा सर्वात सुरक्षित विस्तार आम्हाला बनवायचा आहे.
या अनुभवाबद्दल आम्ही तुम्हाला खाली सांगू.

लेखात कोड उदाहरणे आणि स्क्रीनशॉट्ससह ब्राउझर विस्तार कसा लिहावा याबद्दल चरण-दर-चरण सूचना असतील. आपण सर्व कोड शोधू शकता भांडार. प्रत्येक कमिट तार्किकदृष्ट्या या लेखाच्या विभागाशी संबंधित आहे.

ब्राउझर विस्तारांचा संक्षिप्त इतिहास

ब्राउझरचे विस्तार बरेच दिवसांपासून आहेत. ते 1999 मध्ये इंटरनेट एक्सप्लोररमध्ये, 2004 मध्ये फायरफॉक्समध्ये दिसले. तथापि, बर्याच काळापासून विस्तारासाठी कोणतेही एक मानक नव्हते.

आम्ही असे म्हणू शकतो की ते Google Chrome च्या चौथ्या आवृत्तीमध्ये विस्तारांसह दिसले. अर्थात, तेव्हा कोणतेही स्पेसिफिकेशन नव्हते, परंतु क्रोम एपीआय हाच त्याचा आधार बनला: ब्राउझर मार्केटचा बराचसा भाग जिंकून आणि अंगभूत ऍप्लिकेशन स्टोअर असल्याने, क्रोमने ब्राउझर विस्तारांचे मानक निश्चित केले.

Mozilla चे स्वतःचे मानक होते, परंतु Chrome विस्तारांची लोकप्रियता पाहून कंपनीने एक सुसंगत API बनवण्याचा निर्णय घेतला. 2015 मध्ये, Mozilla च्या पुढाकाराने, क्रॉस-ब्राउझर विस्तार वैशिष्ट्यांवर काम करण्यासाठी वर्ल्ड वाइड वेब कन्सोर्टियम (W3C) मध्ये एक विशेष गट तयार करण्यात आला.

Chrome साठी विद्यमान API विस्तार आधार म्हणून घेतले गेले. हे काम मायक्रोसॉफ्टच्या समर्थनासह केले गेले (गुगलने मानकांच्या विकासात भाग घेण्यास नकार दिला), आणि परिणामी एक मसुदा दिसला. तपशील.

औपचारिकपणे, तपशील एज, फायरफॉक्स आणि ऑपेरा द्वारे समर्थित आहे (लक्षात ठेवा की Chrome या सूचीमध्ये नाही). परंतु खरं तर, मानक मोठ्या प्रमाणात क्रोमशी सुसंगत आहे, कारण ते प्रत्यक्षात त्याच्या विस्तारांवर आधारित लिहिलेले आहे. तुम्ही WebExtensions API बद्दल अधिक वाचू शकता येथे.

विस्तार रचना

एक्स्टेंशनसाठी आवश्यक असलेली एकमेव फाइल मॅनिफेस्ट (manifest.json) आहे. हा विस्ताराचा "प्रवेश बिंदू" देखील आहे.

प्रकट

तपशीलानुसार, मॅनिफेस्ट फाइल वैध JSON फाइल आहे. कोणत्या ब्राउझरमध्ये कोणत्या की समर्थित आहेत याबद्दल माहितीसह मॅनिफेस्ट कीचे संपूर्ण वर्णन येथे.

स्पेसिफिकेशनमध्ये नसलेल्या की "कदाचित" दुर्लक्षित केल्या जाऊ शकतात (दोन्ही क्रोम आणि फायरफॉक्स एरर नोंदवतात, परंतु विस्तार कार्य करत राहतात).

आणि मी काही मुद्द्यांकडे लक्ष वेधू इच्छितो.

  1. पार्श्वभूमी — एक ऑब्जेक्ट ज्यामध्ये खालील फील्ड समाविष्ट आहेत:
    1. स्क्रिप्ट — स्क्रिप्ट्सची अॅरे जी पार्श्वभूमीच्या संदर्भात कार्यान्वित केली जाईल (आम्ही याबद्दल थोड्या वेळाने बोलू);
    2. पृष्ठ - रिकाम्या पृष्ठावर कार्यान्वित केल्या जाणाऱ्या स्क्रिप्ट्सऐवजी, आपण सामग्रीसह html निर्दिष्ट करू शकता. या प्रकरणात, स्क्रिप्ट फील्डकडे दुर्लक्ष केले जाईल, आणि स्क्रिप्ट सामग्री पृष्ठामध्ये समाविष्ट करणे आवश्यक आहे;
    3. टिकून राहाणे — बायनरी ध्वज, निर्दिष्ट न केल्यास, ब्राउझर पार्श्वभूमी प्रक्रिया "मारून टाकेल" जेव्हा ते असे समजते की ते काहीही करत नाही आणि आवश्यक असल्यास ते रीस्टार्ट करेल. अन्यथा, ब्राउझर बंद केल्यावरच पृष्ठ अनलोड केले जाईल. Firefox मध्ये समर्थित नाही.
  2. content_scripts — ऑब्जेक्ट्सची अॅरे जी तुम्हाला वेगवेगळ्या वेब पेजेसवर वेगवेगळ्या स्क्रिप्ट लोड करण्याची परवानगी देते. प्रत्येक ऑब्जेक्टमध्ये खालील महत्त्वपूर्ण फील्ड असतात:
    1. सामने - नमुना url, जे विशिष्ट सामग्री स्क्रिप्ट समाविष्ट केले जाईल की नाही हे निर्धारित करते.
    2. js — या सामन्यात लोड केल्या जाणाऱ्या स्क्रिप्टची सूची;
    3. exclude_maches - फील्डमधून वगळले जाते match या फील्डशी जुळणाऱ्या URL.
  3. पृष्ठ_क्रिया - प्रत्यक्षात एक ऑब्जेक्ट आहे जो ब्राउझरमध्ये अॅड्रेस बारच्या पुढे प्रदर्शित होणाऱ्या चिन्हासाठी आणि त्याच्याशी संवाद साधण्यासाठी जबाबदार आहे. हे तुम्हाला पॉपअप विंडो प्रदर्शित करण्यास देखील अनुमती देते, जी तुमची स्वतःची HTML, CSS आणि JS वापरून परिभाषित केली जाते.
    1. default_popup — पॉपअप इंटरफेससह HTML फाइलचा मार्ग, CSS आणि JS असू शकतात.
  4. परवानग्या — विस्तार अधिकार व्यवस्थापित करण्यासाठी अॅरे. 3 प्रकारचे अधिकार आहेत, ज्यांचे तपशीलवार वर्णन केले आहे येथे
  5. web_accessible_resources — वेब पृष्ठ विनंती करू शकणारे विस्तार संसाधने, उदाहरणार्थ, प्रतिमा, JS, CSS, HTML फाइल्स.
  6. बाह्य_जोडण्यायोग्य — येथे तुम्ही इतर विस्तारांचे आयडी आणि वेब पेजेसचे डोमेन स्पष्टपणे निर्दिष्ट करू शकता ज्यावरून तुम्ही कनेक्ट करू शकता. डोमेन द्वितीय स्तर किंवा उच्च असू शकते. Firefox मध्ये काम करत नाही.

अंमलबजावणी संदर्भ

विस्तारामध्ये तीन कोड एक्झिक्युशन संदर्भ आहेत, म्हणजेच, ब्राउझर API मध्ये प्रवेशाच्या विविध स्तरांसह अनुप्रयोगामध्ये तीन भाग असतात.

विस्तार संदर्भ

बहुतेक API येथे उपलब्ध आहेत. या संदर्भात ते "जगतात":

  1. पार्श्वभूमी पृष्ठ - विस्ताराचा "बॅकएंड" भाग. फाइल मॅनिफेस्टमध्ये "पार्श्वभूमी" की वापरून निर्दिष्ट केली आहे.
  2. पॉपअप पृष्ठ — एक पॉपअप पृष्ठ जे तुम्ही विस्तार चिन्हावर क्लिक करता तेव्हा दिसते. जाहीरनाम्यात browser_action -> default_popup.
  3. सानुकूल पृष्ठ — विस्तार पृष्ठ, दृश्याच्या वेगळ्या टॅबमध्ये “लिव्हिंग” chrome-extension://<id_расширения>/customPage.html.

हा संदर्भ ब्राउझर विंडो आणि टॅबपासून स्वतंत्रपणे अस्तित्वात आहे. पार्श्वभूमी पृष्ठ एका प्रतमध्ये अस्तित्वात आहे आणि नेहमी कार्य करते (अपवाद इव्हेंट पृष्ठ आहे, जेव्हा पार्श्वभूमी स्क्रिप्ट एखाद्या इव्हेंटद्वारे लॉन्च केली जाते आणि त्याच्या अंमलबजावणीनंतर "मृत्यू" होते). पॉपअप पृष्ठ जेव्हा पॉपअप विंडो उघडली जाते तेव्हा अस्तित्वात असते, आणि सानुकूल पृष्ठ — सोबतचा टॅब खुला असताना. या संदर्भातील इतर टॅब आणि त्यांच्या सामग्रीमध्ये प्रवेश नाही.

सामग्री स्क्रिप्ट संदर्भ

सामग्री स्क्रिप्ट फाइल प्रत्येक ब्राउझर टॅबसह लॉन्च केली जाते. त्याला एक्स्टेंशनच्या API चा भाग आणि वेब पृष्ठाच्या DOM ट्रीमध्ये प्रवेश आहे. ही सामग्री स्क्रिप्ट आहे जी पृष्ठासह परस्परसंवादासाठी जबाबदार असतात. DOM ट्रीमध्ये फेरफार करणारे विस्तार हे सामग्री स्क्रिप्टमध्ये करतात - उदाहरणार्थ, जाहिरात अवरोधक किंवा अनुवादक. तसेच, सामग्री स्क्रिप्ट मानक द्वारे पृष्ठाशी संवाद साधू शकते postMessage.

वेब पृष्ठ संदर्भ

हे स्वतःच वास्तविक वेब पृष्ठ आहे. या पृष्ठाचे डोमेन मॅनिफेस्टमध्ये स्पष्टपणे सूचित केलेले नाही अशा प्रकरणांशिवाय विस्ताराशी त्याचा काहीही संबंध नाही आणि तेथे प्रवेश नाही (खालील याबद्दल अधिक).

संदेशाची देवाणघेवाण

अनुप्रयोगाच्या विविध भागांनी एकमेकांशी संदेशांची देवाणघेवाण करणे आवश्यक आहे. यासाठी एपीआय आहे runtime.sendMessage संदेश पाठवण्यासाठी background и tabs.sendMessage पृष्ठावर संदेश पाठविण्यासाठी (सामग्री स्क्रिप्ट, पॉपअप किंवा वेब पृष्ठ उपलब्ध असल्यास externally_connectable). खाली Chrome API मध्ये प्रवेश करताना एक उदाहरण आहे.

// Сообщением может быть любой JSON сериализуемый объект
const msg = {a: 'foo', b: 'bar'};

// extensionId можно не указывать, если мы хотим послать сообщение 'своему' расширению (из ui или контент скрипта)
chrome.runtime.sendMessage(extensionId, msg);

// Так выглядит обработчик
chrome.runtime.onMessage.addListener((msg) => console.log(msg))

// Можно слать сообщения вкладкам зная их id
chrome.tabs.sendMessage(tabId, msg)

// Получить к вкладкам и их id можно, например, вот так
chrome.tabs.query(
    {currentWindow: true, active : true},
    function(tabArray){
      tabArray.forEach(tab => console.log(tab.id))
    }
)

संपूर्ण संप्रेषणासाठी, आपण याद्वारे कनेक्शन तयार करू शकता runtime.connect. प्रतिसादात आम्हाला मिळेल runtime.Port, ज्यावर, ते उघडे असताना, तुम्ही कितीही संदेश पाठवू शकता. क्लायंटच्या बाजूने, उदाहरणार्थ, contentscript, हे असे दिसते:

// Опять же extensionId можно не указывать при коммуникации внутри одного расширения. Подключение можно именовать
const port = chrome.runtime.connect({name: "knockknock"});
port.postMessage({joke: "Knock knock"});
port.onMessage.addListener(function(msg) {
    if (msg.question === "Who's there?")
        port.postMessage({answer: "Madame"});
    else if (msg.question === "Madame who?")
        port.postMessage({answer: "Madame... Bovary"});

सर्व्हर किंवा पार्श्वभूमी:

// Обработчик для подключения 'своих' вкладок. Контент скриптов, popup или страниц расширения
chrome.runtime.onConnect.addListener(function(port) {
    console.assert(port.name === "knockknock");
    port.onMessage.addListener(function(msg) {
        if (msg.joke === "Knock knock")
            port.postMessage({question: "Who's there?"});
        else if (msg.answer === "Madame")
            port.postMessage({question: "Madame who?"});
        else if (msg.answer === "Madame... Bovary")
            port.postMessage({question: "I don't get it."});
    });
});

// Обработчик для подключения внешних вкладок. Других расширений или веб страниц, которым разрешен доступ в манифесте
chrome.runtime.onConnectExternal.addListener(function(port) {
    ...
});

एक कार्यक्रम देखील आहे onDisconnect आणि पद्धत disconnect.

अर्ज आकृती

चला एक ब्राउझर विस्तार बनवू जे खाजगी की संग्रहित करते, सार्वजनिक माहितीमध्ये प्रवेश प्रदान करते (पत्ता, सार्वजनिक की पृष्ठाशी संप्रेषण करते आणि तृतीय-पक्ष अनुप्रयोगांना व्यवहारांसाठी स्वाक्षरीची विनंती करण्यास अनुमती देते.

अनुप्रयोग विकास

आमच्या अनुप्रयोगाने वापरकर्त्याशी संवाद साधला पाहिजे आणि कॉल पद्धती (उदाहरणार्थ, व्यवहारांवर स्वाक्षरी करण्यासाठी) API सह पृष्ठ प्रदान केले पाहिजे. फक्त एकासह करा contentscript कार्य करणार नाही, कारण त्यास फक्त DOM मध्ये प्रवेश आहे, परंतु पृष्ठाच्या JS वर नाही. द्वारे कनेक्ट करा runtime.connect आम्ही करू शकत नाही, कारण सर्व डोमेनवर API आवश्यक आहे आणि मॅनिफेस्टमध्ये केवळ विशिष्ट निर्दिष्ट केले जाऊ शकतात. परिणामी, आकृती असे दिसेल:

सुरक्षित ब्राउझर विस्तार लिहित आहे

दुसरी स्क्रिप्ट असेल - inpage, जे आम्ही पृष्ठावर इंजेक्ट करू. हे त्याच्या संदर्भात चालेल आणि विस्तारासह कार्य करण्यासाठी API प्रदान करेल.

Начало

सर्व ब्राउझर विस्तार कोड येथे उपलब्ध आहे GitHub. वर्णनादरम्यान कमिटचे दुवे असतील.

चला घोषणापत्रासह प्रारंभ करूया:

{
  // Имя и описание, версия. Все это будет видно в браузере в chrome://extensions/?id=<id расширения>
  "name": "Signer",
  "description": "Extension demo",
  "version": "0.0.1",
  "manifest_version": 2,

  // Скрипты, которые будут исполнятся в background, их может быть несколько
  "background": {
    "scripts": ["background.js"]
  },

  // Какой html использовать для popup
  "browser_action": {
    "default_title": "My Extension",
    "default_popup": "popup.html"
  },

  // Контент скрипты.
  // У нас один объект: для всех url начинающихся с http или https мы запускаем
  // contenscript context со скриптом contentscript.js. Запускать сразу по получении документа для всех фреймов
  "content_scripts": [
    {
      "matches": [
        "http://*/*",
        "https://*/*"
      ],
      "js": [
        "contentscript.js"
      ],
      "run_at": "document_start",
      "all_frames": true
    }
  ],
  // Разрешен доступ к localStorage и idle api
  "permissions": [
    "storage",
    // "unlimitedStorage",
    //"clipboardWrite",
    "idle"
    //"activeTab",
    //"webRequest",
    //"notifications",
    //"tabs"
  ],
  // Здесь указываются ресурсы, к которым будет иметь доступ веб страница. Тоесть их можно будет запрашивать fetche'м или просто xhr
  "web_accessible_resources": ["inpage.js"]
}

रिक्त background.js, popup.js, inpage.js आणि contentscript.js तयार करा. आम्ही popup.html जोडतो - आणि आमचा अनुप्रयोग आधीपासूनच Google Chrome मध्ये लोड केला जाऊ शकतो आणि ते कार्य करते याची खात्री करा.

हे सत्यापित करण्यासाठी, तुम्ही कोड घेऊ शकता येथून. आम्ही जे केले त्याव्यतिरिक्त, लिंकने वेबपॅक वापरून प्रकल्पाची असेंब्ली कॉन्फिगर केली. ब्राउझरमध्ये अॅप्लिकेशन जोडण्यासाठी, chrome://extensions मध्ये तुम्हाला लोड अनपॅक केलेले आणि संबंधित एक्स्टेंशन असलेले फोल्डर निवडावे लागेल - आमच्या बाबतीत जि.

सुरक्षित ब्राउझर विस्तार लिहित आहे

आता आमचा विस्तार स्थापित आणि कार्यरत आहे. तुम्ही खालीलप्रमाणे विविध संदर्भांसाठी विकसक साधने चालवू शकता:

पॉपअप ->

सुरक्षित ब्राउझर विस्तार लिहित आहे

सामग्री स्क्रिप्ट कन्सोलमध्ये प्रवेश ज्या पृष्ठावर लॉन्च केला आहे त्या पृष्ठाच्या कन्सोलद्वारे केला जातो.सुरक्षित ब्राउझर विस्तार लिहित आहे

संदेशाची देवाणघेवाण

म्हणून, आम्हाला दोन संप्रेषण चॅनेल स्थापित करणे आवश्यक आहे: इनपेज पार्श्वभूमी आणि पॉपअप पार्श्वभूमी. तुम्ही अर्थातच, फक्त पोर्टवर संदेश पाठवू शकता आणि तुमचा स्वतःचा प्रोटोकॉल शोधू शकता, परंतु मी मेटामास्क ओपन सोर्स प्रोजेक्टमध्ये पाहिलेल्या दृष्टिकोनाला प्राधान्य देतो.

इथरियम नेटवर्कसह कार्य करण्यासाठी हे ब्राउझर विस्तार आहे. त्यात, अॅप्लिकेशनचे वेगवेगळे भाग dnode लायब्ररी वापरून RPC द्वारे संवाद साधतात. जर तुम्ही नोडज स्ट्रीमला ट्रान्सपोर्ट (म्हणजे समान इंटरफेस लागू करणारी वस्तू) प्रदान केली तर ते तुम्हाला एक्स्चेंज त्वरीत आणि सोयीस्करपणे आयोजित करण्यास अनुमती देते:

import Dnode from "dnode/browser";

// В этом примере условимся что клиент удаленно вызывает функции на сервере, хотя ничего нам не мешает сделать это двунаправленным

// Cервер
// API, которое мы хотим предоставить
const dnode = Dnode({
    hello: (cb) => cb(null, "world")
})
// Транспорт, поверх которого будет работать dnode. Любой nodejs стрим. В браузере есть бибилиотека 'readable-stream'
connectionStream.pipe(dnode).pipe(connectionStream)

// Клиент
const dnodeClient = Dnode() // Вызов без агрумента значит что мы не предоставляем API на другой стороне

// Выведет в консоль world
dnodeClient.once('remote', remote => {
    remote.hello(((err, value) => console.log(value)))
})

आता आपण अॅप्लिकेशन क्लास तयार करू. हे पॉपअप आणि वेब पृष्ठासाठी API ऑब्जेक्ट्स तयार करेल आणि त्यांच्यासाठी एक dnode तयार करेल:

import Dnode from 'dnode/browser';

export class SignerApp {

    // Возвращает объект API для ui
    popupApi(){
        return {
            hello: cb => cb(null, 'world')
        }
    }

    // Возвращает объет API для страницы
    pageApi(){
        return {
            hello: cb => cb(null, 'world')
        }
    }

    // Подключает popup ui
    connectPopup(connectionStream){
        const api = this.popupApi();
        const dnode = Dnode(api);

        connectionStream.pipe(dnode).pipe(connectionStream);

        dnode.on('remote', (remote) => {
            console.log(remote)
        })
    }

    // Подключает страницу
    connectPage(connectionStream, origin){
        const api = this.popupApi();
        const dnode = Dnode(api);

        connectionStream.pipe(dnode).pipe(connectionStream);

        dnode.on('remote', (remote) => {
            console.log(origin);
            console.log(remote)
        })
    }
}

येथे आणि खाली, ग्लोबल क्रोम ऑब्जेक्ट ऐवजी, आम्ही एक्स्टेंशनApi वापरतो, जो Google च्या ब्राउझरमध्ये Chrome आणि इतर ब्राउझरमध्ये प्रवेश करतो. हे क्रॉस-ब्राउझर सुसंगततेसाठी केले जाते, परंतु या लेखाच्या हेतूंसाठी, तुम्ही फक्त 'chrome.runtime.connect' वापरू शकता.

पार्श्वभूमी स्क्रिप्टमध्ये एक अनुप्रयोग उदाहरण तयार करूया:

import {extensionApi} from "./utils/extensionApi";
import {PortStream} from "./utils/PortStream";
import {SignerApp} from "./SignerApp";

const app = new SignerApp();

// onConnect срабатывает при подключении 'процессов' (contentscript, popup, или страница расширения)
extensionApi.runtime.onConnect.addListener(connectRemote);

function connectRemote(remotePort) {
    const processName = remotePort.name;
    const portStream = new PortStream(remotePort);
    // При установке соединения можно указывать имя, по этому имени мы и оппределяем кто к нам подлючился, контентскрипт или ui
    if (processName === 'contentscript'){
        const origin = remotePort.sender.url
        app.connectPage(portStream, origin)
    }else{
        app.connectPopup(portStream)
    }
}

dnode प्रवाहांसह कार्य करत असल्याने, आणि आम्हाला एक पोर्ट मिळतो, अॅडॉप्टर वर्ग आवश्यक आहे. हे वाचनीय-स्ट्रीम लायब्ररी वापरून बनवले आहे, जे ब्राउझरमध्ये नोडज स्ट्रीम लागू करते:

import {Duplex} from 'readable-stream';

export class PortStream extends Duplex{
    constructor(port){
        super({objectMode: true});
        this._port = port;
        port.onMessage.addListener(this._onMessage.bind(this));
        port.onDisconnect.addListener(this._onDisconnect.bind(this))
    }

    _onMessage(msg) {
        if (Buffer.isBuffer(msg)) {
            delete msg._isBuffer;
            const data = new Buffer(msg);
            this.push(data)
        } else {
            this.push(msg)
        }
    }

    _onDisconnect() {
        this.destroy()
    }

    _read(){}

    _write(msg, encoding, cb) {
        try {
            if (Buffer.isBuffer(msg)) {
                const data = msg.toJSON();
                data._isBuffer = true;
                this._port.postMessage(data)
            } else {
                this._port.postMessage(msg)
            }
        } catch (err) {
            return cb(new Error('PortStream - disconnected'))
        }
        cb()
    }
}

आता UI मध्ये कनेक्शन तयार करूया:

import {extensionApi} from "./utils/extensionApi";
import {PortStream} from "./utils/PortStream";
import Dnode from 'dnode/browser';

const DEV_MODE = process.env.NODE_ENV !== 'production';

setupUi().catch(console.error);

async function setupUi(){
    // Также, как и в классе приложения создаем порт, оборачиваем в stream, делаем  dnode
    const backgroundPort = extensionApi.runtime.connect({name: 'popup'});
    const connectionStream = new PortStream(backgroundPort);

    const dnode = Dnode();

    connectionStream.pipe(dnode).pipe(connectionStream);

    const background = await new Promise(resolve => {
        dnode.once('remote', api => {
            resolve(api)
        })
    });

    // Делаем объект API доступным из консоли
    if (DEV_MODE){
        global.background = background;
    }
}

मग आम्ही सामग्री स्क्रिप्टमध्ये कनेक्शन तयार करतो:

import {extensionApi} from "./utils/extensionApi";
import {PortStream} from "./utils/PortStream";
import PostMessageStream from 'post-message-stream';

setupConnection();
injectScript();

function setupConnection(){
    const backgroundPort = extensionApi.runtime.connect({name: 'contentscript'});
    const backgroundStream = new PortStream(backgroundPort);

    const pageStream = new PostMessageStream({
        name: 'content',
        target: 'page',
    });

    pageStream.pipe(backgroundStream).pipe(pageStream);
}

function injectScript(){
    try {
        // inject in-page script
        let script = document.createElement('script');
        script.src = extensionApi.extension.getURL('inpage.js');
        const container = document.head || document.documentElement;
        container.insertBefore(script, container.children[0]);
        script.onload = () => script.remove();
    } catch (e) {
        console.error('Injection failed.', e);
    }
}

आम्हाला API ची आवश्यकता सामग्री स्क्रिप्टमध्ये नसून थेट पृष्ठावर असल्याने, आम्ही दोन गोष्टी करतो:

  1. आम्ही दोन प्रवाह तयार करतो. एक - पृष्ठाकडे, postMessage च्या वर. यासाठी आम्ही हे वापरतो हे पॅकेज मेटामास्कच्या निर्मात्यांकडून. दुसरा प्रवाह पोर्टवरून प्राप्त झालेल्या पार्श्वभूमीसाठी आहे runtime.connect. चला ते विकत घेऊया. आता पृष्ठास पार्श्वभूमीवर एक प्रवाह असेल.
  2. DOM मध्ये स्क्रिप्ट इंजेक्ट करा. स्क्रिप्ट डाउनलोड करा (मॅनिफेस्टमध्ये त्यात प्रवेश करण्याची परवानगी होती) आणि टॅग तयार करा script त्यातील सामग्रीसह:

import PostMessageStream from 'post-message-stream';
import {extensionApi} from "./utils/extensionApi";
import {PortStream} from "./utils/PortStream";

setupConnection();
injectScript();

function setupConnection(){
    // Стрим к бекграунду
    const backgroundPort = extensionApi.runtime.connect({name: 'contentscript'});
    const backgroundStream = new PortStream(backgroundPort);

    // Стрим к странице
    const pageStream = new PostMessageStream({
        name: 'content',
        target: 'page',
    });

    pageStream.pipe(backgroundStream).pipe(pageStream);
}

function injectScript(){
    try {
        // inject in-page script
        let script = document.createElement('script');
        script.src = extensionApi.extension.getURL('inpage.js');
        const container = document.head || document.documentElement;
        container.insertBefore(script, container.children[0]);
        script.onload = () => script.remove();
    } catch (e) {
        console.error('Injection failed.', e);
    }
}

आता आम्ही इनपेजमध्ये एपीआय ऑब्जेक्ट तयार करतो आणि ते ग्लोबलवर सेट करतो:

import PostMessageStream from 'post-message-stream';
import Dnode from 'dnode/browser';

setupInpageApi().catch(console.error);

async function setupInpageApi() {
    // Стрим к контентскрипту
    const connectionStream = new PostMessageStream({
        name: 'page',
        target: 'content',
    });

    const dnode = Dnode();

    connectionStream.pipe(dnode).pipe(connectionStream);

    // Получаем объект API
    const pageApi = await new Promise(resolve => {
        dnode.once('remote', api => {
            resolve(api)
        })
    });

    // Доступ через window
    global.SignerApp = pageApi;
}

आम्ही तयार आहोत पृष्ठ आणि UI साठी स्वतंत्र API सह दूरस्थ प्रक्रिया कॉल (RPC).. पार्श्वभूमीशी नवीन पृष्ठ कनेक्ट करताना आम्ही हे पाहू शकतो:

सुरक्षित ब्राउझर विस्तार लिहित आहे

रिक्त API आणि मूळ. पृष्‍ठाच्या बाजूला, आपण हॅलो फंक्शनला याप्रमाणे कॉल करू शकतो:

सुरक्षित ब्राउझर विस्तार लिहित आहे

आधुनिक JS मध्ये कॉलबॅक फंक्शन्ससह कार्य करणे वाईट शिष्टाचार आहे, म्हणून आपण एक dnode तयार करण्यासाठी एक लहान मदतनीस लिहूया जो तुम्हाला एपीआय ऑब्जेक्ट युटिल्समध्ये पास करण्यास अनुमती देतो.

API ऑब्जेक्ट्स आता यासारखे दिसतील:

export class SignerApp {

    popupApi() {
        return {
            hello: async () => "world"
        }
    }

...

}

याप्रमाणे रिमोटवरून ऑब्जेक्ट मिळवणे:

import {cbToPromise, transformMethods} from "../../src/utils/setupDnode";

const pageApi = await new Promise(resolve => {
    dnode.once('remote', remoteApi => {
        // С помощью утилит меняем все callback на promise
        resolve(transformMethods(cbToPromise, remoteApi))
    })
});

आणि कॉलिंग फंक्शन्स एक वचन देते:

सुरक्षित ब्राउझर विस्तार लिहित आहे

असिंक्रोनस फंक्शन्ससह आवृत्ती उपलब्ध आहे येथे.

एकूणच, RPC आणि प्रवाहाचा दृष्टीकोन खूपच लवचिक दिसतो: आम्ही स्टीम मल्टीप्लेक्सिंग वापरू शकतो आणि वेगवेगळ्या कार्यांसाठी अनेक भिन्न API तयार करू शकतो. तत्त्वानुसार, डनोड कुठेही वापरला जाऊ शकतो, मुख्य गोष्ट म्हणजे नोडज प्रवाहाच्या स्वरूपात वाहतूक गुंडाळणे.

पर्यायी JSON फॉरमॅट आहे, जो JSON RPC 2 प्रोटोकॉल लागू करतो. तथापि, ते विशिष्ट वाहतूक (TCP आणि HTTP(S)) सह कार्य करते, जे आमच्या बाबतीत लागू होत नाही.

अंतर्गत राज्य आणि स्थानिक स्टोरेज

आम्हाला अनुप्रयोगाची अंतर्गत स्थिती संग्रहित करणे आवश्यक आहे - किमान स्वाक्षरी की. आम्ही पॉपअप API मध्ये अनुप्रयोग आणि ते बदलण्याच्या पद्धती अगदी सहजपणे जोडू शकतो:

import {setupDnode} from "./utils/setupDnode";

export class SignerApp {

    constructor(){
        this.store = {
            keys: [],
        };
    }

    addKey(key){
        this.store.keys.push(key)
    }

    removeKey(index){
        this.store.keys.splice(index,1)
    }

    popupApi(){
        return {
            addKey: async (key) => this.addKey(key),
            removeKey: async (index) => this.removeKey(index)
        }
    }

    ...

} 

पार्श्वभूमीत, आम्ही सर्व काही फंक्शनमध्ये गुंडाळू आणि ऍप्लिकेशन ऑब्जेक्ट विंडोवर लिहू जेणेकरुन आम्ही कन्सोलमधून त्याच्यासह कार्य करू शकू:

import {extensionApi} from "./utils/extensionApi";
import {PortStream} from "./utils/PortStream";
import {SignerApp} from "./SignerApp";

const DEV_MODE = process.env.NODE_ENV !== 'production';

setupApp();

function setupApp() {
    const app = new SignerApp();

    if (DEV_MODE) {
        global.app = app;
    }

    extensionApi.runtime.onConnect.addListener(connectRemote);

    function connectRemote(remotePort) {
        const processName = remotePort.name;
        const portStream = new PortStream(remotePort);
        if (processName === 'contentscript') {
            const origin = remotePort.sender.url;
            app.connectPage(portStream, origin)
        } else {
            app.connectPopup(portStream)
        }
    }
}

चला UI कन्सोल मधून काही की जोडू आणि राज्याचे काय होते ते पाहू:

सुरक्षित ब्राउझर विस्तार लिहित आहे

रीस्टार्ट करताना कळा हरवल्या जाणार नाहीत म्हणून स्थिती कायम ठेवणे आवश्यक आहे.

आम्ही ते लोकल स्टोरेजमध्ये संग्रहित करू, प्रत्येक बदलासह ते ओव्हरराईट करू. त्यानंतर, UI साठी देखील त्यात प्रवेश आवश्यक असेल आणि मला बदलांचे सदस्यत्व घ्यायचे आहे. यावर आधारित, निरीक्षण करण्यायोग्य स्टोरेज तयार करणे आणि त्यातील बदलांचे सदस्यत्व घेणे सोयीचे होईल.

आम्ही mobx लायब्ररी वापरू.https://github.com/mobxjs/mobx). निवड त्यावर पडली कारण मला त्यात काम करण्याची गरज नव्हती, परंतु मला खरोखर त्याचा अभ्यास करायचा होता.

चला प्रारंभिक अवस्थेचे प्रारंभ जोडू आणि स्टोअरला निरीक्षण करण्यायोग्य बनवू:

import {observable, action} from 'mobx';
import {setupDnode} from "./utils/setupDnode";

export class SignerApp {

    constructor(initState = {}) {
        // Внешне store так и останется тем же объектом, только теперь все его поля стали proxy, которые отслеживают доступ к ним
        this.store =  observable.object({
            keys: initState.keys || [],
        });
    }

    // Методы, которые меняют observable принято оборачивать декоратором
    @action
    addKey(key) {
        this.store.keys.push(key)
    }

    @action
    removeKey(index) {
        this.store.keys.splice(index, 1)
    }

    ...

}

“अंडर द हुड,” mobx ने सर्व स्टोअर फील्ड प्रॉक्सीने बदलले आहेत आणि त्यांना सर्व कॉल्स इंटरसेप्ट करतात. या संदेशांचे सदस्यत्व घेणे शक्य होईल.

खाली मी "बदलताना" हा शब्द वापरतो, जरी हे पूर्णपणे बरोबर नाही. Mobx फील्डमध्ये प्रवेश ट्रॅक करते. लायब्ररी तयार केलेल्या प्रॉक्सी ऑब्जेक्ट्सचे गेटर्स आणि सेटर्स वापरले जातात.

अॅक्शन डेकोरेटर्स दोन उद्देश पूर्ण करतात:

  1. अंमलबजावणीच्या ध्वजांसह कठोर मोडमध्ये, mobx थेट राज्य बदलण्यास प्रतिबंधित करते. कठोर परिस्थितीत काम करणे चांगले मानले जाते.
  2. जरी एखादे कार्य अनेक वेळा स्थिती बदलते - उदाहरणार्थ, आम्ही कोडच्या अनेक ओळींमध्ये अनेक फील्ड बदलतो - ते पूर्ण झाल्यावरच निरीक्षकांना सूचित केले जाते. हे विशेषतः फ्रंटएंडसाठी महत्वाचे आहे, जेथे अनावश्यक स्थिती अद्यतनांमुळे घटकांचे अनावश्यक प्रस्तुतीकरण होते. आमच्या बाबतीत, पहिला किंवा दुसरा विशेषत: संबंधित नाही, परंतु आम्ही सर्वोत्तम पद्धतींचे पालन करू. सर्व फंक्शन्समध्ये डेकोरेटर जोडण्याची प्रथा आहे जी निरीक्षण केलेल्या फील्डची स्थिती बदलते.

पार्श्वभूमीत आम्ही लोकल स्टोरेजमध्ये प्रारंभ आणि राज्य जतन करू:

import {reaction, toJS} from 'mobx';
import {extensionApi} from "./utils/extensionApi";
import {PortStream} from "./utils/PortStream";
import {SignerApp} from "./SignerApp";
// Вспомогательные методы. Записывают/читают объект в/из localStorage виде JSON строки по ключу 'store'
import {loadState, saveState} from "./utils/localStorage";

const DEV_MODE = process.env.NODE_ENV !== 'production';

setupApp();

function setupApp() {
    const initState = loadState();
    const app = new SignerApp(initState);

    if (DEV_MODE) {
        global.app = app;
    }

    // Setup state persistence

    // Результат reaction присваивается переменной, чтобы подписку можно было отменить. Нам это не нужно, оставлено для примера
    const localStorageReaction = reaction(
        () => toJS(app.store), // Функция-селектор данных
        saveState // Функция, которая будет вызвана при изменении данных, которые возвращает селектор
    );

    extensionApi.runtime.onConnect.addListener(connectRemote);

    function connectRemote(remotePort) {
        const processName = remotePort.name;
        const portStream = new PortStream(remotePort);
        if (processName === 'contentscript') {
            const origin = remotePort.sender.url
            app.connectPage(portStream, origin)
        } else {
            app.connectPopup(portStream)
        }
    }
}

प्रतिक्रिया कार्य येथे मनोरंजक आहे. यात दोन युक्तिवाद आहेत:

  1. डेटा निवडक.
  2. एक हँडलर ज्याला हा डेटा प्रत्येक वेळी बदलताना कॉल केला जाईल.

redux च्या विपरीत, जिथे आम्ही स्पष्टपणे एक युक्तिवाद म्हणून स्थिती प्राप्त करतो, mobx लक्षात ठेवतो की आम्ही निवडकर्त्याच्या आत कोणत्या निरीक्षणासाठी प्रवेश करतो आणि जेव्हा ते बदलतात तेव्हाच हँडलरला कॉल करते.

आम्ही कोणत्या निरीक्षणासाठी सदस्यता घेतली हे mobx कसे ठरवते हे समजून घेणे महत्त्वाचे आहे. जर मी कोडमध्ये निवडकर्ता असे लिहिले() => app.store, नंतर प्रतिक्रिया कधीही कॉल केली जाणार नाही, कारण स्टोरेज स्वतःच निरीक्षण करण्यायोग्य नाही, फक्त त्याचे फील्ड आहेत.

मी असे लिहिले तर () => app.store.keys, नंतर पुन्हा काहीही होणार नाही, कारण अॅरे घटक जोडताना/काढत असताना, त्याचा संदर्भ बदलणार नाही.

Mobx प्रथमच निवडक म्हणून काम करते आणि फक्त आम्ही प्रवेश केलेल्या निरीक्षणावर लक्ष ठेवते. हे प्रॉक्सी गेटर्सद्वारे केले जाते. म्हणून, येथे अंगभूत कार्य वापरले जाते toJS. हे मूळ फील्डसह बदललेल्या सर्व प्रॉक्सीसह नवीन ऑब्जेक्ट परत करते. अंमलबजावणी दरम्यान, ते ऑब्जेक्टचे सर्व फील्ड वाचते - म्हणून गेटर्स ट्रिगर केले जातात.

पॉपअप कन्सोलमध्ये आम्ही पुन्हा अनेक की जोडू. यावेळी ते लोकल स्टोरेजमध्ये देखील संपले:

सुरक्षित ब्राउझर विस्तार लिहित आहे

जेव्हा पार्श्वभूमी पृष्ठ रीलोड केले जाते, तेव्हा माहिती जागेवर राहते.

या बिंदूपर्यंतचे सर्व अनुप्रयोग कोड पाहिले जाऊ शकतात येथे.

खाजगी की सुरक्षित स्टोरेज

स्पष्ट मजकूरात खाजगी की संग्रहित करणे असुरक्षित आहे: तुम्हाला हॅक होण्याची, तुमच्या संगणकावर प्रवेश मिळण्याची आणि अशीच काही शक्यता असते. म्हणून, लोकल स्टोरेजमध्ये आम्ही कीज पासवर्ड-एनक्रिप्टेड स्वरूपात साठवू.

अधिक सुरक्षिततेसाठी, आम्ही ऍप्लिकेशनमध्ये एक लॉक केलेली स्थिती जोडू, ज्यामध्ये की अजिबात प्रवेश होणार नाही. कालबाह्य झाल्यामुळे आम्ही विस्तार लॉक केलेल्या स्थितीत स्वयंचलितपणे हस्तांतरित करू.

Mobx तुम्हाला डेटाचा किमान संच संचयित करण्याची परवानगी देतो आणि बाकीची आपोआप त्यावर आधारित गणना केली जाते. हे तथाकथित गणना केलेले गुणधर्म आहेत. त्यांची तुलना डेटाबेसमधील दृश्यांशी केली जाऊ शकते:

import {observable, action} from 'mobx';
import {setupDnode} from "./utils/setupDnode";
// Утилиты для безопасного шифрования строк. Используют crypto-js
import {encrypt, decrypt} from "./utils/cryptoUtils";

export class SignerApp {
    constructor(initState = {}) {
        this.store = observable.object({
            // Храним пароль и зашифрованные ключи. Если пароль null - приложение locked
            password: null,
            vault: initState.vault,

            // Геттеры для вычислимых полей. Можно провести аналогию с view в бд.
            get locked(){
                return this.password == null
            },
            get keys(){
                return this.locked ?
                    undefined :
                    SignerApp._decryptVault(this.vault, this.password)
            },
            get initialized(){
                return this.vault !== undefined
            }
        })
    }
    // Инициализация пустого хранилища новым паролем
    @action
    initVault(password){
        this.store.vault = SignerApp._encryptVault([], password)
    }
    @action
    lock() {
        this.store.password = null
    }
    @action
    unlock(password) {
        this._checkPassword(password);
        this.store.password = password
    }
    @action
    addKey(key) {
        this._checkLocked();
        this.store.vault = SignerApp._encryptVault(this.store.keys.concat(key), this.store.password)
    }
    @action
    removeKey(index) {
        this._checkLocked();
        this.store.vault = SignerApp._encryptVault([
                ...this.store.keys.slice(0, index),
                ...this.store.keys.slice(index + 1)
            ],
            this.store.password
        )
    }

    ... // код подключения и api

    // private
    _checkPassword(password) {
        SignerApp._decryptVault(this.store.vault, password);
    }

    _checkLocked() {
        if (this.store.locked){
            throw new Error('App is locked')
        }
    }

    // Методы для шифровки/дешифровки хранилища
    static _encryptVault(obj, pass){
        const jsonString = JSON.stringify(obj)
        return encrypt(jsonString, pass)
    }

    static _decryptVault(str, pass){
        if (str === undefined){
            throw new Error('Vault not initialized')
        }
        try {
            const jsonString = decrypt(str, pass)
            return JSON.parse(jsonString)
        }catch (e) {
            throw new Error('Wrong password')
        }
    }
}

आता आम्ही फक्त एनक्रिप्टेड की आणि पासवर्ड साठवतो. बाकी सर्व काही मोजले जाते. आम्ही राज्यातील पासवर्ड काढून लॉक केलेल्या स्थितीत हस्तांतरण करतो. पब्लिक एपीआयकडे आता स्टोरेज सुरू करण्याची पद्धत आहे.

एनक्रिप्शनसाठी लिहिलेले क्रिप्टो-जेएस वापरून उपयुक्तता:

import CryptoJS from 'crypto-js'

// Используется для осложнения подбора пароля перебором. На каждый вариант пароля злоумышленнику придется сделать 5000 хешей
function strengthenPassword(pass, rounds = 5000) {
    while (rounds-- > 0){
        pass = CryptoJS.SHA256(pass).toString()
    }
    return pass
}

export function encrypt(str, pass){
    const strongPass = strengthenPassword(pass);
    return CryptoJS.AES.encrypt(str, strongPass).toString()
}

export function decrypt(str, pass){
    const strongPass = strengthenPassword(pass)
    const decrypted = CryptoJS.AES.decrypt(str, strongPass);
    return decrypted.toString(CryptoJS.enc.Utf8)
}

ब्राउझरमध्ये निष्क्रिय API आहे ज्याद्वारे तुम्ही इव्हेंटची सदस्यता घेऊ शकता - स्थिती बदल. राज्य, त्यानुसार, असू शकते idle, active и locked. निष्क्रियतेसाठी तुम्ही कालबाह्य सेट करू शकता आणि OS स्वतः अवरोधित केल्यावर लॉक सेट केले जाते. आम्ही लोकल स्टोरेजमध्ये सेव्ह करण्यासाठी निवडकर्ता देखील बदलू:

import {reaction, toJS} from 'mobx';
import {extensionApi} from "./utils/extensionApi";
import {PortStream} from "./utils/PortStream";
import {SignerApp} from "./SignerApp";
import {loadState, saveState} from "./utils/localStorage";

const DEV_MODE = process.env.NODE_ENV !== 'production';
const IDLE_INTERVAL = 30;

setupApp();

function setupApp() {
    const initState = loadState();
    const app = new SignerApp(initState);

    if (DEV_MODE) {
        global.app = app;
    }

    // Теперь мы явно узываем поле, которому будет происходить доступ, reaction отработает нормально
    reaction(
        () => ({
            vault: app.store.vault
        }),
        saveState
    );

    // Таймаут бездействия, когда сработает событие
    extensionApi.idle.setDetectionInterval(IDLE_INTERVAL);
    // Если пользователь залочил экран или бездействовал в течение указанного интервала лочим приложение
    extensionApi.idle.onStateChanged.addListener(state => {
        if (['locked', 'idle'].indexOf(state) > -1) {
            app.lock()
        }
    });

    // Connect to other contexts
    extensionApi.runtime.onConnect.addListener(connectRemote);

    function connectRemote(remotePort) {
        const processName = remotePort.name;
        const portStream = new PortStream(remotePort);
        if (processName === 'contentscript') {
            const origin = remotePort.sender.url
            app.connectPage(portStream, origin)
        } else {
            app.connectPopup(portStream)
        }
    }
}

या चरणापूर्वीचा कोड आहे येथे.

व्यवहार

तर, आम्ही सर्वात महत्वाच्या गोष्टीकडे आलो: ब्लॉकचेनवर व्यवहार तयार करणे आणि त्यावर स्वाक्षरी करणे. आम्ही WAVES ब्लॉकचेन आणि लायब्ररी वापरू लहरी व्यवहार.

प्रथम, स्वाक्षरी करणे आवश्यक असलेल्या संदेशांची अॅरे राज्यामध्ये जोडू या, नंतर नवीन संदेश जोडण्यासाठी, स्वाक्षरीची पुष्टी करण्यासाठी आणि नकार देण्यासाठी पद्धती जोडा:

import {action, observable, reaction} from 'mobx';
import uuid from 'uuid/v4';
import {signTx} from '@waves/waves-transactions'
import {setupDnode} from "./utils/setupDnode";
import {decrypt, encrypt} from "./utils/cryptoUtils";

export class SignerApp {

    ...

    @action
    newMessage(data, origin) {
        // Для каждого сообщения создаем метаданные с id, статусом, выременем создания и тд.
        const message = observable.object({
            id: uuid(), // Идентификатор, используюю uuid
            origin, // Origin будем впоследствии показывать в интерфейсе
            data, //
            status: 'new', // Статусов будет четыре: new, signed, rejected и failed
            timestamp: Date.now()
        });
        console.log(`new message: ${JSON.stringify(message, null, 2)}`);

        this.store.messages.push(message);

        // Возвращаем промис внутри которого mobx мониторит изменения сообщения. Как только статус поменяется мы зарезолвим его
        return new Promise((resolve, reject) => {
            reaction(
                () => message.status, //Будем обсервить статус сообщеня
                (status, reaction) => { // второй аргумент это ссылка на сам reaction, чтобы его можно было уничтожть внутри вызова
                    switch (status) {
                        case 'signed':
                            resolve(message.data);
                            break;
                        case 'rejected':
                            reject(new Error('User rejected message'));
                            break;
                        case 'failed':
                            reject(new Error(message.err.message));
                            break;
                        default:
                            return
                    }
                    reaction.dispose()
                }
            )
        })
    }
    @action
    approve(id, keyIndex = 0) {
        const message = this.store.messages.find(msg => msg.id === id);
        if (message == null) throw new Error(`No msg with id:${id}`);
        try {
            message.data = signTx(message.data, this.store.keys[keyIndex]);
            message.status = 'signed'
        } catch (e) {
            message.err = {
                stack: e.stack,
                message: e.message
            };
            message.status = 'failed'
            throw e
        }
    }
    @action
    reject(id) {
        const message = this.store.messages.find(msg => msg.id === id);
        if (message == null) throw new Error(`No msg with id:${id}`);
        message.status = 'rejected'
    }

    ...
}

जेव्हा आम्हाला नवीन संदेश प्राप्त होतो, तेव्हा आम्ही त्यात मेटाडेटा जोडतो, करा observable आणि जोडा store.messages.

आपण नाही तर observable मॅन्युअली, मग अ‍ॅरेमध्ये संदेश जोडताना mobx ते स्वतः करेल. तथापि, ते एक नवीन ऑब्जेक्ट तयार करेल ज्याचा आमच्याकडे संदर्भ नसेल, परंतु आम्हाला पुढील चरणासाठी त्याची आवश्यकता असेल.

पुढे, आम्ही संदेशाची स्थिती बदलल्यावर निराकरण करणारे वचन देतो. प्रतिक्रियेद्वारे स्थितीचे परीक्षण केले जाते, जे स्थिती बदलते तेव्हा "स्वतःला मारून टाकते".

पद्धत कोड approve и reject अगदी सोपे: आवश्यक असल्यास स्वाक्षरी केल्यावर आम्ही फक्त संदेशाची स्थिती बदलतो.

आम्ही UI API मध्ये मंजूर आणि नाकारतो, पृष्ठ API मध्ये newMessage ठेवतो:

export class SignerApp {
    ...
    popupApi() {
        return {
            addKey: async (key) => this.addKey(key),
            removeKey: async (index) => this.removeKey(index),

            lock: async () => this.lock(),
            unlock: async (password) => this.unlock(password),
            initVault: async (password) => this.initVault(password),

            approve: async (id, keyIndex) => this.approve(id, keyIndex),
            reject: async (id) => this.reject(id)
        }
    }

    pageApi(origin) {
        return {
            signTransaction: async (txParams) => this.newMessage(txParams, origin)
        }
    }

    ...
}

आता विस्तारासह व्यवहारावर स्वाक्षरी करण्याचा प्रयत्न करूया:

सुरक्षित ब्राउझर विस्तार लिहित आहे

सर्वसाधारणपणे, सर्वकाही तयार आहे, जे काही शिल्लक आहे ते आहे साधे UI जोडा.

UI

इंटरफेसला ऍप्लिकेशन स्थितीत प्रवेश आवश्यक आहे. UI बाजूला आम्ही करू observable स्टेट करा आणि API मध्ये एक फंक्शन जोडा जे ही स्थिती बदलेल. जोडूया observable पार्श्वभूमीतून प्राप्त API ऑब्जेक्टवर:

import {observable} from 'mobx'
import {extensionApi} from "./utils/extensionApi";
import {PortStream} from "./utils/PortStream";
import {cbToPromise, setupDnode, transformMethods} from "./utils/setupDnode";
import {initApp} from "./ui/index";

const DEV_MODE = process.env.NODE_ENV !== 'production';

setupUi().catch(console.error);

async function setupUi() {
    // Подключаемся к порту, создаем из него стрим
    const backgroundPort = extensionApi.runtime.connect({name: 'popup'});
    const connectionStream = new PortStream(backgroundPort);

    // Создаем пустой observable для состояния background'a
    let backgroundState = observable.object({});
    const api = {
        //Отдаем бекграунду функцию, которая будет обновлять observable
        updateState: async state => {
            Object.assign(backgroundState, state)
        }
    };

    // Делаем RPC объект
    const dnode = setupDnode(connectionStream, api);
    const background = await new Promise(resolve => {
        dnode.once('remote', remoteApi => {
            resolve(transformMethods(cbToPromise, remoteApi))
        })
    });

    // Добавляем в background observable со стейтом
    background.state = backgroundState;

    if (DEV_MODE) {
        global.background = background;
    }

    // Запуск интерфейса
    await initApp(background)
}

शेवटी आम्ही ऍप्लिकेशन इंटरफेस प्रस्तुत करणे सुरू करतो. हे एक प्रतिक्रिया अनुप्रयोग आहे. पार्श्वभूमी ऑब्जेक्ट फक्त प्रॉप्स वापरून पास केले जाते. अर्थातच, पद्धतींसाठी स्वतंत्र सेवा आणि राज्यासाठी स्टोअर बनवणे योग्य आहे, परंतु या लेखाच्या हेतूंसाठी हे पुरेसे आहे:

import {render} from 'react-dom'
import App from './App'
import React from "react";

// Инициализируем приложение с background объектом в качест ве props
export async function initApp(background){
    render(
        <App background={background}/>,
        document.getElementById('app-content')
    );
}

डेटा बदलतो तेव्हा mobx सह रेंडरिंग सुरू करणे खूप सोपे आहे. आम्ही फक्त पॅकेजमधून निरीक्षक डेकोरेटरला टांगतो mobx-प्रतिक्रिया घटकावर, आणि रेंडर आपोआप कॉल केला जाईल जेव्हा घटक बदलाद्वारे संदर्भित केलेले कोणतेही निरीक्षणे. तुम्हाला कोणत्याही mapStateToProps किंवा redux प्रमाणे कनेक्ट करण्याची आवश्यकता नाही. सर्व काही बॉक्सच्या बाहेर कार्य करते:

import React, {Component, Fragment} from 'react'
import {observer} from "mobx-react";
import Init from './components/Initialize'
import Keys from './components/Keys'
import Sign from './components/Sign'
import Unlock from './components/Unlock'

@observer // У Компонета с этим декоратом будет автоматически вызван метод render, если будут изменены observable на которые он ссылается
export default class App extends Component {

    // Правильно конечно вынести логику рендера страниц в роутинг и не использовать вложенные тернарные операторы,
    // и привязывать observable и методы background непосредственно к тем компонентам, которые их используют
    render() {
        const {keys, messages, initialized, locked} = this.props.background.state;
        const {lock, unlock, addKey, removeKey, initVault, deleteVault, approve, reject} = this.props.background;

        return <Fragment>
            {!initialized
                ?
                <Init onInit={initVault}/>
                :
                locked
                    ?
                    <Unlock onUnlock={unlock}/>
                    :
                    messages.length > 0
                        ?
                        <Sign keys={keys} message={messages[messages.length - 1]} onApprove={approve} onReject={reject}/>
                        :
                        <Keys keys={keys} onAdd={addKey} onRemove={removeKey}/>
            }
            <div>
                {!locked && <button onClick={() => lock()}>Lock App</button>}
                {initialized && <button onClick={() => deleteVault()}>Delete all keys and init</button>}
            </div>
        </Fragment>
    }
}

उर्वरित घटक कोडमध्ये पाहिले जाऊ शकतात UI फोल्डरमध्ये.

आता अॅप्लिकेशन क्लासमध्ये तुम्हाला UI साठी स्टेट सिलेक्टर बनवणे आणि UI बदलल्यावर सूचित करणे आवश्यक आहे. हे करण्यासाठी, एक पद्धत जोडूया getState и reactionकॉल करणे remote.updateState:

import {action, observable, reaction} from 'mobx';
import uuid from 'uuid/v4';
import {signTx} from '@waves/waves-transactions'
import {setupDnode} from "./utils/setupDnode";
import {decrypt, encrypt} from "./utils/cryptoUtils";

export class SignerApp {

    ...

    // public
    getState() {
        return {
            keys: this.store.keys,
            messages: this.store.newMessages,
            initialized: this.store.initialized,
            locked: this.store.locked
        }
    }

    ...

    //
    connectPopup(connectionStream) {
        const api = this.popupApi();
        const dnode = setupDnode(connectionStream, api);

        dnode.once('remote', (remote) => {
            // Создаем reaction на изменения стейта, который сделает вызовет удаленну процедуру и обновит стейт в ui процессе
            const updateStateReaction = reaction(
                () => this.getState(),
                (state) => remote.updateState(state),
                // Третьим аргументом можно передавать параметры. fireImmediatly значит что reaction выполниться первый раз сразу.
                // Это необходимо, чтобы получить начальное состояние. Delay позволяет установить debounce
                {fireImmediately: true, delay: 500}
            );
            // Удалим подписку при отключении клиента
            dnode.once('end', () => updateStateReaction.dispose())

        })
    }

    ...
}

एखादी वस्तू प्राप्त करताना remote तयार केले आहे reaction UI बाजूला फंक्शन कॉल करणारी स्थिती बदलण्यासाठी.

अंतिम स्पर्श म्हणजे विस्तार चिन्हावर नवीन संदेशांचे प्रदर्शन जोडणे:

function setupApp() {
...

    // Reaction на выставление текста беджа.
    reaction(
        () => app.store.newMessages.length > 0 ? app.store.newMessages.length.toString() : '',
        text => extensionApi.browserAction.setBadgeText({text}),
        {fireImmediately: true}
    );

...
}

तर, अर्ज तयार आहे. वेब पृष्ठे व्यवहारांसाठी स्वाक्षरीची विनंती करू शकतात:

सुरक्षित ब्राउझर विस्तार लिहित आहे

सुरक्षित ब्राउझर विस्तार लिहित आहे

कोड येथे उपलब्ध आहे दुवा.

निष्कर्ष

जर तुम्ही लेख शेवटपर्यंत वाचला असेल, परंतु तरीही प्रश्न असतील तर तुम्ही त्यांना येथे विचारू शकता विस्तारासह भांडार. तेथे तुम्हाला प्रत्येक नियुक्त चरणासाठी कमिट देखील आढळतील.

आणि जर तुम्हाला वास्तविक विस्तारासाठी कोड पाहण्यात स्वारस्य असेल, तर तुम्ही हे शोधू शकता येथे.

कडून कोड, भांडार आणि नोकरीचे वर्णन siemarell

स्त्रोत: www.habr.com

एक टिप्पणी जोडा