From 6da39d80a1a8b7c7b53a00b9c3b2b7f6fdc56cab Mon Sep 17 00:00:00 2001 From: Kimberly Date: Thu, 15 Feb 2018 13:09:58 -0700 Subject: [PATCH] # This is a combination of 2 commits. # This is the 1st commit message: MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit # This is a combination of 2 commits. # This is the 1st commit message: # This is a combination of 3 commits. # This is the 1st commit message: # This is a combination of 9 commits. # This is the 1st commit message: # This is a combination of 15 commits. # This is the 1st commit message: # This is a combination of 49 commits. # This is the 1st commit message: # This is a combination of 2 commits. # This is the 1st commit message: # This is a combination of 25 commits. # This is the 1st commit message: flake8 # This is the commit message #2: Refresh OAuth Access Tokens (#547) # This is the commit message #3: Styling adjustments for the account dropdown. (#549) # This is the commit message #4: Update Version and Release Notes to 0.1.7-alpha (#558) # This is the commit message #5: chore(package): update eslint-plugin-no-unsanitized to version 3.0.0 (#537) # This is the commit message #6: chore(package): update mocha to version 5.0.1 (#544) # This is the commit message #7: chore(package): update stylelint to version 9.1.1 (#566) # This is the commit message #8: chore(package): update stylelint-config-recommended to version 2.1.0 (#551) # This is the commit message #9: Add Support For Telemetry Scalars; Update Metrics.md (#552) # This is the commit message #10: Provide documentation about how to post PRs (#507) # This is the commit message #11: Ensure FxA account is stored always (#570) # This is the commit message #12: Update Version and Release Notes to 0.1.7-alpha1 (#571) # This is the commit message #13: Updated text color of entry list empty state for better contrast (#560) # This is the commit message #14: Add config file for pyup.io to update weekly (#586) * create pyup.io config file * update all dependencies, not just security * remove extra whitespace # This is the commit message #15: Update flake8-isort from 2.2.2 to 2.4 (#587) # This is the commit message #16: Update pytest from 3.3.1 to 3.4.1 (#589) # This is the commit message #17: Update pytest-selenium from 1.11.3 to 1.11.4 (#590) # This is the commit message #18: Update pypom from 1.2.0 to 1.3.0 (#594) # This is the commit message #19: Update selenium from 3.8.0 to 3.9.0 (#592) # This is the commit message #20: Update pytest-xdist from 1.20.1 to 1.22.2 (#591) # This is the commit message #21: Update flake8-docstrings from 1.1.0 to 1.3.0 (#588) # This is the commit message #22: More account dropdown style updates (#597) * Styling updates based off Jim's code review of #549 -Updated pseudo-elements to have two colons instead of one -Moved toolbar margin update to be specific to navigation bar -Removed arrowhead down icon that is no longer in use -Updated dropdown hover, focus and active states after reviewing how Firefox browser handles similar items * Updating toolbar to use px instead of ch to be consistent. # This is the commit message #23: Include whitespace in item fields so new lines are output (#596) * Include whitespace in item fields so new lines are output * move white-space to all input text fields # This is the commit message #24: Automatically focus the filter input when the page loads; resolves #573 # This is the commit message #25: Styling updates around entry list items (#553) * Styling updates around entry list items -Active and focus states -Increased padding to match mocks -New chevron icon -Updated aside background color -Moved border between search and entries in order to apply active border color to top entry item -Small copy change to empty state * Changes based off PR feedback -Removed arrowhead right icon that is no longer used -Removed some styling around border between filter and list items until we get the filter updated. * Add custom "no username" instruction text when adding entry * Remove string interpolation and use better const name * Ran cheveron-right through svgo, sizing was also updated. # This is the commit message #2: 0.1.7-alpha1 (#574) # This is the commit message #2: Updating styling of 0 entries (home) page to reflect intended typography and spacing (#607) - Into image shadow to Shadow 10 - Updated image border - Defined line height for intro paragraphs # This is the commit message #3: Create a and update the styles for the unlock do… (#601) * Create a and update the styles for the unlock doorhanger; resolves #482 * latest Nessie with darker waves # This is the commit message #4: Update selenium from 3.9.0 to 3.10.0 # This is the commit message #5: chore(package): update documentation to version 6.0.0 # This is the commit message #6: Require python 3.6 in deploy stage so pip can install tornado # This is the commit message #7: Create/Edit Entry Details Styling Adjustments (#567) * Styling adjustments for entry details -Added show/hide icons for password -Reordered fields to match new designs -Removed monospace class from username field -Fixed sizing/spacing to match designs -Updated to use box-sizing: border box, and fixed issues this caused in other areas -Updated placeholder copy and labels to match design changes -Added min-width to buttons * Adding min-width to normal button style and larger min-width for primary buttons * Styling for dialog box and adding new critical/red button style * Changes based on PR review -ran svgo on show icon to optimize -created new notes class in entry details for styling -removed box sizing and associated style changes -moved password styles to input.css -Removed show/hide button text and keys -Changed critical theme to danger theme -Reverted dialog to use primary button by default * Re-localize the show/hide password input buttons * Fix input tests to also check for first "input" class * Better RegEx for munged password input class tests * Fixing missed styles from box-sizing removal that needed updated * Provide the option of using "danger" buttons in dialog boxes These are currently used in two places: * Deleting items * Resetting the datastore * restore indentation * Fixes based on PR changes requested -Ran show icon through svgo -Moved input max-width for entry creation/edition fields to item-fields.css -Created new "wide" button sub-theme for instances to include min-width property -Added size theme to save/edit button on entry details * Adding .input class to item fields for styling and removing unnecessary selector # This is the commit message #8: Improvements to search bar styling in full manage mode (#562) * Styling adjustments for entry list filter. -Added search icon -Updated padding/colors -Included clear icon but needs implemented -Removed "entries" from placeholder copy * Styling updates for entries list filter in aside -Fixed padding/sizing to match design -Added search icon -Replaced text link with 'clear icon' -Updated placeholder copy to 'search lockbox' * Changes after review with Ryan -Darkening placeholder text -Updated border radius on inputs -Reverting search to use regular weight * Moving location of styles for filter/search in order to cover both doorhanger and full page view * Adding MPL-2.0 license to item-filter file and new line on app.css * Updates based on PR feedback -Added new line at end of SVG files -Removed item-filter.css and moved styles to input.css file -Wrapped filter in element and applied padding there -Added title attribute to clear button * Removing "clear" text from filter button * Update input test regex to include first input filter class * Restore localization of "Clear" text on search * Better RegEx for munged input class tests * Removed some unnecessary styles left behind and updated padding for consistency. * Ran clear and search svgs through svgo * Restore filter styles lost in master branch merge 28e31d5 # This is the commit message #9: Make sure options_ui is fully displayed on about:addons (#603) * Make sure options_ui is fully displayed on about:addons * flex-direction is actually not necessary, not sure why i ended up with it * add margin so button focus state fits and text aligns with items above * add min-height to give the warning dialog more room to breathe # This is the commit message #10: added accessibility test plan to docs # This is the commit message #11: Attempt to speed up branch and PR builds with Travis CI caching (#604) * Speed up branch and PR builds with Travis CI caching * add pip cache # This is the commit message #12: writing jenkins config # This is the commit message #13: fixed typo # This is the commit message #14: edited jenkins config # This is the commit message #15: edited jenkins config # This is the commit message #16: edited jenkins config # This is the commit message #17: edited jenkins config # This is the commit message #18: edited jenkins config # This is the commit message #19: edited jenkins config # This is the commit message #20: edited jenkins config # This is the commit message #21: edited jenkins config # This is the commit message #22: edited jenkins config # This is the commit message #23: edited jenkins config # This is the commit message #24: edited jenkins config # This is the commit message #25: added jenkins build script # This is the commit message #26: edited jenkins config # This is the commit message #27: edited jenkins config # This is the commit message #28: edited jenkins config # This is the commit message #29: edited jenkins config # This is the commit message #30: edited jenkins config # This is the commit message #31: edited jenkins config # This is the commit message #32: edited jenkins config # This is the commit message #33: edited jenkins config # This is the commit message #34: edited jenkins config # This is the commit message #35: including a11y tests # This is the commit message #36: flake8 # This is the commit message #37: flake8 # This is the commit message #38: removed pypom from test reqs # This is the commit message #39: edited jenkins config # This is the commit message #40: edited jenkins config # This is the commit message #41: edited jenkins config # This is the commit message #42: edited jenkins config # This is the commit message #43: edited jenkins config # This is the commit message #44: Styling adjustment to align entry detail buttons during edit and view mode change (#608) # This is the commit message #45: Fix to actually update its state from Redux # This is the commit message #46: Fix importing of chai-enzyme in test # This is the commit message #47: Automatically select the filter input's text on load (if there is any) # This is the commit message #48: Pre-fill the URL of the current tab into the doorhanger's filter # This is the commit message #49: Prepare 0.1.8 release (#612) * bump version numbers to 0.1.8-alpha * draft release notes based on current progress and fixed release note headings in previous release * recent additions for release notes * last items included in this release * remove depdencies updates from release notes # This is the commit message #2: Fix the color of the horizontal line in the item details of the manager # This is the commit message #3: edited jenkins config # This is the commit message #4: edited jenkins config # This is the commit message #5: edited jenkins config # This is the commit message #6: edited jenkins config # This is the commit message #7: edited jenkins config # This is the commit message #8: edited jenkins config # This is the commit message #9: edited jenkins config # This is the commit message #10: edited jenkins config # This is the commit message #11: edited jenkins config # This is the commit message #12: edited jenkins config # This is the commit message #13: edited jenkins config # This is the commit message #14: Move python runtime for deploy stage out of 'on' requirement So the pages actually build using python 3.6 instead of just checking to make sure its in the 3.6 runtime # This is the commit message #15: Downgrade Travis deploy script to fix pages deploy See travis-ci/travis-ci#9312 # This is the commit message #2: Improve implementation of (formerly ) Unfortunately, to have behavior like , the only way I can figure out is just to use the XUL box model. :( # This is the commit message #3: Allow greater customization of # This is the commit message #4: Provide visuals for the quick-copy buttons in the doorhanger list view # This is the commit message #5: Add support for copying the username and password from the quick-copy buttons # This is the commit message #6: Add telemetry for the quick-copy buttons # This is the commit message #7: Address styling concerns # This is the commit message #8: Add tests for quick-copy behavior # This is the commit message #9: Address #570 Post-Merge Code Review Comments (#647) # This is the commit message #2: changes to metrics # This is the commit message #3: Update stylelint to the latest version 🚀 (#658) * chore(package): update stylelint to version 9.1.2 * chore(package): update stylelint to version 9.1.3 # This is the commit message #2: Update flake8-isort from 2.4 to 2.5 (#671) # This is the commit message #2: Docs are wrong becuase sync is not currently supported fixes #670 --- .gitignore | 2 + .pyup.yml | 3 + .travis.yml | 8 + Jenkinsfile | 56 ++ docs/.dir-locals.el | 2 + docs/accessibility-test-plan.md | 48 ++ docs/contributing.md | 24 +- docs/faqs.md | 2 +- docs/metrics.md | 140 ++-- docs/release-notes.md | 73 ++ package-lock.json | 778 +++++++++++++----- package.json | 12 +- requirements/flake8.txt | 4 +- requirements/tests.txt | 10 +- src/bootstrap.js | 45 +- src/webextension/background/accounts/index.js | 205 +++-- src/webextension/background/browser-action.js | 9 +- src/webextension/background/datastore.js | 4 + src/webextension/background/index.js | 7 +- src/webextension/background/message-ports.js | 33 +- src/webextension/background/telemetry.js | 13 +- src/webextension/icons/account.svg | 4 +- src/webextension/icons/arrowhead-down-16.svg | 3 - src/webextension/icons/arrowhead-right-16.svg | 6 - src/webextension/icons/chevron-right.svg | 3 + src/webextension/icons/clear.svg | 3 + src/webextension/icons/hide.svg | 7 + src/webextension/icons/options.svg | 2 +- src/webextension/icons/search.svg | 3 + src/webextension/icons/show.svg | 6 + src/webextension/icons/signout.svg | 4 +- src/webextension/images/nessie_v2.svg | 479 +++-------- .../list/components/item-fields.css | 18 +- .../list/components/item-fields.js | 48 +- .../list/components/item-list.css | 15 +- src/webextension/list/components/item-list.js | 15 +- .../list/components/item-summary.css | 46 +- .../list/components/item-summary.js | 75 +- .../list/containers/item-filter.js | 11 +- .../manage/components/account-summary.css | 67 +- .../list/manage/components/account-summary.js | 2 +- .../list/manage/components/app.css | 17 +- .../list/manage/components/app.js | 65 +- .../manage/components/edit-item-details.js | 2 +- .../list/manage/components/intro-page.css | 9 +- .../list/manage/components/item-details.css | 5 +- .../list/manage/containers/all-items.css | 10 +- .../list/manage/containers/modals.js | 1 + src/webextension/list/popup/components/app.js | 28 +- .../list/popup/components/item-list-panel.js | 16 +- .../list/popup/containers/all-items.js | 18 +- .../popup/containers/current-selection.js | 5 +- src/webextension/list/popup/index.js | 13 +- src/webextension/locales/en-US/list.ftl | 17 +- src/webextension/locales/en-US/unlock.ftl | 3 + src/webextension/locales/en-US/widgets.ftl | 9 +- src/webextension/manifest.json.tpl | 3 +- src/webextension/settings/components/app.css | 8 +- .../settings/containers/modals.js | 1 + src/webextension/unlock/components/app.css | 62 +- src/webextension/unlock/components/app.js | 56 +- src/webextension/widgets/button-stack.js | 44 - src/webextension/widgets/button.css | 15 +- src/webextension/widgets/button.js | 2 + .../widgets/copy-to-clipboard-button.js | 43 +- src/webextension/widgets/dialog-box.css | 8 +- src/webextension/widgets/dialog-box.js | 37 +- src/webextension/widgets/filter-input.js | 21 +- src/webextension/widgets/input.css | 70 +- src/webextension/widgets/input.js | 5 +- src/webextension/widgets/link.css | 1 + src/webextension/widgets/panel.css | 61 +- src/webextension/widgets/panel.js | 40 + src/webextension/widgets/password-input.js | 16 +- .../widgets/{button-stack.css => stack.css} | 17 +- src/webextension/widgets/stack.js | 50 ++ src/webextension/widgets/text-area.css | 8 +- src/webextension/widgets/text-area.js | 7 +- src/webextension/widgets/toolbar.css | 4 +- test/integration/pages/base.py | 2 +- test/integration/pages/home.py | 11 +- test/integration/pages/login.py | 5 +- test/unit/background/accounts-test.js | 165 +++- test/unit/background/message-ports-test.js | 10 +- test/unit/background/telemetry-test.js | 50 +- test/unit/bootstrap-test.js | 20 +- test/unit/chai-focus.js | 17 + test/unit/common.js | 8 +- test/unit/list/components/item-list-test.js | 11 + .../unit/list/components/item-summary-test.js | 57 +- test/unit/list/containers/item-filter-test.js | 24 +- .../manage/components/account-summary-test.js | 4 +- test/unit/list/manage/components/app-test.js | 24 +- .../components/edit-item-details-test.js | 2 +- .../current-account-summary-test.js | 2 +- test/unit/list/popup/components/app-test.js | 31 +- .../list/popup/containers/all-items-test.js | 4 + test/unit/mocks/xpcom.js | 2 + .../widgets/copy-to-clipboard-button-test.js | 31 +- test/unit/widgets/dialog-box-test.js | 35 +- test/unit/widgets/filter-input-test.js | 29 +- test/unit/widgets/input-test.js | 12 +- test/unit/widgets/password-input-test.js | 4 +- .../{button-stack-test.js => stack-test.js} | 18 +- test/unit/widgets/text-area-test.js | 12 +- tox.ini | 42 +- 106 files changed, 2492 insertions(+), 1232 deletions(-) create mode 100644 .pyup.yml create mode 100644 Jenkinsfile create mode 100644 docs/.dir-locals.el create mode 100644 docs/accessibility-test-plan.md delete mode 100644 src/webextension/icons/arrowhead-down-16.svg delete mode 100644 src/webextension/icons/arrowhead-right-16.svg create mode 100644 src/webextension/icons/chevron-right.svg create mode 100644 src/webextension/icons/clear.svg create mode 100644 src/webextension/icons/hide.svg create mode 100644 src/webextension/icons/search.svg create mode 100644 src/webextension/icons/show.svg delete mode 100644 src/webextension/widgets/button-stack.js rename src/webextension/widgets/{button-stack.css => stack.css} (58%) create mode 100644 src/webextension/widgets/stack.js rename test/unit/widgets/{button-stack-test.js => stack-test.js} (78%) diff --git a/.gitignore b/.gitignore index 65835965..6f874c19 100644 --- a/.gitignore +++ b/.gitignore @@ -12,3 +12,5 @@ __pycache__ .tox *.log *.pyc +.pytest_cache +/results/ diff --git a/.pyup.yml b/.pyup.yml new file mode 100644 index 00000000..29c7c230 --- /dev/null +++ b/.pyup.yml @@ -0,0 +1,3 @@ +# see https://pyup.io/docs/configuration/ for all available options + +schedule: every week diff --git a/.travis.yml b/.travis.yml index eee45206..3f6650d4 100644 --- a/.travis.yml +++ b/.travis.yml @@ -5,6 +5,11 @@ _aliases: language: node_js node_js: stable +cache: + directories: + - node_modules +cache: pip + sudo: required jobs: include: @@ -47,8 +52,11 @@ jobs: - stage: deploy if: branch = master <<: *node + python: '3.6' script: skip deploy: + edge: + branch: v1.8.47 provider: pages on: branch: master diff --git a/Jenkinsfile b/Jenkinsfile new file mode 100644 index 00000000..70e5949b --- /dev/null +++ b/Jenkinsfile @@ -0,0 +1,56 @@ +#!/usr/bin/env groovy + +/** Desired capabilities */ +def capabilities = [ + browserName: 'Firefox', + version: '58.0', + platform: 'Windows 10' +] + +pipeline { + agent any + stages { + stage('Checkout') { + steps { + deleteDir() + checkout scm + stash 'workspace' + } + } + stage('Lint') { + steps { + deleteDir() + unstash 'workspace' + ansiColor('xterm') { + sh """ + pip install -r requirements/flake8.txt + python -m flake8 + """ + } + } + } + stage('Test') { + environment { + SAUCELABS = credentials('SAUCELABS') + } + steps { + unstash 'workspace' + ansiColor('xterm') { + sh """ + npm install + git init + npm run package + mkdir results + tox -e sauce + """ + } + } + post { + always { + stash includes: 'results/*', name: 'results' + archiveArtifacts 'results/*' + } + } + } + } +} diff --git a/docs/.dir-locals.el b/docs/.dir-locals.el new file mode 100644 index 00000000..e23c659d --- /dev/null +++ b/docs/.dir-locals.el @@ -0,0 +1,2 @@ +((markdown-mode + (mode . visual-line))) diff --git a/docs/accessibility-test-plan.md b/docs/accessibility-test-plan.md new file mode 100644 index 00000000..ebc28d2b --- /dev/null +++ b/docs/accessibility-test-plan.md @@ -0,0 +1,48 @@ +# Manual Test Plan + +## Keyboard Navigation +_< 3 hours_ + +Navigate the web content using only the keyboard +- Tab Order:`TAB`, as well as `SHIFT + TAB`, follows a logical and intuitive order. +- All controls, links, buttons, etc., get focused. +- Focus should be visibly apparent. +- When a modal or pop up window opens, focus shifts to the pop up. + - Focus and tab order is constrained within the modal. + - Modal can be exited via keyboard. + - When a pop up window is closed, focus returns to a logical point. +- Ensure that all content that is visually hidden is also hidden from the keyboard and/or a screen reader. _Except content specifically for screen reader users._ +- Firefox toolbar icon can be accessed by keyboard and screen reader users. + +## Page Structure +_3 hours_ +- Links and buttons have text that give context. +- All images contain alt text to describe the purpose or content of the image to a non-sighted user. +- A logical hierarchy of header tags has been used. +- [ARIA landmark roles](https://accessibility.oit.ncsu.edu/it-accessibility-at-nc-state/developers/accessibility-handbook/aria-landmarks/) have been utilized where applicable. +- ARIA attributes have been put into use. + - Any custom controls have been given the proper role attribute. + - State changes are provided via [ARIA states](http://www.w3.org/TR/wai-aria/states_and_properties#attrs_widgets_header). +- Form input fields and buttons have been labeled for screen reader users. + +## Color +_< 1 hour_ +- Text has sufficient color contrast against its background. + - Contrast ratio of 4.5:1 for normal text (less than 18 point or 14 point bold.) + - Contrast ratio of 3:1 for large text (at least 18 point or 14 point bold). +- Information conveyed via color is also available by other means. +- Buttons and links are visually apparent when they have focus. +- Focus remains apparent, and contrast remains sufficient, when simulating varying types of colorblindness. [Color blind simulators](https://www.toptal.com/designers/colorfilter) + +## UX Design Considerations +- Interactions available with a mouse are also available using a keyboard. +- Information conveyed visually is also available by other means. + - Pop-ups and other state changes are announced to screen readers using appropriate ARIA attributes. + - Temporary on-screen events give users ample time to receive conveyed information. +- Animations are not excessively flashy, as this can cause seizures in some users. + +## Tools and Resources +Use a variety of testing tools to analyze pages: +- aXe developer tools by Deque +- Browser developer tools to inspect HTML for logical heading structure and ARIA roles +- [Color blind simulators](https://www.toptal.com/designers/colorfilter) diff --git a/docs/contributing.md b/docs/contributing.md index 773ddd14..ee6f2fb8 100644 --- a/docs/contributing.md +++ b/docs/contributing.md @@ -6,7 +6,7 @@ The following are guidelines for contributing to this project. ## Code of Conduct -This repository is governed by Mozilla's code of conduct and etiquette guidelines. For more details please see the [Mozilla Community Participation Guidelines](https://www.mozilla.org/about/governance/policies/participation/) and [Developer Etiquette Guidelines](https://bugzilla.mozilla.org/page.cgi?id=etiquette.html). +This repository is governed by Mozilla's code of conduct and etiquette guidelines. For more details please see the [Mozilla Community Participation Guidelines][community-guidelines] and [Developer Etiquette Guidelines][etiquette-guidelines]. ## How to Get Started @@ -14,16 +14,32 @@ Please refer to installation and build instructions in the [install documentatio ## How to Report Bugs -Please open [a new issue in the appropriate GitHub repository](https://github.com/mozilla-lockbox/lockbox-extension/issues/new) with steps to reproduce the problem you're experiencing. +Please open [a new issue in the appropriate GitHub repository][new-issue] with steps to reproduce the problem you're experiencing. Be sure to include as much information including screenshots, text output, and both your expected and actual results. ## How to Request Enhancements -First, please refer to the applicable [GitHub repository](https://github.com/orgs/mozilla-lockbox/) and search [the repository's GitHub issues](https://github.com/mozilla-lockbox/lockbox-extension/issues) to make sure your idea has not been (or is not still) considered. It may also be easier to see and search across all projects combined on our [Waffle.io planning board](https://waffle.io/mozilla-lockbox/lockbox-extension). +First, please refer to the applicable [GitHub repository][github-repos] and search [the repository's GitHub issues][issues-list] to make sure your idea has not been (or is not still) considered. It may also be easier to see and search across all projects combined on our [Waffle.io planning board][waffle]. -Then, please [create a new issue in the GitHub repository](https://github.com/mozilla-lockbox/lockbox-extension/issues/new) describing your enhancement. +Then, please [create a new issue in the GitHub repository][new-issue] describing your enhancement. Be sure to include as much detail as possible including step-by-step descriptions, specific examples, screenshots or mockups, and reasoning for why the enhancement might be worthwhile. Please keep in mind, by opening an issue we provide no guarantee the enhancement will be implemented. + +## How to Contribute Code + +Before you get started writing code, be sure what you plan to work on is something we'll be able to accept. The easiest way is to look through out list of [good first issues][good-first-issues] and find something that sounds interesting. + +If there's something else you'd like to work on, just add a comment in the relevant issue and we'll be happy to discuss your plans. If you have an idea that doesn't have an issue at all, be sure to [file an issue](#how-to-request-enhancements) first. + +Once you have a patch ready, submit a pull request and request a review from **@mozilla-lockbox/engineering**. From there, we'll guide you through the review process to the eventual landing of your code! We aim to respond to all review requests within two business days. + +[community-guidelines]: https://www.mozilla.org/about/governance/policies/participation/ +[etiquette-guidelines]: https://bugzilla.mozilla.org/page.cgi?id=etiquette.html +[new-issue]: https://github.com/mozilla-lockbox/lockbox-extension/issues/new +[github-repos]: https://github.com/orgs/mozilla-lockbox/ +[issues-list]: https://github.com/mozilla-lockbox/lockbox-extension/issues +[waffle]: https://waffle.io/mozilla-lockbox/lockbox-extension +[good-first-issues]: https://github.com/mozilla-lockbox/lockbox-extension/issues?q=is%3Aopen+is%3Aissue+label%3A%22good+first+issue%22 diff --git a/docs/faqs.md b/docs/faqs.md index 6e2496a7..87992113 100644 --- a/docs/faqs.md +++ b/docs/faqs.md @@ -42,7 +42,7 @@ The alpha version of Lockbox hasn’t been tested widely with other password man ## Do Lockbox entries sync to other computers with Lockbox installed? -Yes, if you secure Lockbox with a Firefox Account. +No, not currently. ## Can I try Lockbox if I don’t have a Mozilla.com email address? diff --git a/docs/metrics.md b/docs/metrics.md index 513cd2de..3f2c54ec 100644 --- a/docs/metrics.md +++ b/docs/metrics.md @@ -1,60 +1,72 @@ # Lockbox Metrics Plan -_Last Updated: Nov 28, 2017_ +_Last Updated: Feb 19, 2018_ - [Analysis](#analysis) - [Collection](#collection) - [Event Registration and Recording](#event-registration-and-recording) -- [Metrics Overview](#metrics-overview) -- [Non-Event Metrics](#non-event-metrics) +- [Scalar Metrics](#scalar-metrics) - [List of Events Currently Recorded](#list-of-events-currently-recorded) +- [Sketch of iOS Telemetry Plan](#sketch-of-ios-telemetry-plan) - [References](#references) -This is the metrics collection plan for Lockbox's alpha release. It documents all events currently collected through telemetry, as well those planned for collection but not currently implemented. It will be updated to reflect all new and planned data collection. +This is the metrics collection plan for Lockbox. It documents all events currently collected through telemetry, as well those planned for collection but not currently implemented. It will be updated periodically to reflect all new and planned data collection. ## Analysis -Data collection is done solely for the purpose of product development, improvement and maintenance. We are particularly interested in data that will help us answer the following questions. +Data collection is done solely for the purpose of product development, improvement and maintenance. Specifically, it is done to help its creators examine the validity of the following hypothesis. + +Core Hypothesis: **We believe that people want the browser to do more than only remember their passwords.** + +We will know this to be true when: + +1. The password generator is clicked 20% of the time a new entry is created. **This will indicate that users value the ability to create secure passwords.** (Events regarding password generation will be added when feature development is complete) + +2. 75% of Lockbox downloads result in a Firefox account attached. **This will indicate that users value secure storage for their credentials.** We currently record the event `fxaUpgrade` to know whether a user has attached an Firefox account to their lockbox installation. + +3. 60% of users choose to import their existing credentials from Firefox into Lockbox. **This will indicate that users trust lockbox more than the browser for managing their credentials.** (Events regarding the importing of credentials from the firefox password manager will be added when feature development is complete) + +4. We observe increased engagement with the management system (Create-Read-Update-Delete; CRUD) for users that import their credentials. **This will indicate that users value greater visibility into the number of accounts that they have.** We currently record `render` events when a user opens the credential manager. + +5. CRUD usage rates are comparable to Firefox login manager usage prior to lockbox installation. **This will indicate that users value Lockbox's credential management system, and use it to access credentials when they are needed.** Firefox telemetry currently collects data on password autofill usage, which requires credentials be stored in the firefox password manager. We plan to compare lockbox credential usage (e.g. via the `usernameCopied` and `passwordCopied` lockbox events) to pre-lockbox autofill frequencies on a per-user basis. + +6. 20% of Lockbox users access their datastore on more than 1 device, **indicating that users value having a single datastore of credentials.** + + +Other questions we aim to answer through data collection, but are not directly related to the hypothesis above: - Do people Save Passwords in Lockbox? - - How many? (measured by count of items saved per user) - - How often? (number of credentials saved per user per time interval) + - How many? (measured by count of items saved per user) + - How often? (number of credentials saved per user per time interval) - Do people create their own passwords or use Lockbox to generate them? - - Ratio: (Number of times the PW generator is used when storing an item) / (number of credentials stored) + - Ratio: (Number of times the PW generator is used when storing an item) / (number of credentials stored) +- When using the pw generator, do people create purely random passwords or customize them with their own input? (if this is going to be in the final design) - Do people use the passwords they store on Lockbox? - - How many times (per some unit of time) do stored credentials get auto-filled or copied to the clipboard? - - How many times do users click to reveal a password? -- How well does does auto-filling credentials work **(not in alpha)**? - - When credentials are auto-filled, are they filled into the correct fields? - - How often must the user make a change to what fields were filled? - - How to measure these things??? is there an easier way than just asking if the auto-fill didn't work? + - How many times (per some unit of time) do stored credentials get filled? + - How many times (per some unit of time) do stored credentials get copied? +- How many times do users click to reveal a password? - Do people continue to use Lockbox after first use? - - Out of those who install, how many use it more than once? + - Out of those who install, how many use it more than once? - Where are the drop-off points in the user flow? - - Do the majority of people make it all the way through the setup process? - - Once initially setup, do people continue to add credentials? -- What are people's opinions on LB - - Feedback forms -- Do people sync their passwords **(not in alpha)**? - - How does syncing affect engagement? -- What type of sites do people use Lockbox with? - - Do they use Lockbox only for their most sensitive accounts? + - Do the majority of people make it all the way through the setup process? + - Once initially setup, do people continue to add credentials? +- Do people sync their passwords between Firefox instances? + - How does syncing affect engagement? -## Collection -**Note:** *This is the collection plan for our internal alpha release. For our beta release will be taking advantage of test pilot's telemetry API. The metrics plan for beta will be described in a separate document.* +## Collection At this point, all measurements related to Lockbox will be made client-side. However, future releases will give users the option to sync their Lockbox data via an FxA account, at which point additional measurements will be logged server-side through the FxA data pipeline. We are not directly responsible for the measurements made through that mechanism. -For our internal alpha release, we will be making use of the public JavaScript API that allows recording and sending of event data through an add-on. **This means that for our alpha release we will only be collecting event-based data**. The API is documented here: +For our internal alpha release, we will be making use of the public JavaScript API that allows recording and sending of event data and scalar data through an add-on. The API is documented here: https://firefox-source-docs.mozilla.org/toolkit/components/telemetry/telemetry/collection/events.html#the-api -Once the events are logged in the client they should appear in [about:telemetry](about:telemetry). From there they will be submitted in the main ping payload under `processes.dynamic.events` and available through the usual services (STMO and ATMO), as well as amplitude. +Once events are logged in the client they should appear in [about:telemetry](about:telemetry). From there they will be submitted in the main ping payload under `processes.dynamic.events` and available through the usual services (STMO and ATMO), as well as amplitude. ### Event Registration and Recording @@ -82,7 +94,6 @@ Services.telemetry.registerEvents("event_category", { For our purposes, we will use the `extra` field for a few purposes: -- To log the FxA user id of the client logging the event (e.g. `"fxauid": uid`) - To log the UUID of the item that has been added or changed (e.g. `"item_id": UUID`) - To log the fields that are modified when an item is updated in the datastore (e.g. `"fields": "password,notes"` (because the value has to be a string we will have to concat the fields that were updated somehow) @@ -94,37 +105,39 @@ When recording, we can use `null` for `value`. See the Events section for specific examples of event registration and recording. -## Metrics Overview - -(this section may redundant with analysis section) -For alpha, we'd like to (ideally) like to be able to track the following general categories of things: +#### Scalar recording -- The setup flow, so we can know at what points (if any) people quit the flow before finishing it -- Top-level interactions centered around use of the Lockbox toolbar icon. This includes interactions within the initial doorhanger that is displayed when the user clicks the icon. -- Interactions with the list of the user's Lockbox items (credentials) -- Interactions with the add / modify dialogs used to enter / edit item information -- Changes to the datastore that actually contains the user's items, in addition to user actions that lead to those changes -- When the user submits feedback about Lockbox -- Usage of the copy and reveal functions for stored Lockbox items. +We use the js api for scalar recording as well. Here registration happens with the following syntax: -Each of these are described below within their own Events subsection. - - -## Non-Event Metrics +```javascript +Services.telemetry.registerScalars(category, { + "scalar_name": { + kind: services.Telemetry.SCALAR_TYPE_COUNT, // SCALAR_TYPE_COUNT, SCALAR_TYPE_BOOLEAN. or SCALAR_TYPE_STRING + keyed: false, + record_on_release: false, // NEEDS TO BE SET TO RECORD ON RELEASE CHANNEL + expired: false, + } +``` +We set scalar values in the following way: -These are the metrics we plan to collect regarding the state of user datastores. Note that we won't be able to record these directly for alpha. We will have to infer them from the event data. +```javascript +Services.telemetry.scalarSet( + "category.scalar_name", value +); +``` +e.g. `lockboxV1.datastoreCount` for the scalar name. -- `n_items` The number of credentials that exist in the user's datastore. Integer +We can also use `scalarAdd` to increment a scalar value by some amount. -- `n_notes` The number of items for which the user has manually entered custom notes for. Integer +## Scalar Metrics -- `timestamp_last` The timestamp of the last edit the user made to the datastore. Does not necessarily correspond to the last time they opened the CRUD editor. +These are the metrics we currently collect regarding the state of user datastores. -- `n_uses` The number of times a user copied or auto-filled a Lockbox item. +- `datastoreCount` (integer). Current count of the number of items in the user's datastore. Note that this scalar is only updated when the user renders their full item list, either in the management view or in the doorhanger. So when testing whether this scalar is accurately updated, please re-render the item list. ## List of Events Currently Recorded -All events are currently implemented under the **category: lockboxV0**. The `extra` field always contains `fxauid` where possible (i.e. after FxA auth). For events pertaining to a particular Lockbox item, `itemid` is also included. They are listed and grouped together based on the contents of the event's `method` field. +All events are currently implemented under the **category: lockboxV2**. The `extra` field contains `itemid` for events pertaining to a particular Lockbox item. They are listed and grouped together below based on the contents of the event's `method` field. 1. `startup` fires when the webextension is loaded. **objects**: webextension. Note that this event fires whenever the browser is started, so is not indicative of direct user interaction with Lockbox. @@ -150,20 +163,35 @@ All events are currently implemented under the **category: lockboxV0**. The `ext 12. `resetCompleted` fires when the user completes a reset of their Lockbox data in the Lockbox settings. **objects**: settings -## List of Planned Events +13. `fxaStart` fires when a user clicks the sign-in or sign-up button from the manager or firstrun screen. used to log user initiating fxa auth process. **objects**: welcomeSignin, manageAcctCreate, manageAcctSignin, unlockSignin + +14. `fxaAuth` has methods `fxaUpgrade`, `fxaSignin` and `fxaSignout` that fire when the user initially adds their fxa account to lockbox, signs in after having added their fxa account previously, or signs out of fxa. **object**: accounts + +15. `fxaFail` fires when the fxa process fails **objects**: accounts + +## Sketch of iOS Telemetry Plan + +These events are based on the invisionapp design plan + +1. `fxaStart` fires when a user taps the sign-in or sign-up button from the welcome screen. used to log user initiating fxa auth process. + +2. `fxaAuth` has methods `fxaSignin` and `fxaSignout` that fire when the user signs into or out of their fxa account. + +3. `touchIdEnabled`, `faceIdEnabled`, `touchIdSkipped`, `faceIdSkipped` fire when a user successfully authorizes the use of touchID/faceID to unlock lockbox or taps to skip. -These events will be implemented once the corresponding functionality is available. +4. `browserEnabled` fires when user successfully adds lockbox integration to their browser. **objects**: `firefox`, `chrome`, `safari` -1. `fxaSignIn` fires when someone clicks the signin button during firstrun. **objects**: fxaSignInPage. +5. `usernameCopied` and `passwordCopied` fire when a user copies their username or password from an item. **objects**: itemList -2. `confirmPW` fires when the user clicks the button to confirm their FxA pw. **objects**: confirmPWButton +6. `itemSelected` fires when a user taps an item in the itemlist. **objects** itemList -3. `doorhangerItemSelected` fires when a user clicks an item in the doorhanger's itemlist. **objects** doorhanger +7. `itemShow` fires when a user taps to view item details in the itemlist. **objects** itemList -4. `doorhangerItemCopied` fires when a user copies an item's username/password from the doorhanger's entry view. **objects** doorhangerEntryDetails. +8. `settingsTap` fires when a user taps an entry on the settings page. **object** will label the setting that was tapped -5. `doorhangerAddClick` fires when the user clicks to add a new entry from the doorhanger. **objects** doorhanger +9. `settingsChanged` fires when a user toggles a changeable setting. **object** will label the setting that was toggled +10. `feedbackSent` Fires when a user sends feedback through the feedback form. --- diff --git a/docs/release-notes.md b/docs/release-notes.md index 40192c23..67ec2e72 100644 --- a/docs/release-notes.md +++ b/docs/release-notes.md @@ -1,5 +1,78 @@ # Lockbox Release Notes +## 0.1.8-alpha + +_Date: 2018-03-08_ + +### What's New + +- Added support For Telemetry Scalars so we can easily tell how many items are stored in your Lockbox datastore ([#552](https://github.com/mozilla-lockbox/lockbox-extension/pull/552)) +- Made it so the search box is automatically focused so you can start typing to search your entries immediately ([#573](https://github.com/mozilla-lockbox/lockbox-extension/issues/573)) +- The doorhanger also automatically fills the search with the domain of the current page you're viewing to help you quickly see matching entries ([#616](https://github.com/mozilla-lockbox/lockbox-extension/pull/616)) +- Improved contrast and colors of the entry list for accessibility ([#560](https://github.com/mozilla-lockbox/lockbox-extension/pull/560)) +- Improved interface colors and styles for entry list items ([#553](https://github.com/mozilla-lockbox/lockbox-extension/pull/553)) +- Improved search bar interface style with a simpler "Clear" button ([#562](https://github.com/mozilla-lockbox/lockbox-extension/pull/562)) +- Re-ordered the create/edit entry interface with improved spacing and styles ([#567](https://github.com/mozilla-lockbox/lockbox-extension/pull/567)) + +### What's Fixed + +- Item notes with line breaks properly show those separate lines when editing ([#596](https://github.com/mozilla-lockbox/lockbox-extension/pull/596)) +- The "reset Lockbox" settings interface fits better in about:addons ([#603](https://github.com/mozilla-lockbox/lockbox-extension/pull/603)) +- Fixed alignment on the entry detail view so the buttons don't jump when switching between edit and view modes ([#608](https://github.com/mozilla-lockbox/lockbox-extension/pull/608)) + +### Known Issues + +* Profile information about you is only fetched and updated when you sign in; any changes made to your Firefox Accounts display name or avatar will not be displayed in Lockbox until you sign in again. +* Once you link a Firefox Account to Lockbox, you cannot unlink it from that account. +* Once you link a Firefox Account to Lockbox, signing in with a different account can render Lockbox unusable until you quit and restart Firefox. +* Once you link a Firefox Account to Lockbox, resetting your Firefox Account password through "forgot your password" will render all your logins inaccessible; the only recourse is to reset Lockbox and start over. +* Firefox's default prompt to save logins is only disabled on new installs of this extension; updating Lockbox will not change your current Firefox preferences. + +## 0.1.7-alpha1 + +_Date: 2018-02-23_ + +This release is to address a critical defect from the previous release. + +### What's Fixed + +* If you linked your Firefox Account, Lockbox would stop working when you restart Firefox ([#568](https://github.com/mozilla-lockbox/lockbox-extension/issues/568)) + +### Known Issues + +* Profile information about you is only fetched and updated when you sign in; any changes made to your Firefox Accounts display name or avatar will not be displayed in Lockbox until you sign in again. +* Once you link a Firefox Account to Lockbox, you cannot unlink it from that account. +* Once you link a Firefox Account to Lockbox, signing in with a different account can render Lockbox unusable until you quit and restart Firefox. +* Once you link a Firefox Account to Lockbox, resetting your Firefox Account password through "forgot your password" will render all your logins inaccessible; the only recourse is to reset Lockbox and start over. +* Firefox's default prompt to save logins is only disabled on new installs of this extension; updating Lockbox will not change your current Firefox preferences. + + +## 0.1.7-alpha + +_Date: 2018-02-22_ + +### What's New + +* To provide you with a closer brand experience with Firefox, the visual design and interaction of the account actions dropdown in the full editor view better aligns with Firefox's [Photon design](https://design.firefox.com/photon) ([#524](https://github.com/mozilla-lockbox/lockbox-extension/issues/524)) + +### What's Fixed + +* If you use Lockbox in "Guest" mode (not linked to a Firefox Account), it can be unusable when Firefox is next restarted ([#542](https://github.com/mozilla-lockbox/lockbox-extension/issues/542)) +* Via Greenkeeper: + * Updated `events` dependency ([#509](https://github.com/mozilla-lockbox/lockbox-extension/pull/509)) + * Updated `babel-minify-webpack-plugin` dependency ([#525](https://github.com/mozilla-lockbox/lockbox-extension/pull/525)) + * Updated `eslint-plugin-node` dependency ([#518](https://github.com/mozilla-lockbox/lockbox-extension/pull/518)) + * Updated `eslint-plugin-mozilla` dependency ([#546](https://github.com/mozilla-lockbox/lockbox-extension/pull/546)) + +### Known Issues + +* Profile information about you is only fetched and updated when you sign in; any changes made to your Firefox Accounts display name or avatar will not be displayed in Lockbox until you sign in again. +* Once you link a Firefox Account to Lockbox, you cannot unlink it from that account. +* Once you link a Firefox Account to Lockbox, signing in with a different account can render Lockbox unusable until you quit and restart Firefox. +* Once you link a Firefox Account to Lockbox, resetting your Firefox Account password through "forgot your password" will render all your logins inaccessible; the only recourse is to reset Lockbox and start over. +* Firefox's default prompt to save logins is only disabled on new installs of this extension; updating Lockbox will not change your current Firefox preferences. + + ## 0.1.6-alpha _Date: 2018-02-08_ diff --git a/package-lock.json b/package-lock.json index 458783b4..3c504406 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "lockbox", - "version": "0.1.6-alpha", + "version": "0.1.8-alpha", "lockfileVersion": 1, "requires": true, "dependencies": { @@ -4979,68 +4979,70 @@ } }, "documentation": { - "version": "5.3.5", - "resolved": "https://registry.npmjs.org/documentation/-/documentation-5.3.5.tgz", - "integrity": "sha512-VApv5dCFxjD5WZxEYGqZypK7L3l00clw9qbSTYCwdnvsaAhu7/U+FCHvkdtnCtkmLupIF/SxXPfXc4kB3UfXEg==", + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/documentation/-/documentation-6.0.0.tgz", + "integrity": "sha512-7oqdQdk1+LpfXLd0N7oFSr0JteWvcEy21VziBP+YYFO7EBFuW7zidsLQ1ponKr3IXPRfLz9cVLBHrdqpXAKQ0g==", "dev": true, "requires": { "ansi-html": "0.0.7", "babel-core": "6.26.0", - "babel-generator": "6.25.0", + "babel-generator": "6.26.1", "babel-plugin-system-import-transformer": "3.1.0", "babel-plugin-transform-decorators-legacy": "1.3.4", "babel-preset-env": "1.6.1", "babel-preset-react": "6.24.1", "babel-preset-stage-0": "6.24.1", - "babel-traverse": "6.25.0", - "babel-types": "6.25.0", + "babel-traverse": "6.26.0", + "babel-types": "6.26.0", "babelify": "8.0.0", - "babylon": "6.17.4", - "chalk": "2.3.0", - "chokidar": "1.7.0", + "babylon": "6.18.0", + "chalk": "2.3.2", + "chokidar": "2.0.2", "concat-stream": "1.6.0", "disparity": "2.0.0", "doctrine-temporary-fork": "2.0.0-alpha-allowarrayindex", "get-port": "3.2.0", - "git-url-parse": "6.2.2", + "git-url-parse": "8.1.0", "github-slugger": "1.2.0", "glob": "7.1.2", "globals-docs": "2.4.0", "highlight.js": "9.12.0", "js-yaml": "3.10.0", - "lodash": "4.11.1", + "lodash": "4.17.5", "mdast-util-inject": "1.1.0", - "micromatch": "3.1.5", - "mime": "1.6.0", + "micromatch": "3.1.9", + "mime": "2.2.0", "module-deps-sortable": "4.0.6", "parse-filepath": "1.0.2", "pify": "3.0.0", "read-pkg-up": "3.0.0", - "remark": "8.0.0", + "remark": "9.0.0", "remark-html": "7.0.0", - "remark-toc": "4.0.1", + "remark-reference-links": "4.0.1", + "remark-toc": "5.0.0", "remote-origin-url": "0.4.0", - "shelljs": "0.7.8", + "shelljs": "0.8.1", "stream-array": "1.1.2", "strip-json-comments": "2.0.1", - "tiny-lr": "1.0.5", + "tiny-lr": "1.1.1", "unist-builder": "1.0.2", "unist-util-visit": "1.3.0", "vfile": "2.3.0", "vfile-reporter": "4.0.0", "vfile-sort": "2.1.0", "vinyl": "2.1.0", - "vinyl-fs": "3.0.1", + "vinyl-fs": "3.0.2", "yargs": "9.0.1" }, "dependencies": { - "ansi-styles": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.0.tgz", - "integrity": "sha512-NnSOmMEYtVR2JVMIGTzynRkkaxtiq1xnFBcdQD/DnNCYPoEPsVJhM98BDyaoNOQIi7p4okdi3E27eN7GQbsUug==", + "anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", "dev": true, "requires": { - "color-convert": "1.9.0" + "micromatch": "3.1.9", + "normalize-path": "2.1.1" } }, "arr-diff": { @@ -5055,10 +5057,97 @@ "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", "dev": true }, + "babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", + "dev": true, + "requires": { + "chalk": "1.1.3", + "esutils": "2.0.2", + "js-tokens": "3.0.2" + }, + "dependencies": { + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "2.2.1", + "escape-string-regexp": "1.0.5", + "has-ansi": "2.0.0", + "strip-ansi": "3.0.1", + "supports-color": "2.0.0" + } + } + } + }, + "babel-generator": { + "version": "6.26.1", + "resolved": "https://registry.npmjs.org/babel-generator/-/babel-generator-6.26.1.tgz", + "integrity": "sha512-HyfwY6ApZj7BYTcJURpM5tznulaBvyio7/0d4zFOeMPUmfxkCjHocCuoLa2SAGzBI8AREcH3eP3758F672DppA==", + "dev": true, + "requires": { + "babel-messages": "6.23.0", + "babel-runtime": "6.26.0", + "babel-types": "6.26.0", + "detect-indent": "4.0.0", + "jsesc": "1.3.0", + "lodash": "4.17.5", + "source-map": "0.5.7", + "trim-right": "1.0.1" + } + }, + "babel-runtime": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-runtime/-/babel-runtime-6.26.0.tgz", + "integrity": "sha1-llxwWGaOgrVde/4E/yM3vItWR/4=", + "dev": true, + "requires": { + "core-js": "2.4.1", + "regenerator-runtime": "0.11.0" + } + }, + "babel-traverse": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", + "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", + "dev": true, + "requires": { + "babel-code-frame": "6.26.0", + "babel-messages": "6.23.0", + "babel-runtime": "6.26.0", + "babel-types": "6.26.0", + "babylon": "6.18.0", + "debug": "2.6.8", + "globals": "9.18.0", + "invariant": "2.2.2", + "lodash": "4.17.5" + } + }, + "babel-types": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-types/-/babel-types-6.26.0.tgz", + "integrity": "sha1-o7Bz+Uq0nrb6Vc1lInozQ4BjJJc=", + "dev": true, + "requires": { + "babel-runtime": "6.26.0", + "esutils": "2.0.2", + "lodash": "4.17.5", + "to-fast-properties": "1.0.3" + } + }, + "babylon": { + "version": "6.18.0", + "resolved": "https://registry.npmjs.org/babylon/-/babylon-6.18.0.tgz", + "integrity": "sha512-q/UEjfGJ2Cm3oKV71DJz9d25TPnq5rhBVL2Q4fA5wcC3jcrdn7+SssEybFIxwAvvP+YCsCYNKughoF33GxgycQ==", + "dev": true + }, "braces": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.0.tgz", - "integrity": "sha512-P4O8UQRdGiMLWSizsApmXVQDBS6KCt7dSexgLKBmH5Hr1CZq7vsnscFh8oR1sP1ab1Zj0uCHCEzZeV6SfUf3rA==", + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.1.tgz", + "integrity": "sha512-SO5lYHA3vO6gz66erVvedSCkp7AKWdv6VcQ2N4ysXfPxdAlxAMMAdwegGGcv1Bqwm7naF1hNdk5d6AAIEHV2nQ==", "dev": true, "requires": { "arr-flatten": "1.1.0", @@ -5067,22 +5156,93 @@ "extend-shallow": "2.0.1", "fill-range": "4.0.0", "isobject": "3.0.1", + "kind-of": "6.0.2", "repeat-element": "1.1.2", "snapdragon": "0.8.1", "snapdragon-node": "2.1.1", "split-string": "3.1.0", "to-regex": "3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "1.0.2" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + } } }, "chalk": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.3.0.tgz", - "integrity": "sha512-Az5zJR2CBujap2rqXGaJKaPHyJ0IrUimvYNX+ncCy8PJP4ltOGTrHUIo097ZaL2zMeKYpiCdqDvS6zdrTFok3Q==", + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.3.2.tgz", + "integrity": "sha512-ZM4j2/ld/YZDc3Ma8PgN7gyAk+kHMMMyzLNryCPGhWrsfAuDVeuid5bpRFTDgMH9JBK2lA4dyyAkkZYF/WcqDQ==", "dev": true, "requires": { - "ansi-styles": "3.2.0", + "ansi-styles": "3.2.1", "escape-string-regexp": "1.0.5", - "supports-color": "4.5.0" + "supports-color": "5.3.0" + }, + "dependencies": { + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "1.9.0" + } + }, + "supports-color": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.3.0.tgz", + "integrity": "sha512-0aP01LLIskjKs3lq52EC0aGBAJhLq7B2Rd8HC/DR/PtNNpcLilNmHC12O+hu0usQpo7wtHNRqtrhBwtDb0+dNg==", + "dev": true, + "requires": { + "has-flag": "3.0.0" + } + } + } + }, + "chokidar": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.0.2.tgz", + "integrity": "sha512-l32Hw3wqB0L2kGVmSbK/a+xXLDrUEsc84pSgMkmwygHvD7ubRsP/vxxHa5BtB6oix1XLLVCHyYMsckRXxThmZw==", + "dev": true, + "requires": { + "anymatch": "2.0.0", + "async-each": "1.0.1", + "braces": "2.3.1", + "fsevents": "1.1.3", + "glob-parent": "3.1.0", + "inherits": "2.0.3", + "is-binary-path": "1.0.1", + "is-glob": "4.0.0", + "normalize-path": "2.1.1", + "path-is-absolute": "1.0.1", + "readdirp": "2.1.0", + "upath": "1.0.4" + } + }, + "define-property": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", + "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", + "dev": true, + "requires": { + "is-descriptor": "1.0.2", + "isobject": "3.0.1" } }, "expand-brackets": { @@ -5108,6 +5268,53 @@ "requires": { "is-descriptor": "0.1.6" } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + }, + "is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "0.1.6", + "is-data-descriptor": "0.1.4", + "kind-of": "5.1.0" + } + }, + "kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true + } + } + }, + "extend-shallow": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", + "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", + "dev": true, + "requires": { + "assign-symbols": "1.0.0", + "is-extendable": "1.0.1" + }, + "dependencies": { + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "2.0.4" + } } } }, @@ -5125,6 +5332,26 @@ "regex-not": "1.0.0", "snapdragon": "0.8.1", "to-regex": "3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "1.0.2" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + } } }, "fill-range": { @@ -5137,12 +5364,53 @@ "is-number": "3.0.0", "repeat-string": "1.6.1", "to-regex-range": "2.1.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + } + } + }, + "git-url-parse": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/git-url-parse/-/git-url-parse-8.1.0.tgz", + "integrity": "sha512-tSdNasqIc9cjK75DRsirb5sqVJ4V4cCmCuuOyyx2SuYeJx4o9AOx+/ZCSwRrYjZ8zavtuhGjCqXlCo9Db0YIVA==", + "dev": true, + "requires": { + "git-up": "2.0.10" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "3.1.0", + "path-dirname": "1.0.2" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "2.1.1" + } + } } }, "has-flag": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-2.0.0.tgz", - "integrity": "sha1-6CB68cx7MNRGzHC3NLXovhj4jVE=", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", "dev": true }, "is-accessor-descriptor": { @@ -5185,23 +5453,19 @@ } } }, - "is-descriptor": { - "version": "0.1.6", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", - "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true + }, + "is-glob": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.0.tgz", + "integrity": "sha1-lSHHaEXMJhCoUgPd8ICpWML/q8A=", "dev": true, "requires": { - "is-accessor-descriptor": "0.1.6", - "is-data-descriptor": "0.1.4", - "kind-of": "5.1.0" - }, - "dependencies": { - "kind-of": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", - "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", - "dev": true - } + "is-extglob": "2.1.1" } }, "is-number": { @@ -5224,6 +5488,35 @@ } } }, + "is-odd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-odd/-/is-odd-2.0.0.tgz", + "integrity": "sha512-OTiixgpZAT1M4NHgS5IguFp/Vz2VI3U7Goh4/HA1adtwyLtSBrxYlcSYkhpAE07s4fKEcjrFxyvtQBND4vFQyQ==", + "dev": true, + "requires": { + "is-number": "4.0.0" + }, + "dependencies": { + "is-number": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-4.0.0.tgz", + "integrity": "sha512-rSklcAIlf1OmFdyAqbnWTLVelsQ58uvZ66S/ZyawjWqIviTWCjg2PzVGw8WUA+nNuPTqb4wgA+NszrJ+08LlgQ==", + "dev": true + } + } + }, + "is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "dev": true + }, + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", + "dev": true + }, "isobject": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", @@ -5248,21 +5541,59 @@ "strip-bom": "3.0.0" } }, + "lodash": { + "version": "4.17.5", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.5.tgz", + "integrity": "sha512-svL3uiZf1RwhH+cWrfZn3A4+U58wbP0tGVTLQPbjplZxZ8ROD9VLuNgsRniTlLe7OlSqR79RUehXgpBW/s0IQw==", + "dev": true + }, "micromatch": { - "version": "3.1.5", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.5.tgz", - "integrity": "sha512-ykttrLPQrz1PUJcXjwsTUjGoPJ64StIGNE2lGVD1c9CuguJ+L7/navsE8IcDNndOoCMvYV0qc/exfVbMHkUhvA==", + "version": "3.1.9", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.9.tgz", + "integrity": "sha512-SlIz6sv5UPaAVVFRKodKjCg48EbNoIhgetzfK/Cy0v5U52Z6zB136M8tp0UC9jM53LYbmIRihJszvvqpKkfm9g==", "dev": true, "requires": { "arr-diff": "4.0.0", "array-unique": "0.3.2", - "braces": "2.3.0", - "define-property": "1.0.0", - "extend-shallow": "2.0.1", + "braces": "2.3.1", + "define-property": "2.0.2", + "extend-shallow": "3.0.2", "extglob": "2.0.4", "fragment-cache": "0.2.1", "kind-of": "6.0.2", - "nanomatch": "1.2.7", + "nanomatch": "1.2.9", + "object.pick": "1.3.0", + "regex-not": "1.0.0", + "snapdragon": "0.8.1", + "to-regex": "3.0.1" + } + }, + "mime": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.2.0.tgz", + "integrity": "sha512-0Qz9uF1ATtl8RKJG4VRfOymh7PyEor6NbrI/61lRfuRe4vx9SNATrvAeTj2EWVRKjEQGskrzWkJBBY5NbaVHIA==", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "nanomatch": { + "version": "1.2.9", + "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.9.tgz", + "integrity": "sha512-n8R9bS8yQ6eSXaV6jHUpKzD8gLsin02w1HSFiegwrs9E098Ylhw5jdyKPaYqvHknHaSCKTPp7C8dGCQ0q9koXA==", + "dev": true, + "requires": { + "arr-diff": "4.0.0", + "array-unique": "0.3.2", + "define-property": "2.0.2", + "extend-shallow": "3.0.2", + "fragment-cache": "0.2.1", + "is-odd": "2.0.0", + "is-windows": "1.0.2", + "kind-of": "6.0.2", "object.pick": "1.3.0", "regex-not": "1.0.0", "snapdragon": "0.8.1", @@ -5294,6 +5625,18 @@ "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", "dev": true }, + "process-nextick-args": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.0.tgz", + "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw==", + "dev": true + }, + "qs": { + "version": "6.5.1", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz", + "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==", + "dev": true + }, "read-pkg": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-3.0.0.tgz", @@ -5315,19 +5658,178 @@ "read-pkg": "3.0.0" } }, + "readable-stream": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.5.tgz", + "integrity": "sha512-tK0yDhrkygt/knjowCUiWP9YdV7c5R+8cR0r/kt9ZhBU906Fs6RpQJCEilamRJj1Nx2rWI6LkW9gKqjTkshhEw==", + "dev": true, + "requires": { + "core-util-is": "1.0.2", + "inherits": "2.0.3", + "isarray": "1.0.0", + "process-nextick-args": "2.0.0", + "safe-buffer": "5.1.1", + "string_decoder": "1.0.3", + "util-deprecate": "1.0.2" + } + }, + "remark": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/remark/-/remark-9.0.0.tgz", + "integrity": "sha512-amw8rGdD5lHbMEakiEsllmkdBP+/KpjW/PRK6NSGPZKCQowh0BT4IWXDAkRMyG3SB9dKPXWMviFjNusXzXNn3A==", + "dev": true, + "requires": { + "remark-parse": "5.0.0", + "remark-stringify": "5.0.0", + "unified": "6.1.6" + } + }, + "remark-parse": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/remark-parse/-/remark-parse-5.0.0.tgz", + "integrity": "sha512-b3iXszZLH1TLoyUzrATcTQUZrwNl1rE70rVdSruJFlDaJ9z5aMkhrG43Pp68OgfHndL/ADz6V69Zow8cTQu+JA==", + "dev": true, + "requires": { + "collapse-white-space": "1.0.3", + "is-alphabetical": "1.0.1", + "is-decimal": "1.0.1", + "is-whitespace-character": "1.0.1", + "is-word-character": "1.0.1", + "markdown-escapes": "1.0.1", + "parse-entities": "1.1.1", + "repeat-string": "1.6.1", + "state-toggle": "1.0.0", + "trim": "0.0.1", + "trim-trailing-lines": "1.1.0", + "unherit": "1.1.0", + "unist-util-remove-position": "1.1.1", + "vfile-location": "2.0.2", + "xtend": "4.0.1" + } + }, + "remark-slug": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/remark-slug/-/remark-slug-5.0.0.tgz", + "integrity": "sha512-bRFK90ia6iooqC5KH6e9nEIL3OwRbTPU6ed2fm/fa66uofKdmRcsmRVMwND3pXLbvH2F022cETYlE7YlVs7LNQ==", + "dev": true, + "requires": { + "github-slugger": "1.2.0", + "mdast-util-to-string": "1.0.4", + "unist-util-visit": "1.3.0" + } + }, + "remark-stringify": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/remark-stringify/-/remark-stringify-5.0.0.tgz", + "integrity": "sha512-Ws5MdA69ftqQ/yhRF9XhVV29mhxbfGhbz0Rx5bQH+oJcNhhSM6nCu1EpLod+DjrFGrU0BMPs+czVmJZU7xiS7w==", + "dev": true, + "requires": { + "ccount": "1.0.2", + "is-alphanumeric": "1.0.0", + "is-decimal": "1.0.1", + "is-whitespace-character": "1.0.1", + "longest-streak": "2.0.2", + "markdown-escapes": "1.0.1", + "markdown-table": "1.1.1", + "mdast-util-compact": "1.0.1", + "parse-entities": "1.1.1", + "repeat-string": "1.6.1", + "state-toggle": "1.0.0", + "stringify-entities": "1.3.1", + "unherit": "1.1.0", + "xtend": "4.0.1" + } + }, + "remark-toc": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/remark-toc/-/remark-toc-5.0.0.tgz", + "integrity": "sha512-j2A/fpio1nzNRJtY6nVaFUCtXNfFPxaj6I5UHFsFgo4xKmc0VokRRIzGqz4Vfs7u+dPrHjnoHkImu1Dia0jDSQ==", + "dev": true, + "requires": { + "mdast-util-toc": "2.0.1", + "remark-slug": "5.0.0" + } + }, + "shelljs": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/shelljs/-/shelljs-0.8.1.tgz", + "integrity": "sha512-YA/iYtZpzFe5HyWVGrb02FjPxc4EMCfpoU/Phg9fQoyMC72u9598OUBrsU8IrtwAKG0tO8IYaqbaLIw+k3IRGA==", + "dev": true, + "requires": { + "glob": "7.1.2", + "interpret": "1.1.0", + "rechoir": "0.6.2" + } + }, + "source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "dev": true + }, + "string_decoder": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.0.3.tgz", + "integrity": "sha512-4AH6Z5fzNNBcH+6XDMfA/BTt87skxqJlO0lAh3Dker5zThcAxG6mKz+iGu308UKoPPQ8Dcqx/4JhujzltRa+hQ==", + "dev": true, + "requires": { + "safe-buffer": "5.1.1" + } + }, "strip-bom": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", "dev": true }, - "supports-color": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-4.5.0.tgz", - "integrity": "sha1-vnoN5ITexcXN34s9WRJQRJEvY1s=", + "tiny-lr": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/tiny-lr/-/tiny-lr-1.1.1.tgz", + "integrity": "sha512-44yhA3tsaRoMOjQQ+5v5mVdqef+kH6Qze9jTpqtVufgYjYt08zyZAwNwwVBj3i1rJMnR52IxOW0LK0vBzgAkuA==", "dev": true, "requires": { - "has-flag": "2.0.0" + "body": "5.1.0", + "debug": "3.1.0", + "faye-websocket": "0.10.0", + "livereload-js": "2.3.0", + "object-assign": "4.1.0", + "qs": "6.5.1" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + } + } + }, + "vinyl-fs": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/vinyl-fs/-/vinyl-fs-3.0.2.tgz", + "integrity": "sha512-AUSFda1OukBwuLPBTbyuO4IRWgfXmqC4UTW0f8xrCa8Hkv9oyIU+NSqBlgfOLZRoUt7cHdo75hKQghCywpIyIw==", + "dev": true, + "requires": { + "fs-mkdirp-stream": "1.0.0", + "glob-stream": "6.1.0", + "graceful-fs": "4.1.11", + "is-valid-glob": "1.0.0", + "lazystream": "1.0.0", + "lead": "1.0.0", + "object.assign": "4.1.0", + "pumpify": "1.4.0", + "readable-stream": "2.3.5", + "remove-bom-buffer": "3.0.0", + "remove-bom-stream": "1.2.0", + "resolve-options": "1.1.0", + "through2": "2.0.3", + "to-through": "2.0.0", + "value-or-function": "3.0.0", + "vinyl": "2.1.0", + "vinyl-sourcemap": "1.1.0" } } } @@ -8433,15 +8935,6 @@ "parse-url": "1.3.11" } }, - "git-url-parse": { - "version": "6.2.2", - "resolved": "https://registry.npmjs.org/git-url-parse/-/git-url-parse-6.2.2.tgz", - "integrity": "sha1-vkkCThS4SHVTQ2tFcri0OVMvqHE=", - "dev": true, - "requires": { - "git-up": "2.0.10" - } - }, "github-slugger": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/github-slugger/-/github-slugger-1.2.0.tgz", @@ -9824,26 +10317,6 @@ "integrity": "sha1-PkcprB9f3gJc19g6iW2rn09n2w8=", "dev": true }, - "is-odd": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-odd/-/is-odd-1.0.0.tgz", - "integrity": "sha1-O4qTLrAos3dcObsJ6RdnrM22kIg=", - "dev": true, - "requires": { - "is-number": "3.0.0" - }, - "dependencies": { - "is-number": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", - "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", - "dev": true, - "requires": { - "kind-of": "3.2.2" - } - } - } - }, "is-path-cwd": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz", @@ -11855,45 +12328,6 @@ "dev": true, "optional": true }, - "nanomatch": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.7.tgz", - "integrity": "sha512-/5ldsnyurvEw7wNpxLFgjVvBLMta43niEYOy0CJ4ntcYSbx6bugRUTQeFb4BR/WanEL1o3aQgHuVLHQaB6tOqg==", - "dev": true, - "requires": { - "arr-diff": "4.0.0", - "array-unique": "0.3.2", - "define-property": "1.0.0", - "extend-shallow": "2.0.1", - "fragment-cache": "0.2.1", - "is-odd": "1.0.0", - "kind-of": "5.1.0", - "object.pick": "1.3.0", - "regex-not": "1.0.0", - "snapdragon": "0.8.1", - "to-regex": "3.0.1" - }, - "dependencies": { - "arr-diff": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", - "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", - "dev": true - }, - "array-unique": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", - "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", - "dev": true - }, - "kind-of": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", - "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", - "dev": true - } - } - }, "natural-compare": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", @@ -15040,14 +15474,12 @@ "xtend": "4.0.1" } }, - "remark-slug": { - "version": "4.2.3", - "resolved": "https://registry.npmjs.org/remark-slug/-/remark-slug-4.2.3.tgz", - "integrity": "sha1-jZh9Dl5j1KSeo3uQ/pmaPc/IG3I=", + "remark-reference-links": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/remark-reference-links/-/remark-reference-links-4.0.1.tgz", + "integrity": "sha1-AhrtHFXBh9cSs8dtAFe/UQ0wC6c=", "dev": true, "requires": { - "github-slugger": "1.2.0", - "mdast-util-to-string": "1.0.4", "unist-util-visit": "1.3.0" } }, @@ -15073,16 +15505,6 @@ "xtend": "4.0.1" } }, - "remark-toc": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/remark-toc/-/remark-toc-4.0.1.tgz", - "integrity": "sha1-/zb/beVOoH3Vnj9TNKSjqsHpMYU=", - "dev": true, - "requires": { - "mdast-util-toc": "2.0.1", - "remark-slug": "4.2.3" - } - }, "remote-origin-url": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/remote-origin-url/-/remote-origin-url-0.4.0.tgz", @@ -15599,17 +16021,6 @@ "jsonify": "0.0.0" } }, - "shelljs": { - "version": "0.7.8", - "resolved": "https://registry.npmjs.org/shelljs/-/shelljs-0.7.8.tgz", - "integrity": "sha1-3svPh0sNHl+3LhSxZKloMEjprLM=", - "dev": true, - "requires": { - "glob": "7.1.2", - "interpret": "1.1.0", - "rechoir": "0.6.2" - } - }, "sign-addon": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/sign-addon/-/sign-addon-0.2.0.tgz", @@ -17495,28 +17906,6 @@ "dev": true, "optional": true }, - "tiny-lr": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/tiny-lr/-/tiny-lr-1.0.5.tgz", - "integrity": "sha512-YrxUSiMgOVh3PnAqtdAUQuUVEVRnqcRCxJ3BHrl/aaWV2fplKKB60oClM0GH2Gio2hcXvkxMUxsC/vXZrQePlg==", - "dev": true, - "requires": { - "body": "5.1.0", - "debug": "2.6.8", - "faye-websocket": "0.10.0", - "livereload-js": "2.3.0", - "object-assign": "4.1.0", - "qs": "6.5.1" - }, - "dependencies": { - "qs": { - "version": "6.5.1", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz", - "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==", - "dev": true - } - } - }, "tmp": { "version": "0.0.28", "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.28.tgz", @@ -18164,6 +18553,12 @@ } } }, + "upath": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/upath/-/upath-1.0.4.tgz", + "integrity": "sha512-d4SJySNBXDaQp+DPrziv3xGS6w3d2Xt69FijJr86zMPBy23JEloMCEOUBBzuN7xCtjLCnmB9tI/z7SBCahHBOw==", + "dev": true + }, "upper-case": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/upper-case/-/upper-case-1.1.3.tgz", @@ -18496,31 +18891,6 @@ } } }, - "vinyl-fs": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/vinyl-fs/-/vinyl-fs-3.0.1.tgz", - "integrity": "sha1-dK9faDahz0FNNe6zwQ8uZfwqLBA=", - "dev": true, - "requires": { - "flush-write-stream": "1.0.2", - "fs-mkdirp-stream": "1.0.0", - "glob-stream": "6.1.0", - "graceful-fs": "4.1.11", - "is-valid-glob": "1.0.0", - "lazystream": "1.0.0", - "lead": "1.0.0", - "object.assign": "4.1.0", - "pumpify": "1.4.0", - "remove-bom-buffer": "3.0.0", - "remove-bom-stream": "1.2.0", - "resolve-options": "1.1.0", - "through2": "2.0.3", - "to-through": "2.0.0", - "value-or-function": "3.0.0", - "vinyl": "2.1.0", - "vinyl-sourcemap": "1.1.0" - } - }, "vinyl-sourcemap": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/vinyl-sourcemap/-/vinyl-sourcemap-1.1.0.tgz", diff --git a/package.json b/package.json index 8700071b..0275d67f 100644 --- a/package.json +++ b/package.json @@ -2,7 +2,7 @@ "title": "Lockbox", "name": "lockbox", "id": "lockbox@mozilla.com", - "version": "0.1.6-alpha", + "version": "0.1.8-alpha", "main": "dist/bootstrap.js", "description": "The simple way to store, retrieve and manage website login info", "author": "Lockbox Team ", @@ -96,14 +96,14 @@ "copy-webpack-plugin": "^4.3.1", "cross-env": "^5.1.3", "css-loader": "^0.28.9", - "documentation": "^5.3.5", + "documentation": "^6.0.0", "enzyme": "^3.3.0", "enzyme-adapter-react-16": "^1.1.1", "eslint": "^4.15.0", "eslint-config-standard": "^11.0.0-beta.0", "eslint-plugin-import": "^2.8.0", "eslint-plugin-mozilla": "0.6.0", - "eslint-plugin-no-unsanitized": "^2.0.2", + "eslint-plugin-no-unsanitized": "^3.0.0", "eslint-plugin-node": "^6.0.0", "eslint-plugin-only-warn": "^1.0.1", "eslint-plugin-promise": "^3.6.0", @@ -124,7 +124,7 @@ "karma-mocha-reporter": "^2.2.5", "karma-sourcemap-loader": "^0.3.7", "karma-webpack": "^2.0.9", - "mocha": "^4.1.0", + "mocha": "^5.0.1", "nsp": "^3.1.0", "react-document-title": "^2.0.3", "react-test-renderer": "^16.2.0", @@ -133,8 +133,8 @@ "sinon-chai": "^2.14.0", "source-map-support": "^0.5.1", "style-loader": "^0.20.1", - "stylelint": "8.4.0", - "stylelint-config-recommended": "2.0.1", + "stylelint": "9.1.3", + "stylelint-config-recommended": "2.1.0", "webpack": "^3.10.0", "webpack-combine-loaders": "^2.0.3", "webpack-node-externals": "^1.6.0" diff --git a/requirements/flake8.txt b/requirements/flake8.txt index 29226fd5..88beaaac 100644 --- a/requirements/flake8.txt +++ b/requirements/flake8.txt @@ -1,3 +1,3 @@ flake8==3.5.0 -flake8-isort==2.2.2 -flake8-docstrings==1.1.0 +flake8-isort==2.5 +flake8-docstrings==1.3.0 diff --git a/requirements/tests.txt b/requirements/tests.txt index ad662a78..238d50a2 100644 --- a/requirements/tests.txt +++ b/requirements/tests.txt @@ -1,6 +1,6 @@ fxapom==1.10.1 -PyPOM==1.2.0 -pytest==3.3.1 -pytest-selenium==1.11.3 -pytest-xdist==1.20.1 -selenium==3.8.0 +pytest==3.4.1 +pytest-selenium==1.11.4 +pytest-xdist==1.22.2 +selenium==3.10.0 +pytest-html diff --git a/src/bootstrap.js b/src/bootstrap.js index fa785480..9fd955d0 100644 --- a/src/bootstrap.js +++ b/src/bootstrap.js @@ -57,82 +57,70 @@ function startup({webExtension}, reason) { "iconClick": { methods: ["iconClick"], objects: ["toolbar"], - extra_keys: ["fxauid"], }, "displayView": { methods: ["render"], objects: ["firstrun", "popupUnlock", "manage", "doorhanger"], - extra_keys: ["fxauid"], }, "itemAdding": { methods: ["itemAdding"], objects: ["manage"], - extra_keys: ["fxauid"], }, "itemUpdating": { methods: ["itemUpdating"], objects: ["manage"], - extra_keys: ["fxauid"], }, "itemDeleting": { methods: ["itemDeleting"], objects: ["manage"], - extra_keys: ["fxauid"], }, "itemAdded": { methods: ["itemAdded"], objects: ["manage"], - extra_keys: ["itemid", "fxauid"], + extra_keys: ["itemid"], }, "itemUpdated": { methods: ["itemUpdated"], objects: ["manage"], - extra_keys: ["itemid", "fxauid"], + extra_keys: ["itemid"], }, "itemDeleted": { methods: ["itemDeleted"], objects: ["manage"], - extra_keys: ["itemid", "fxauid"], + extra_keys: ["itemid"], }, "itemSelected": { methods: ["itemSelected"], objects: ["manage", "doorhanger"], - extra_keys: ["fxauid"], }, "addClick": { methods: ["addClick"], objects: ["manage"], - extra_keys: ["fxauid"], }, "datastore": { methods: ["added", "updated", "deleted"], objects: ["datastore"], - extra_keys: ["itemid", "fxauid", "fields"], + extra_keys: ["itemid", "fields"], }, "feedback": { methods: ["feedbackClick"], objects: ["manage"], - extra_keys: ["fxauid"], }, "faq": { methods: ["faqClick"], objects: ["manage"], - extra_keys: ["fxauid"], }, "itemCopied": { methods: ["usernameCopied", "passwordCopied"], objects: ["manage", "doorhanger"], - extra_keys: ["fxauid"], }, "resetRequested": { methods: ["resetRequested"], objects: ["settings"], - extra_keys: ["fxauid"], }, "resetCompleted": { methods: ["resetCompleted"], objects: ["settings"], - extra_keys: ["fxauid"], }, "setupGuest": { methods: ["click"], @@ -145,7 +133,6 @@ function startup({webExtension}, reason) { "fxaAuth": { methods: ["fxaUpgrade", "fxaSignin", "fxaSignout"], objects: ["accounts"], - extra_keys: ["fxauid"], }, "fxaFail": { methods: ["fxaFailed"], @@ -162,6 +149,24 @@ function startup({webExtension}, reason) { } } + try { + Services.telemetry.registerScalars(TELEMETRY_CATEGORY, { + "datastoreCount": { + kind: Services.telemetry.SCALAR_TYPE_COUNT, + keyed: false, + record_on_release: false, + expired: false, + }, + }); + } catch (e) { + if (e.message === "Attempt to register scalar that is already registered.") { + // eslint-disable-next-line no-console + console.log("telemetry scalar already registered; skipping registration"); + } else { + throw e; + } + } + webExtension.startup().then(({browser}) => { Services.telemetry.recordEvent(TELEMETRY_CATEGORY, "startup", "webextension"); @@ -173,6 +178,12 @@ function startup({webExtension}, reason) { message.extra || null ); respond({}); + break; + case "telemetry_scalar": + Services.telemetry.scalarSet( + `${TELEMETRY_CATEGORY}.${message.name}`, message.value + ); + respond({}); } }); diff --git a/src/webextension/background/accounts/index.js b/src/webextension/background/accounts/index.js index 202f34ab..7c62cd62 100644 --- a/src/webextension/background/accounts/index.js +++ b/src/webextension/background/accounts/index.js @@ -75,10 +75,11 @@ export const APP_KEY_NAME = "https://identity.mozilla.com/apps/lockbox"; export const DEFAULT_AVATAR_PATH = "icons/default-avatar.svg"; export class Account { - constructor({config = DEFAULT_CONFIG, info}) { + constructor({config = DEFAULT_CONFIG, info, storage}) { // TODO: verify configuration (when there is one) this.config = config; this.info = info || undefined; + this.storage = storage; } toJSON() { @@ -98,6 +99,14 @@ export class Account { info, }; } + async save() { + if (this.storage) { + const account = this.toJSON(); + await this.storage.set({ account }); + } + + return this; + } get mode() { const info = this.info; @@ -126,7 +135,7 @@ export class Account { let cfg = configs[this.config]; const props = {}; - let url, request; + let url; // request authorization cfg = { @@ -151,52 +160,14 @@ export class Account { } else { tokenParams.client_secret = cfg.client_secret; } - url = cfg.token_endpoint; - request = { - method: "post", - headers: { - "content-type": "application/json", - }, - cache: "no-cache", - body: JSON.stringify(tokenParams), - }; - const oauthInfo = await fetchFromEndPoint("token", url, request); - // console.log(`oauth info == ${JSON.stringify(oauthInfo)}`); + await this.updateAccessToken(tokenParams, props.appKey); - const keys = new Map(); - if (oauthInfo.keys_jwe) { - let bundle = await jose.JWE.createDecrypt(props.appKey).decrypt(oauthInfo.keys_jwe); - bundle = JSON.parse(new TextDecoder().decode(bundle.payload)); - const pending = Object.keys(bundle).map(async (name) => { - let key = bundle[name]; - key = await jose.JWK.asKey(key); - keys.set(name, key); - }); - await Promise.all(pending); - } + // update user info + await this.updateUserInfo(); - // retrieve user info - url = cfg.userinfo_endpoint; - request = { - method: "get", - headers: { - authorization: `Bearer ${oauthInfo.access_token}`, - }, - cache: "no-cache", - }; - const userInfo = await fetchFromEndPoint("userinfo", url, request); + // retain it all + await this.save(); - this.info = { - uid: userInfo.uid, - email: userInfo.email, - displayName: userInfo.displayName, - avatar: userInfo.avatar, - access_token: oauthInfo.access_token, - expires_at: (Date.now() / 1000) + oauthInfo.expires_in, - id_token: oauthInfo.id_token, - refresh_token: oauthInfo.refresh_token, - keys, - }; return this; } @@ -216,6 +187,8 @@ export class Account { } // XXXX: something server side? + await this.save(); + return this; } @@ -228,34 +201,146 @@ export class Account { avatar: this.avatar, }; } + + async token() { + // always return null if user is GUEST + if (this.mode === GUEST) { + // XXXX: use DataStoreError + throw new Error("AUTH: requires FxA"); + } + + // check if token present / unexpired / valid + let info = await this.updateUserInfo(); + if (!info || !info.access_token) { + // refresh + await this.updateAccessToken(); + info = await this.updateUserInfo(); + } + await this.save(); + + if (!info || !info.access_token) { + // XXXX: use DataStoreError + throw new Error("AUTH: no access token"); + } + + return info.access_token; + } + + + async updateAccessToken(params, appKey) { + const cfg = configs[this.config]; + let info = this.info || {}; + + if (!params) { + // assume "refresh_token" exchange + if (!info.refresh_token) { + // XXXX: use DataStoreError + throw new Error("AUTH: no refresh token"); + } + + params = { + grant_type: "refresh_token", + refresh_token: info.refresh_token, + client_id: cfg.client_id, + }; + } + + const request = { + method: "post", + headers: { + "content-type": "application/json", + }, + cache: "no-cache", + body: JSON.stringify(params), + }; + const oauthInfo = await fetchFromEndPoint("token", cfg.token_endpoint, request); + let keys = info.keys || new Map(); + if (oauthInfo.keys_jwe && appKey) { + // forget previous keys before decrypting new bundle + keys.clear(); + + let bundle = await jose.JWE.createDecrypt(appKey).decrypt(oauthInfo.keys_jwe); + bundle = JSON.parse(new TextDecoder().decode(bundle.payload)); + const pending = Object.keys(bundle).map(async (name) => { + let key = bundle[name]; + key = await jose.JWK.asKey(key); + keys.set(name, key); + }); + await Promise.all(pending); + } + + this.info = info = { + ...info, + access_token: oauthInfo.access_token, + expires_at: Math.floor(Date.now() / 1000) + oauthInfo.expires_in, + id_token: oauthInfo.id_token, + refresh_token: oauthInfo.refresh_token, + keys, + }; + + return info; + } + + async updateUserInfo() { + let info = this.info || {}; + const { access_token, expires_at } = info; + if (!access_token || !expires_at || Date.now() > (expires_at * 1000)) { + // need a new access token + return null; + } + + const url = configs[this.config].userinfo_endpoint; + const request = { + method: "get", + headers: { + "authorization": `Bearer ${access_token}`, + }, + cache: "no-cache", + }; + try { + const userInfo = await fetchFromEndPoint("userinfo", url, request); + // since it's present ... update user info + this.info = info = { + ...info, + uid: userInfo.uid, + email: userInfo.email, + displayName: userInfo.displayName, + avatar: userInfo.avatar, + }; + } catch (err) { + return null; + } + + return info; + } + } let account; export default function getAccount() { - if (!account) { - account = new Account({}); - } return account; } +export function setAccount(acct) { + account = acct || undefined; +} + export async function loadAccount(storage) { const stored = await storage.get("account"); if (stored && stored.account) { - account = new Account(stored.account); + account = new Account({ + ...stored.account, + storage, + }); + } else { + account = new Account({ + storage, + }); } return getAccount(); } -export async function saveAccount(storage) { - const account = getAccount().toJSON(); - await storage.set({ account }); -} - -export function setAccount(config, info) { - account = config ? new Account({config, info}) : undefined; -} - export async function openAccount(storage) { let account; @@ -266,8 +351,8 @@ export async function openAccount(storage) { console.log(`loaded account for (${account.mode.toString()}) '${account.uid || ""}'`); } catch (err) { // eslint-disable-next-line no-console - console.error(`loading account failed (fallback to empty GUEST): ${err.message}`); - account = getAccount(); + console.error(`loading account failed: ${err.message}`); + throw err; } return account; diff --git a/src/webextension/background/browser-action.js b/src/webextension/background/browser-action.js index f4a79dcc..e359e5d8 100644 --- a/src/webextension/background/browser-action.js +++ b/src/webextension/background/browser-action.js @@ -60,8 +60,13 @@ export default async function updateBrowserAction({account = getAccount(), datas if (account.mode === accounts.GUEST) { // unlock on user's behalf ... // XXXX: is this a bad idea or terrible idea? - await datastore.unlock(DEFAULT_APP_KEY); - return installEntriesAction(); + try { + await datastore.unlock(DEFAULT_APP_KEY); + return installEntriesAction(); + } catch (err) { + // eslint-disable-next-line no-console + console.warn("datastore is in an inconsistent state. You may need to reset and start again"); + } } // setup unlock popup return installPopup("unlock/index.html"); diff --git a/src/webextension/background/datastore.js b/src/webextension/background/datastore.js index 951b14f5..00f35926 100644 --- a/src/webextension/background/datastore.js +++ b/src/webextension/background/datastore.js @@ -26,6 +26,10 @@ export const DEFAULT_APP_KEY = { "k": "WsTdZ2tjji2W36JN9vk9s2AYsvp8eYy1pBbKPgcSLL4", }; +export function clearDataStore() { + datastore = undefined; +} + export default async function openDataStore(cfg = {}) { if (!datastore) { datastore = await DataStore.open({ diff --git a/src/webextension/background/index.js b/src/webextension/background/index.js index 86ea42bb..4b6a5f78 100644 --- a/src/webextension/background/index.js +++ b/src/webextension/background/index.js @@ -10,7 +10,12 @@ import updateBrowserAction from "./browser-action"; openAccount(browser.storage.local).then(async (account) => { let datastore = await openDataStore({ salt: account.uid }); if (datastore.initialized && account.mode === GUEST) { - await datastore.unlock(DEFAULT_APP_KEY); + try { + await datastore.unlock(DEFAULT_APP_KEY); + } catch (err) { + // eslint-disable-next-line no-console + console.error("datastore is in an inconsistent state."); + } } initializeMessagePorts(); diff --git a/src/webextension/background/message-ports.js b/src/webextension/background/message-ports.js index 48cf7136..14e2542d 100644 --- a/src/webextension/background/message-ports.js +++ b/src/webextension/background/message-ports.js @@ -2,7 +2,7 @@ * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ -import openDataStore, { DEFAULT_APP_KEY } from "./datastore"; +import openDataStore, { clearDataStore, DEFAULT_APP_KEY } from "./datastore"; import getAccount, * as accounts from "./accounts"; import updateBrowserAction from "./browser-action"; import * as telemetry from "./telemetry"; @@ -60,8 +60,6 @@ export default function initializeMessagePorts() { await datastore.initialize({ appKey: DEFAULT_APP_KEY, }); - // TODO: be more implicit on saving account info - await accounts.saveAccount(browser.storage.local); await updateBrowserAction({datastore}); if (message.view) { openView(message.view); @@ -80,12 +78,10 @@ export default function initializeMessagePorts() { await datastore.unlock(DEFAULT_APP_KEY); } await datastore.initialize({ appKey, salt, rebase: true }); - // FIXME: be more implicit on saving account info - await accounts.saveAccount(browser.storage.local); await updateBrowserAction({ account, datastore }); telemetry.recordEvent("fxaUpgrade", "accounts"); } catch (err) { - telemetry.recordEvent("fxaFailed", "accounts", err.message); + telemetry.recordEvent("fxaFailed", "accounts", { message: err.message }); throw err; } @@ -111,25 +107,33 @@ export default function initializeMessagePorts() { openView("firstrun"); return {}; + }).catch((err) => { + // eslint-disable-next-line no-console + console.log(`failed to reset: ${err.message}`); + throw err; }); case "signin": - return openDataStore().then(async (datastore) => { - const account = getAccount(); - let appKey; + return Promise.resolve(getAccount()).then(async (account) => { + let appKey = DEFAULT_APP_KEY; try { - if (account.mode === accounts.UNAUTHENTICATED) { + if (account.mode !== accounts.AUTHENTICATED) { await account.signIn(); } if (account.mode === accounts.AUTHENTICATED) { appKey = account.keys.get(accounts.APP_KEY_NAME); } + + // XXXX: Find a better way to affect recovery + clearDataStore(); + const datastore = await openDataStore({ salt: account.uid }); + await datastore.unlock(appKey); await updateBrowserAction({ datastore }); telemetry.recordEvent("fxaSignin", "accounts"); } catch (err) { - telemetry.recordEvent("fxaFailed", "accounts", err.message); + telemetry.recordEvent("fxaFailed", "accounts", { message: err.message }); throw err; } @@ -157,8 +161,10 @@ export default function initializeMessagePorts() { case "list_items": return openDataStore().then(async (ds) => { - return {items: Array.from((await ds.list()).values(), - makeItemSummary)}; + var entries = Array.from((await ds.list()).values(), + makeItemSummary); + telemetry.setScalar("datastoreCount", entries.length); + return {items: entries}; }); case "add_item": return openDataStore().then(async (ds) => { @@ -182,7 +188,6 @@ export default function initializeMessagePorts() { return openDataStore().then(async (ds) => { return {item: await ds.get(message.id)}; }); - case "proxy_telemetry_event": return telemetry.recordEvent(message.method, message.object, message.extra); diff --git a/src/webextension/background/telemetry.js b/src/webextension/background/telemetry.js index 1bad78ad..afdbfc48 100644 --- a/src/webextension/background/telemetry.js +++ b/src/webextension/background/telemetry.js @@ -2,15 +2,14 @@ * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ -import getAccount from "./accounts"; - export async function recordEvent(method, object, extra) { - const fxauid = getAccount().uid; - if (fxauid) { - extra = {...(extra || {}), fxauid}; - } - return browser.runtime.sendMessage({ type: "telemetry_event", method, object, extra, }); } + +export async function setScalar(name, value) { + return browser.runtime.sendMessage({ + type: "telemetry_scalar", name, value, + }); +} diff --git a/src/webextension/icons/account.svg b/src/webextension/icons/account.svg index 0b032a1a..9ab76308 100644 --- a/src/webextension/icons/account.svg +++ b/src/webextension/icons/account.svg @@ -1,3 +1,3 @@ - - + + diff --git a/src/webextension/icons/arrowhead-down-16.svg b/src/webextension/icons/arrowhead-down-16.svg deleted file mode 100644 index a56e6870..00000000 --- a/src/webextension/icons/arrowhead-down-16.svg +++ /dev/null @@ -1,3 +0,0 @@ - - - diff --git a/src/webextension/icons/arrowhead-right-16.svg b/src/webextension/icons/arrowhead-right-16.svg deleted file mode 100644 index cc6a0285..00000000 --- a/src/webextension/icons/arrowhead-right-16.svg +++ /dev/null @@ -1,6 +0,0 @@ - - - - diff --git a/src/webextension/icons/chevron-right.svg b/src/webextension/icons/chevron-right.svg new file mode 100644 index 00000000..b2a2bef8 --- /dev/null +++ b/src/webextension/icons/chevron-right.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/webextension/icons/clear.svg b/src/webextension/icons/clear.svg new file mode 100644 index 00000000..52782336 --- /dev/null +++ b/src/webextension/icons/clear.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/webextension/icons/hide.svg b/src/webextension/icons/hide.svg new file mode 100644 index 00000000..dbbbbd44 --- /dev/null +++ b/src/webextension/icons/hide.svg @@ -0,0 +1,7 @@ + + + + + + + diff --git a/src/webextension/icons/options.svg b/src/webextension/icons/options.svg index 01cf3b4f..2724f667 100644 --- a/src/webextension/icons/options.svg +++ b/src/webextension/icons/options.svg @@ -1,3 +1,3 @@ - + diff --git a/src/webextension/icons/search.svg b/src/webextension/icons/search.svg new file mode 100644 index 00000000..689a5f68 --- /dev/null +++ b/src/webextension/icons/search.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/webextension/icons/show.svg b/src/webextension/icons/show.svg new file mode 100644 index 00000000..4f184003 --- /dev/null +++ b/src/webextension/icons/show.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/webextension/icons/signout.svg b/src/webextension/icons/signout.svg index 73730be2..1b83001d 100644 --- a/src/webextension/icons/signout.svg +++ b/src/webextension/icons/signout.svg @@ -1,3 +1,3 @@ - - + + diff --git a/src/webextension/images/nessie_v2.svg b/src/webextension/images/nessie_v2.svg index 8a202723..5999a97a 100644 --- a/src/webextension/images/nessie_v2.svg +++ b/src/webextension/images/nessie_v2.svg @@ -1,367 +1,114 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + + + Global / Nessie + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/webextension/list/components/item-fields.css b/src/webextension/list/components/item-fields.css index 8d470585..171a7a83 100644 --- a/src/webextension/list/components/item-fields.css +++ b/src/webextension/list/components/item-fields.css @@ -5,7 +5,7 @@ .item-fields { display: grid; grid-template-columns: minmax(200px, 1fr); - max-width: 320px; + max-width: 360px; } .item-fields > label, @@ -17,6 +17,22 @@ resize: none; } +.input { + max-width: 242px; +} + +.password { + max-width: 258px; +} + +.notes { + min-height: 60px; +} + +.notes-read-only { + min-height: 73px; +} + .first-label { margin-top: 0; } diff --git a/src/webextension/list/components/item-fields.js b/src/webextension/list/components/item-fields.js index df00ec0a..97b9c1fc 100644 --- a/src/webextension/list/components/item-fields.js +++ b/src/webextension/list/components/item-fields.js @@ -28,26 +28,26 @@ const fieldsPropTypes = PropTypes.shape({ export function ItemFields({fields, onCopy}) { return (
+
+ + tITLe + + {fields.title} +
- oRIGIn + oRIGIn {fields.origin}
-
- - tITLe - - {fields.title} -
uSERNAMe
- + {fields.username} @@ -74,7 +74,7 @@ export function ItemFields({fields, onCopy}) { nOTEs - {fields.notes} + {fields.notes}
); @@ -108,23 +108,23 @@ export class EditItemFields extends React.Component { return (