summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--.travis.yml42
-rwxr-xr-x[-rw-r--r--]AUTHORS104
-rw-r--r--CONTRIBUTING.md29
-rwxr-xr-xCONTRIBUTORS902
-rw-r--r--LICENSE673
-rw-r--r--README.md70
-rw-r--r--app.json12
-rw-r--r--config.json.example29
-rw-r--r--contribute/developer-certificate-of-origin35
-rw-r--r--docs/guides/auth.md212
-rw-r--r--docs/guides/images/auth/application-page.pngbin0 -> 123152 bytes
-rw-r--r--docs/guides/images/auth/create-oauth-app.pngbin0 -> 27920 bytes
-rw-r--r--docs/guides/images/auth/create-twitter-app.pngbin0 -> 115406 bytes
-rw-r--r--docs/guides/images/auth/onelogin-add-app.pngbin0 -> 40519 bytes
-rw-r--r--docs/guides/images/auth/onelogin-copy-idp-metadata.pngbin0 -> 239493 bytes
-rw-r--r--docs/guides/images/auth/onelogin-edit-app-name.pngbin0 -> 122369 bytes
-rw-r--r--docs/guides/images/auth/onelogin-edit-sp-metadata.pngbin0 -> 184470 bytes
-rw-r--r--docs/guides/images/auth/onelogin-select-template.pngbin0 -> 73244 bytes
-rw-r--r--docs/guides/images/auth/onelogin-use-dashboard.pngbin0 -> 27216 bytes
-rw-r--r--docs/guides/images/auth/register-oauth-application-form.pngbin0 -> 61453 bytes
-rw-r--r--docs/guides/images/auth/register-twitter-application.pngbin0 -> 202414 bytes
-rw-r--r--docs/guides/images/auth/twitter-app-confirmation.pngbin0 -> 191064 bytes
-rw-r--r--docs/guides/images/auth/twitter-app-keys.pngbin0 -> 162703 bytes
-rw-r--r--lib/config/default.js19
-rw-r--r--lib/config/dockerSecret.js4
-rw-r--r--lib/config/environment.js25
-rw-r--r--lib/config/index.js2
-rw-r--r--lib/config/utils.js7
-rw-r--r--lib/models/user.js18
-rwxr-xr-xlib/response.js4
-rw-r--r--lib/web/auth/index.js2
-rw-r--r--lib/web/auth/mattermost/index.js49
-rw-r--r--lib/web/auth/saml/index.js95
-rw-r--r--locales/en.json2
-rw-r--r--locales/nl.json2
-rw-r--r--package.json5
-rw-r--r--public/js/extra.js23
-rw-r--r--public/js/render.js2
-rw-r--r--public/views/hackmd/body.ejs4
-rw-r--r--public/views/hackmd/header.ejs4
-rw-r--r--public/views/index/body.ejs4
-rw-r--r--public/views/shared/signin-modal.ejs14
-rw-r--r--yarn.lock145
43 files changed, 2355 insertions, 183 deletions
diff --git a/.travis.yml b/.travis.yml
index 61ced7a9..ee936e57 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -1,12 +1,38 @@
language: node_js
dist: trusty
-node_js:
- - 6
- - 7
-env:
- - CXX=g++-4.8
cache: yarn
+env:
+ global:
+ - CXX=g++-4.8
+ - YARN_VERSION=1.3.2
-before_install:
- - curl -o- -L https://yarnpkg.com/install.sh | bash -s -- --version 1.1.0
- - export PATH="$HOME/.yarn/bin:$PATH"
+jobs:
+ include:
+ - env: task=npm-test
+ node_js:
+ - 6
+ before_install:
+ - curl -o- -L https://yarnpkg.com/install.sh | bash -s -- --version "$YARN_VERSION"
+ - export PATH="$HOME/.yarn/bin:$PATH"
+ - env: task=npm-test
+ node_js:
+ - 7
+ before_install:
+ - curl -o- -L https://yarnpkg.com/install.sh | bash -s -- --version "$YARN_VERSION"
+ - export PATH="$HOME/.yarn/bin:$PATH"
+ - env: task=ShellCheck
+ script:
+ - shellcheck bin/*
+ language: generic
+ - env: task=doctoc
+ install: npm install doctoc
+ script:
+ - cp README.md README.md.orig
+ - npm run doctoc
+ - diff -q README.md README.md.orig
+ language: generic
+ - env: task=json-lint
+ install: npm install jsonlint
+ script:
+ - npm run jsonlint
+ language: generic
diff --git a/AUTHORS b/AUTHORS
index b155d3fa..ef33e970 100644..100755
--- a/AUTHORS
+++ b/AUTHORS
@@ -1,35 +1,69 @@
-List of HackMD contributors.
-
-bananaapple
-Bartlomiej Szala
-Colin Maudry
-Dmytro Kytsmen
-Fabien Meghazi
-Florian Rhiem
-Ikumi Shimizu
-ivanorsolic
-Jason Croft
-Jannik Lorenz
-James Stephenson
-Jordan Matelsky
-Kenji Doi
-Lars Kajes
-Lapinot
-Laura Kyle
-Marcelo Alencar
-Martijnpold
-Massimo Ghinassi
-Max Wu
-Ömer Erdinç Yağmurlu
-p0v1n0m
-Pablo Guerrero
-paraschadha2052
-Peter Dave Hello
-Qubo
-Sergio Valverde
-Tom Wyckhuys
-Yukai Huang
-Zacharias Traianos
-Zankio
-Xavier
-葉家郡 \ No newline at end of file
+alecdwm <alec@owls.io>
+bananaappletw <bananaappletw@gmail.com>
+Bartlomiej Szala <fenix440@gmail.com>
+BoHong Li <a60814billy@gmail.com>
+Bryan Davis <bd808@wikimedia.org>
+butlerx <butlerx@notthe.cloud>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Christian Schuhmann <madebyherzblut@users.noreply.github.com>
+Colin Maudry <colin@maudry.com>
+Dmytro Kytsmen <dmitrokytsmen@gmail.com>
+Fabien Meghazi <agr@amigrave.com>
+Florian Rhiem <florian.rhiem@gmail.com>
+geekyd <singhsince94@gmail.com>
+GhiMax <ghina8@gmail.com>
+greenkeeperio-bot <support@greenkeeper.io>
+Himura Kazuto <Himura2la@users.noreply.github.com>
+Ho33e5 <ho33e5@gmail.com>
+Ian Dees <ian.dees@gmail.com>
+Ikumi Shimizu <193s@users.noreply.github.com>
+ivanorsolic <ivanorsolic@users.noreply.github.com>
+jackycute <jacky_cute0808@hotmail.com>
+jackycute <jackymaxj@gmail.com>
+Jakub Sygnowski <sygnowski@gmail.com>
+James Stephenson <c4p7.fl1n7@gmail.com>
+Jan Kunzmann <jan-github@phobia.de>
+Jannik Lorenz <dev@janniklorenz.de>
+Jason Croft <jcroft@velocity.org>
+Johannes Weißl <jargon@molb.org>
+Jordan Matelsky <j6k4m8@gmail.com>
+Jun SAKATA <jun.bj141400@gmail.com>
+Kaiyu Shi <skyisno.1@gmail.com>
+knjcode <knjcode@gmail.com>
+Kotaro Yamamoto <kota.crk@gmail.com>
+Lars Karlsson <lars@kajes.se>
+Laura Kyle <laura.kyle91@gmail.com>
+LluisArevalo <thorin119@gmail.com>
+Marcelo Alencar <marceloalves@ufpa.br>
+Martijnpold <martijntje7@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+neopostmodern <clemens@neopostmodern.com>
+NV <nvsofts@gmail.com>
+Ömer Erdinç Yağmurlu <omeryagmurlu@gmail.com>
+p0v1n0m <p0v1n0m@gmail.com>
+Pablo Guerrero <pablo.guerrero@gmail.com>
+Pablo Guerrero <pablo.guerrero@sap.com>
+Paras <paraschadha2052@gmail.com>
+Patrick Andersen <patrick@bacha.dk>
+Peter Dave Hello <hsu@peterdavehello.org>
+Peter Dave Hello <PeterDaveHello@users.noreply.github.com>
+Philipp Zumstein <zuphilip@users.noreply.github.com>
+Raccoon Li <a60814billy@gmail.com>
+robert <ahmerov.rt@molodost.bz>
+Sergio Valverde <svg153@users.noreply.github.com>
+Sheogorath <sheogorath@shivering-isles.com>
+Simon Joda Stößer <SimJoSt@users.noreply.github.com>
+S.Noda <noda@fenrir.co.jp>
+Stratos Gerakakis <stratosgear@gmail.com>
+The Gitter Badger <badger@gitter.im>
+tkqubo <tk.qubo@gmail.com>
+tkykm <tkykm@users.noreply.github.com>
+Tom Wyckhuys <tomwyckhuys@gmail.com>
+Wonder Chang <iwonder.tw@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Xavier Marques <xaviermarques4f@gmail.com>
+xnum <s000032001@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+zachariast <zachariastraianos@gmail.com>
+Zankio <xxoojoeooxx1@gmail.com>
+蒼時弦也 <elct9620@frost.tw>
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index d22f70d9..b002e549 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -6,14 +6,14 @@ email, or any other method with the owners of this repository before making a ch
Please note we have a code of conduct, please follow it in all your interactions with the project.
## Pull Request Process
-
-1. Ensure any install or build dependencies are removed before the end of the layer when doing a
+1. Ensure you signed all your commits with Developer Certificate of Origin (DCO).
+2. Ensure any install or build dependencies are removed before the end of the layer when doing a
build.
-2. Update the README.md with details of changes to the interface, this includes new environment
+3. Update the README.md with details of changes to the interface, this includes new environment
variables, exposed ports, useful file locations and container parameters.
-3. Increase the version numbers in any examples files and the README.md to the new version that this
+4. Increase the version numbers in any examples files and the README.md to the new version that this
Pull Request would represent. The versioning scheme we use is [SemVer](http://semver.org/).
-4. You may merge the Pull Request in once you have the sign-off of two other developers, or if you
+5. You may merge the Pull Request in once you have the sign-off of two other developers, or if you
do not have permission to do that, you may request the second reviewer to merge it for you.
## Contributor Code of Conduct
@@ -52,3 +52,22 @@ issue or contacting one or more of the project maintainers.
This Code of Conduct is adapted from the [Contributor Covenant](http://contributor-covenant.org),
version 1.2.0, available at
[http://contributor-covenant.org/version/1/2/0/](http://contributor-covenant.org/version/1/2/0/)
+
+### Sign your work
+
+We use the Developer Certificate of Origin (DCO) as a additional safeguard
+for the HackMD project. This is a well established and widely used
+mechanism to assure contributors have confirmed their right to license
+their contribution under the project's license.
+Please read [contribute/developer-certificate-of-origin][dcofile].
+If you can certify it, then just add a line to every git commit message:
+
+````
+ Signed-off-by: Random J Developer <random@developer.example.org>
+````
+
+Use your real name (sorry, no pseudonyms or anonymous contributions).
+If you set your `user.name` and `user.email` git configs, you can sign your
+commit automatically with `git commit -s`. You can also use git [aliases](https://git-scm.com/book/tr/v2/Git-Basics-Git-Aliases)
+like `git config --global alias.ci 'commit -s'`. Now you can commit with
+`git ci` and the commit will be signed.
diff --git a/CONTRIBUTORS b/CONTRIBUTORS
new file mode 100755
index 00000000..d5e679c4
--- /dev/null
+++ b/CONTRIBUTORS
@@ -0,0 +1,902 @@
+=== .babelrc
+Yukai Huang <yukaihuangtw@gmail.com>
+=== .editorconfig
+bananaappletw <bananaappletw@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== .gitignore
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== .sequelizerc.example
+Yukai Huang <yukaihuangtw@gmail.com>
+=== .travis.yml
+bananaappletw <bananaappletw@gmail.com>
+BoHong Li <a60814billy@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+=== AUTHORS
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== CONTRIBUTING.md
+Max Wu <jackymaxj@gmail.com>
+=== LICENSE
+Cheng-Han, Wu <jackymaxj@gmail.com>
+jackycute <jacky_cute0808@hotmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== README.md
+alecdwm <alec@owls.io>
+bananaappletw <bananaappletw@gmail.com>
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Florian Rhiem <florian.rhiem@gmail.com>
+jackycute <jackymaxj@gmail.com>
+Jannik Lorenz <dev@janniklorenz.de>
+Jason Croft <jcroft@velocity.org>
+Johannes Weißl <jargon@molb.org>
+Jun SAKATA <jun.bj141400@gmail.com>
+Laura Kyle <laura.kyle91@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+neopostmodern <clemens@neopostmodern.com>
+NV <nvsofts@gmail.com>
+Sheogorath <sheogorath@shivering-isles.com>
+The Gitter Badger <badger@gitter.im>
+Wonder Chang <iwonder.tw@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+蒼時弦也 <elct9620@frost.tw>
+=== app.js
+alecdwm <alec@owls.io>
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+James Stephenson <c4p7.fl1n7@gmail.com>
+Jan Kunzmann <jan-github@phobia.de>
+Jason Croft <jcroft@velocity.org>
+Jordan Matelsky <j6k4m8@gmail.com>
+knjcode <knjcode@gmail.com>
+LluisArevalo <thorin119@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+NV <nvsofts@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Raccoon Li <a60814billy@gmail.com>
+robert <ahmerov.rt@molodost.bz>
+Sheogorath <sheogorath@shivering-isles.com>
+S.Noda <noda@fenrir.co.jp>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+xnum <s000032001@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== app.json
+bananaappletw <bananaappletw@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== bin/heroku
+bananaappletw <bananaappletw@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== bin/setup
+Sheogorath <sheogorath@shivering-isles.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== config.json.example
+alecdwm <alec@owls.io>
+bananaappletw <bananaappletw@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== docs/guides/images/s3-image-upload/bucket-policy-editor.png
+Yukai Huang <yukaihuangtw@gmail.com>
+=== docs/guides/images/s3-image-upload/bucket-property.png
+Yukai Huang <yukaihuangtw@gmail.com>
+=== docs/guides/images/s3-image-upload/create-bucket.png
+Yukai Huang <yukaihuangtw@gmail.com>
+=== docs/guides/images/s3-image-upload/custom-policy.png
+Yukai Huang <yukaihuangtw@gmail.com>
+=== docs/guides/images/s3-image-upload/iam-user.png
+Yukai Huang <yukaihuangtw@gmail.com>
+=== docs/guides/images/s3-image-upload/review-policy.png
+Yukai Huang <yukaihuangtw@gmail.com>
+=== docs/guides/s3-image-upload.md
+Johannes Weißl <jargon@molb.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== lib/config/default.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/config/defaultSSL.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/config/dockerSecret.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/config/enum.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/config/environment.js
+BoHong Li <a60814billy@gmail.com>
+Raccoon Li <a60814billy@gmail.com>
+=== lib/config/index.js
+BoHong Li <a60814billy@gmail.com>
+tkykm <tkykm@users.noreply.github.com>
+=== lib/config/oldEnvironment.js
+BoHong Li <a60814billy@gmail.com>
+Raccoon Li <a60814billy@gmail.com>
+=== lib/config/utils.js
+Raccoon Li <a60814billy@gmail.com>
+=== lib/history.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/letter-avatars.js
+alecdwm <alec@owls.io>
+BoHong Li <a60814billy@gmail.com>
+=== lib/logger.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/migrations/20150504155329-create-users.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/migrations/20150508114741-create-notes.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/migrations/20150515125813-create-temp.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/migrations/20150702001020-update-to-0_3_1.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/migrations/20150915153700-change-notes-title-to-text.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/migrations/20160112220142-note-add-lastchange.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/migrations/20160420180355-note-add-alias.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/migrations/20160515114000-user-add-tokens.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+=== lib/migrations/20160607060246-support-revision.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/migrations/20160703062241-support-authorship.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/migrations/20161009040430-support-delete-note.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/migrations/20161201050312-support-email-signin.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/models/author.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/models/index.js
+bananaappletw <bananaappletw@gmail.com>
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== lib/models/note.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+NV <nvsofts@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+蒼時弦也 <elct9620@frost.tw>
+=== lib/models/revision.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/models/temp.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+=== lib/models/user.js
+alecdwm <alec@owls.io>
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Jason Croft <jcroft@velocity.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/ot/client.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/ot/editor-socketio-server.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/ot/index.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/ot/selection.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/ot/server.js
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/ot/simple-text-operation.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/ot/text-operation.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/ot/wrapped-operation.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== lib/realtime.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Florian Rhiem <florian.rhiem@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+蒼時弦也 <elct9620@frost.tw>
+=== lib/response.js
+alecdwm <alec@owls.io>
+BoHong Li <a60814billy@gmail.com>
+butlerx <butlerx@notthe.cloud>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Florian Rhiem <florian.rhiem@gmail.com>
+Ikumi Shimizu <193s@users.noreply.github.com>
+Jannik Lorenz <dev@janniklorenz.de>
+Jason Croft <jcroft@velocity.org>
+Sheogorath <sheogorath@shivering-isles.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+xnum <s000032001@gmail.com>
+蒼時弦也 <elct9620@frost.tw>
+=== lib/utils.js
+BoHong Li <a60814billy@gmail.com>
+butlerx <butlerx@notthe.cloud>
+LluisArevalo <thorin119@gmail.com>
+=== lib/web/auth/dropbox/index.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/auth/email/index.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/auth/facebook/index.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/auth/github/index.js
+BoHong Li <a60814billy@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+=== lib/web/auth/gitlab/index.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/auth/google/index.js
+BoHong Li <a60814billy@gmail.com>
+Kaiyu Shi <skyisno.1@gmail.com>
+=== lib/web/auth/index.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/auth/ldap/index.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/auth/twitter/index.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/auth/utils.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/baseRouter.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/historyRouter.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/imageRouter.js
+BoHong Li <a60814billy@gmail.com>
+Kotaro Yamamoto <kota.crk@gmail.com>
+Raccoon Li <a60814billy@gmail.com>
+=== lib/web/middleware/checkURIValid.js
+BoHong Li <a60814billy@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+=== lib/web/middleware/redirectWithoutTrailingSlashes.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/middleware/tooBusy.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/noteRouter.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/statusRouter.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/userRouter.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/web/utils.js
+BoHong Li <a60814billy@gmail.com>
+=== lib/workers/dmpWorker.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== locales/ca.json
+Xavier Marques <xaviermarques4f@gmail.com>
+=== locales/da.json
+Patrick Andersen <patrick@bacha.dk>
+=== locales/de.json
+Jannik Lorenz <dev@janniklorenz.de>
+Philipp Zumstein <zuphilip@users.noreply.github.com>
+Simon Joda Stößer <SimJoSt@users.noreply.github.com>
+=== locales/el.json
+Stratos Gerakakis <stratosgear@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+zachariast <zachariastraianos@gmail.com>
+=== locales/en.json
+alecdwm <alec@owls.io>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== locales/eo.json
+James Stephenson <c4p7.fl1n7@gmail.com>
+=== locales/es.json
+Pablo Guerrero <pablo.guerrero@sap.com>
+Sergio Valverde <svg153@users.noreply.github.com>
+=== locales/fr.json
+Colin Maudry <colin@maudry.com>
+Ho33e5 <ho33e5@gmail.com>
+=== locales/hi.json
+Paras <paraschadha2052@gmail.com>
+=== locales/hr.json
+ivanorsolic <ivanorsolic@users.noreply.github.com>
+=== locales/it.json
+GhiMax <ghina8@gmail.com>
+=== locales/ja.json
+tkqubo <tk.qubo@gmail.com>
+=== locales/nl.json
+Martijnpold <martijntje7@gmail.com>
+Tom Wyckhuys <tomwyckhuys@gmail.com>
+=== locales/pl.json
+Bartlomiej Szala <fenix440@gmail.com>
+Jakub Sygnowski <sygnowski@gmail.com>
+=== locales/pt.json
+Marcelo Alencar <marceloalves@ufpa.br>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== locales/ru.json
+Himura Kazuto <Himura2la@users.noreply.github.com>
+p0v1n0m <p0v1n0m@gmail.com>
+=== locales/sv.json
+Lars Karlsson <lars@kajes.se>
+Patrick Andersen <patrick@bacha.dk>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== locales/tr.json
+Ömer Erdinç Yağmurlu <omeryagmurlu@gmail.com>
+=== locales/uk.json
+Dmytro Kytsmen <dmitrokytsmen@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== locales/zh.json
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== package.json
+alecdwm <alec@owls.io>
+bananaappletw <bananaappletw@gmail.com>
+BoHong Li <a60814billy@gmail.com>
+Bryan Davis <bd808@wikimedia.org>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Fabien Meghazi <agr@amigrave.com>
+greenkeeperio-bot <support@greenkeeper.io>
+Jason Croft <jcroft@velocity.org>
+Max Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Peter Dave Hello <PeterDaveHello@users.noreply.github.com>
+Sheogorath <sheogorath@shivering-isles.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+xnum <s000032001@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/apple-touch-icon.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/bootstrap-social.css
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/center.css
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/cover.css
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Jason Croft <jcroft@velocity.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/extra.css
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/font.css
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/css/github-extract.css
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/google-font.css
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/css/index.css
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Jason Croft <jcroft@velocity.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/markdown.css
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/mermaid.css
+Cheng-Han, Wu <jackymaxj@gmail.com>
+=== public/css/site.css
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/css/slide-preview.css
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/css/slide.css
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/default.md
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/docs/features.md
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+Pablo Guerrero <pablo.guerrero@gmail.com>
+Sheogorath <sheogorath@shivering-isles.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/docs/release-notes.md
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/docs/slide-example.md
+butlerx <butlerx@notthe.cloud>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/docs/yaml-metadata.md
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/favicon.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/fonts/SourceCodePro-Black.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Black.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Black.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Bold.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Bold.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Bold.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-ExtraLight.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-ExtraLight.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-ExtraLight.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Light.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Light.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Light.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Medium.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Medium.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Medium.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Regular.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Regular.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Regular.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Semibold.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Semibold.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceCodePro-Semibold.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Black.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Black.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Black.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-BlackItalic.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-BlackItalic.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-BlackItalic.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Bold.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Bold.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Bold.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-BoldItalic.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-BoldItalic.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-BoldItalic.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-ExtraLight.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-ExtraLight.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-ExtraLight.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-ExtraLightItalic.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-ExtraLightItalic.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-ExtraLightItalic.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Italic.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Italic.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Italic.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Light.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Light.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Light.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-LightItalic.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-LightItalic.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-LightItalic.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Regular.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Regular.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Regular.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Semibold.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Semibold.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-Semibold.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-SemiboldItalic.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-SemiboldItalic.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSansPro-SemiboldItalic.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Bold.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Bold.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Bold.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Regular.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Regular.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Regular.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Semibold.eot
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Semibold.ttf
+Peter Dave Hello <hsu@peterdavehello.org>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/fonts/SourceSerifPro-Semibold.woff
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/hackmd-icon-1024.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/js/cover.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Jason Croft <jcroft@velocity.org>
+NV <nvsofts@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/extra.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+NV <nvsofts@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/google-drive-picker.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+=== public/js/google-drive-upload.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/js/history.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/htmlExport.js
+BoHong Li <a60814billy@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/index.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Jason Croft <jcroft@velocity.org>
+Laura Kyle <laura.kyle91@gmail.com>
+NV <nvsofts@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+xnum <s000032001@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+Zankio <xxoojoeooxx1@gmail.com>
+蒼時弦也 <elct9620@frost.tw>
+=== public/js/lib/appState.js
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/common/constant.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/js/lib/common/login.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/config/index.js
+BoHong Li <a60814billy@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/editor/config.js
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/editor/index.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/editor/statusbar.html
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/editor/ui-elements.js
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/editor/utils.js
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/modeType.js
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/lib/syncscroll.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/locale.js
+BoHong Li <a60814billy@gmail.com>
+Peter Dave Hello <PeterDaveHello@users.noreply.github.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/pretty.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/render.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/js/reveal-markdown.js
+BoHong Li <a60814billy@gmail.com>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/js/slide.js
+BoHong Li <a60814billy@gmail.com>
+Max Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/screenshot.png
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/uploads/.gitkeep
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/vendor/abcjs_basic_3.1.1-min.js
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/bootstrap/tooltip.min.css
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/bootstrap/tooltip.min.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/codemirror-spell-checker/en_US.aff
+Cheng-Han, Wu <jackymaxj@gmail.com>
+=== public/vendor/codemirror-spell-checker/en_US.dic
+Cheng-Han, Wu <jackymaxj@gmail.com>
+=== public/vendor/codemirror-spell-checker/spell-checker.min.css
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/codemirror-spell-checker/spell-checker.min.js
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/vendor/inlineAttachment/codemirror.inline-attachment.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/inlineAttachment/inline-attachment.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-textcomplete/jquery.textcomplete.js
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-bg_flat_0_aaaaaa_40x100.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-bg_flat_75_ffffff_40x100.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-bg_glass_55_fbf9ee_1x400.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-bg_glass_65_ffffff_1x400.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-bg_glass_75_dadada_1x400.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-bg_glass_75_e6e6e6_1x400.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-bg_glass_95_fef1ec_1x400.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-bg_highlight-soft_75_cccccc_1x100.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-icons_222222_256x240.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-icons_2e83ff_256x240.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-icons_454545_256x240.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-icons_888888_256x240.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/images/ui-icons_cd0a0a_256x240.png
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/jquery-ui.min.css
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/jquery-ui/jquery-ui.min.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/md-toc.js
+BoHong Li <a60814billy@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/ajax-adapter.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/client.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/codemirror-adapter.js
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/compress.sh
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/editor-client.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/ot.min.js
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/selection.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/socketio-adapter.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/text-operation.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/undo-manager.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/ot/wrapped-operation.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/showup/showup.css
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/vendor/showup/showup.js
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/error.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/hackmd.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/hackmd/body.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Florian Rhiem <florian.rhiem@gmail.com>
+Ian Dees <ian.dees@gmail.com>
+Jason Croft <jcroft@velocity.org>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+蒼時弦也 <elct9620@frost.tw>
+=== public/views/hackmd/foot.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Jannik Lorenz <dev@janniklorenz.de>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/views/hackmd/footer.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/hackmd/head.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+xnum <s000032001@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/views/hackmd/header.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Jannik Lorenz <dev@janniklorenz.de>
+Jason Croft <jcroft@velocity.org>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+xnum <s000032001@gmail.com>
+=== public/views/html.hbs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/includes/header.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/views/includes/scripts.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/views/index.ejs
+alecdwm <alec@owls.io>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Florian Rhiem <florian.rhiem@gmail.com>
+James Stephenson <c4p7.fl1n7@gmail.com>
+Jannik Lorenz <dev@janniklorenz.de>
+Jason Croft <jcroft@velocity.org>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/views/index/body.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/index/foot.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/index/footer.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/index/head.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+xnum <s000032001@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/views/index/header.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/pretty.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== public/views/shared/disqus.ejs
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/shared/ga.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/shared/help-modal.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/shared/polyfill.ejs
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/shared/refresh-modal.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/shared/revision-modal.ejs
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/shared/signin-modal.ejs
+alecdwm <alec@owls.io>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Jason Croft <jcroft@velocity.org>
+neopostmodern <clemens@neopostmodern.com>
+Sheogorath <sheogorath@shivering-isles.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== public/views/slide.ejs
+butlerx <butlerx@notthe.cloud>
+Cheng-Han, Wu <jackymaxj@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== tmp/.keep
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+=== webpack.config.js
+BoHong Li <a60814billy@gmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== webpack.production.js
+BoHong Li <a60814billy@gmail.com>
+geekyd <singhsince94@gmail.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== webpackBaseConfig.js
+BoHong Li <a60814billy@gmail.com>
+Peter Dave Hello <hsu@peterdavehello.org>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
+=== yarn.lock
+BoHong Li <a60814billy@gmail.com>
+Christian Schuhmann <madebyherzblut@users.noreply.github.com>
+Wu Cheng-Han <jacky_cute0808@hotmail.com>
+Yukai Huang <yukaihuangtw@gmail.com>
diff --git a/LICENSE b/LICENSE
index f573a0a9..2def0e88 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,22 +1,661 @@
-The MIT License (MIT)
+ GNU AFFERO GENERAL PUBLIC LICENSE
+ Version 3, 19 November 2007
-Copyright (c) 2017 Max Wu <jackymaxj@gmail.com> and others
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
-Permission is hereby granted, free of charge, to any person obtaining a copy
-of this software and associated documentation files (the "Software"), to deal
-in the Software without restriction, including without limitation the rights
-to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
-copies of the Software, and to permit persons to whom the Software is
-furnished to do so, subject to the following conditions:
+ Preamble
-The above copyright notice and this permission notice shall be included in all
-copies or substantial portions of the Software.
+ The GNU Affero General Public License is a free, copyleft license for
+software and other kinds of works, specifically designed to ensure
+cooperation with the community in the case of network server software.
-THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
-IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
-FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
-AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
-LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
-OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
-SOFTWARE.
+ The licenses for most software and other practical works are designed
+to take away your freedom to share and change the works. By contrast,
+our General Public Licenses are intended to guarantee your freedom to
+share and change all versions of a program--to make sure it remains free
+software for all its users.
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+them if you wish), that you receive source code or can get it if you
+want it, that you can change the software or use pieces of it in new
+free programs, and that you know you can do these things.
+
+ Developers that use our General Public Licenses protect your rights
+with two steps: (1) assert copyright on the software, and (2) offer
+you this License which gives you legal permission to copy, distribute
+and/or modify the software.
+
+ A secondary benefit of defending all users' freedom is that
+improvements made in alternate versions of the program, if they
+receive widespread use, become available for other developers to
+incorporate. Many developers of free software are heartened and
+encouraged by the resulting cooperation. However, in the case of
+software used on network servers, this result may fail to come about.
+The GNU General Public License permits making a modified version and
+letting the public access it on a server without ever releasing its
+source code to the public.
+
+ The GNU Affero General Public License is designed specifically to
+ensure that, in such cases, the modified source code becomes available
+to the community. It requires the operator of a network server to
+provide the source code of the modified version running there to the
+users of that server. Therefore, public use of a modified version, on
+a publicly accessible server, gives the public access to the source
+code of the modified version.
+
+ An older license, called the Affero General Public License and
+published by Affero, was designed to accomplish similar goals. This is
+a different license, not a version of the Affero GPL, but Affero has
+released a new version of the Affero GPL which permits relicensing under
+this license.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+
+ TERMS AND CONDITIONS
+
+ 0. Definitions.
+
+ "This License" refers to version 3 of the GNU Affero General Public License.
+
+ "Copyright" also means copyright-like laws that apply to other kinds of
+works, such as semiconductor masks.
+
+ "The Program" refers to any copyrightable work licensed under this
+License. Each licensee is addressed as "you". "Licensees" and
+"recipients" may be individuals or organizations.
+
+ To "modify" a work means to copy from or adapt all or part of the work
+in a fashion requiring copyright permission, other than the making of an
+exact copy. The resulting work is called a "modified version" of the
+earlier work or a work "based on" the earlier work.
+
+ A "covered work" means either the unmodified Program or a work based
+on the Program.
+
+ To "propagate" a work means to do anything with it that, without
+permission, would make you directly or secondarily liable for
+infringement under applicable copyright law, except executing it on a
+computer or modifying a private copy. Propagation includes copying,
+distribution (with or without modification), making available to the
+public, and in some countries other activities as well.
+
+ To "convey" a work means any kind of propagation that enables other
+parties to make or receive copies. Mere interaction with a user through
+a computer network, with no transfer of a copy, is not conveying.
+
+ An interactive user interface displays "Appropriate Legal Notices"
+to the extent that it includes a convenient and prominently visible
+feature that (1) displays an appropriate copyright notice, and (2)
+tells the user that there is no warranty for the work (except to the
+extent that warranties are provided), that licensees may convey the
+work under this License, and how to view a copy of this License. If
+the interface presents a list of user commands or options, such as a
+menu, a prominent item in the list meets this criterion.
+
+ 1. Source Code.
+
+ The "source code" for a work means the preferred form of the work
+for making modifications to it. "Object code" means any non-source
+form of a work.
+
+ A "Standard Interface" means an interface that either is an official
+standard defined by a recognized standards body, or, in the case of
+interfaces specified for a particular programming language, one that
+is widely used among developers working in that language.
+
+ The "System Libraries" of an executable work include anything, other
+than the work as a whole, that (a) is included in the normal form of
+packaging a Major Component, but which is not part of that Major
+Component, and (b) serves only to enable use of the work with that
+Major Component, or to implement a Standard Interface for which an
+implementation is available to the public in source code form. A
+"Major Component", in this context, means a major essential component
+(kernel, window system, and so on) of the specific operating system
+(if any) on which the executable work runs, or a compiler used to
+produce the work, or an object code interpreter used to run it.
+
+ The "Corresponding Source" for a work in object code form means all
+the source code needed to generate, install, and (for an executable
+work) run the object code and to modify the work, including scripts to
+control those activities. However, it does not include the work's
+System Libraries, or general-purpose tools or generally available free
+programs which are used unmodified in performing those activities but
+which are not part of the work. For example, Corresponding Source
+includes interface definition files associated with source files for
+the work, and the source code for shared libraries and dynamically
+linked subprograms that the work is specifically designed to require,
+such as by intimate data communication or control flow between those
+subprograms and other parts of the work.
+
+ The Corresponding Source need not include anything that users
+can regenerate automatically from other parts of the Corresponding
+Source.
+
+ The Corresponding Source for a work in source code form is that
+same work.
+
+ 2. Basic Permissions.
+
+ All rights granted under this License are granted for the term of
+copyright on the Program, and are irrevocable provided the stated
+conditions are met. This License explicitly affirms your unlimited
+permission to run the unmodified Program. The output from running a
+covered work is covered by this License only if the output, given its
+content, constitutes a covered work. This License acknowledges your
+rights of fair use or other equivalent, as provided by copyright law.
+
+ You may make, run and propagate covered works that you do not
+convey, without conditions so long as your license otherwise remains
+in force. You may convey covered works to others for the sole purpose
+of having them make modifications exclusively for you, or provide you
+with facilities for running those works, provided that you comply with
+the terms of this License in conveying all material for which you do
+not control copyright. Those thus making or running the covered works
+for you must do so exclusively on your behalf, under your direction
+and control, on terms that prohibit them from making any copies of
+your copyrighted material outside their relationship with you.
+
+ Conveying under any other circumstances is permitted solely under
+the conditions stated below. Sublicensing is not allowed; section 10
+makes it unnecessary.
+
+ 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
+
+ No covered work shall be deemed part of an effective technological
+measure under any applicable law fulfilling obligations under article
+11 of the WIPO copyright treaty adopted on 20 December 1996, or
+similar laws prohibiting or restricting circumvention of such
+measures.
+
+ When you convey a covered work, you waive any legal power to forbid
+circumvention of technological measures to the extent such circumvention
+is effected by exercising rights under this License with respect to
+the covered work, and you disclaim any intention to limit operation or
+modification of the work as a means of enforcing, against the work's
+users, your or third parties' legal rights to forbid circumvention of
+technological measures.
+
+ 4. Conveying Verbatim Copies.
+
+ You may convey verbatim copies of the Program's source code as you
+receive it, in any medium, provided that you conspicuously and
+appropriately publish on each copy an appropriate copyright notice;
+keep intact all notices stating that this License and any
+non-permissive terms added in accord with section 7 apply to the code;
+keep intact all notices of the absence of any warranty; and give all
+recipients a copy of this License along with the Program.
+
+ You may charge any price or no price for each copy that you convey,
+and you may offer support or warranty protection for a fee.
+
+ 5. Conveying Modified Source Versions.
+
+ You may convey a work based on the Program, or the modifications to
+produce it from the Program, in the form of source code under the
+terms of section 4, provided that you also meet all of these conditions:
+
+ a) The work must carry prominent notices stating that you modified
+ it, and giving a relevant date.
+
+ b) The work must carry prominent notices stating that it is
+ released under this License and any conditions added under section
+ 7. This requirement modifies the requirement in section 4 to
+ "keep intact all notices".
+
+ c) You must license the entire work, as a whole, under this
+ License to anyone who comes into possession of a copy. This
+ License will therefore apply, along with any applicable section 7
+ additional terms, to the whole of the work, and all its parts,
+ regardless of how they are packaged. This License gives no
+ permission to license the work in any other way, but it does not
+ invalidate such permission if you have separately received it.
+
+ d) If the work has interactive user interfaces, each must display
+ Appropriate Legal Notices; however, if the Program has interactive
+ interfaces that do not display Appropriate Legal Notices, your
+ work need not make them do so.
+
+ A compilation of a covered work with other separate and independent
+works, which are not by their nature extensions of the covered work,
+and which are not combined with it such as to form a larger program,
+in or on a volume of a storage or distribution medium, is called an
+"aggregate" if the compilation and its resulting copyright are not
+used to limit the access or legal rights of the compilation's users
+beyond what the individual works permit. Inclusion of a covered work
+in an aggregate does not cause this License to apply to the other
+parts of the aggregate.
+
+ 6. Conveying Non-Source Forms.
+
+ You may convey a covered work in object code form under the terms
+of sections 4 and 5, provided that you also convey the
+machine-readable Corresponding Source under the terms of this License,
+in one of these ways:
+
+ a) Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by the
+ Corresponding Source fixed on a durable physical medium
+ customarily used for software interchange.
+
+ b) Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by a
+ written offer, valid for at least three years and valid for as
+ long as you offer spare parts or customer support for that product
+ model, to give anyone who possesses the object code either (1) a
+ copy of the Corresponding Source for all the software in the
+ product that is covered by this License, on a durable physical
+ medium customarily used for software interchange, for a price no
+ more than your reasonable cost of physically performing this
+ conveying of source, or (2) access to copy the
+ Corresponding Source from a network server at no charge.
+
+ c) Convey individual copies of the object code with a copy of the
+ written offer to provide the Corresponding Source. This
+ alternative is allowed only occasionally and noncommercially, and
+ only if you received the object code with such an offer, in accord
+ with subsection 6b.
+
+ d) Convey the object code by offering access from a designated
+ place (gratis or for a charge), and offer equivalent access to the
+ Corresponding Source in the same way through the same place at no
+ further charge. You need not require recipients to copy the
+ Corresponding Source along with the object code. If the place to
+ copy the object code is a network server, the Corresponding Source
+ may be on a different server (operated by you or a third party)
+ that supports equivalent copying facilities, provided you maintain
+ clear directions next to the object code saying where to find the
+ Corresponding Source. Regardless of what server hosts the
+ Corresponding Source, you remain obligated to ensure that it is
+ available for as long as needed to satisfy these requirements.
+
+ e) Convey the object code using peer-to-peer transmission, provided
+ you inform other peers where the object code and Corresponding
+ Source of the work are being offered to the general public at no
+ charge under subsection 6d.
+
+ A separable portion of the object code, whose source code is excluded
+from the Corresponding Source as a System Library, need not be
+included in conveying the object code work.
+
+ A "User Product" is either (1) a "consumer product", which means any
+tangible personal property which is normally used for personal, family,
+or household purposes, or (2) anything designed or sold for incorporation
+into a dwelling. In determining whether a product is a consumer product,
+doubtful cases shall be resolved in favor of coverage. For a particular
+product received by a particular user, "normally used" refers to a
+typical or common use of that class of product, regardless of the status
+of the particular user or of the way in which the particular user
+actually uses, or expects or is expected to use, the product. A product
+is a consumer product regardless of whether the product has substantial
+commercial, industrial or non-consumer uses, unless such uses represent
+the only significant mode of use of the product.
+
+ "Installation Information" for a User Product means any methods,
+procedures, authorization keys, or other information required to install
+and execute modified versions of a covered work in that User Product from
+a modified version of its Corresponding Source. The information must
+suffice to ensure that the continued functioning of the modified object
+code is in no case prevented or interfered with solely because
+modification has been made.
+
+ If you convey an object code work under this section in, or with, or
+specifically for use in, a User Product, and the conveying occurs as
+part of a transaction in which the right of possession and use of the
+User Product is transferred to the recipient in perpetuity or for a
+fixed term (regardless of how the transaction is characterized), the
+Corresponding Source conveyed under this section must be accompanied
+by the Installation Information. But this requirement does not apply
+if neither you nor any third party retains the ability to install
+modified object code on the User Product (for example, the work has
+been installed in ROM).
+
+ The requirement to provide Installation Information does not include a
+requirement to continue to provide support service, warranty, or updates
+for a work that has been modified or installed by the recipient, or for
+the User Product in which it has been modified or installed. Access to a
+network may be denied when the modification itself materially and
+adversely affects the operation of the network or violates the rules and
+protocols for communication across the network.
+
+ Corresponding Source conveyed, and Installation Information provided,
+in accord with this section must be in a format that is publicly
+documented (and with an implementation available to the public in
+source code form), and must require no special password or key for
+unpacking, reading or copying.
+
+ 7. Additional Terms.
+
+ "Additional permissions" are terms that supplement the terms of this
+License by making exceptions from one or more of its conditions.
+Additional permissions that are applicable to the entire Program shall
+be treated as though they were included in this License, to the extent
+that they are valid under applicable law. If additional permissions
+apply only to part of the Program, that part may be used separately
+under those permissions, but the entire Program remains governed by
+this License without regard to the additional permissions.
+
+ When you convey a copy of a covered work, you may at your option
+remove any additional permissions from that copy, or from any part of
+it. (Additional permissions may be written to require their own
+removal in certain cases when you modify the work.) You may place
+additional permissions on material, added by you to a covered work,
+for which you have or can give appropriate copyright permission.
+
+ Notwithstanding any other provision of this License, for material you
+add to a covered work, you may (if authorized by the copyright holders of
+that material) supplement the terms of this License with terms:
+
+ a) Disclaiming warranty or limiting liability differently from the
+ terms of sections 15 and 16 of this License; or
+
+ b) Requiring preservation of specified reasonable legal notices or
+ author attributions in that material or in the Appropriate Legal
+ Notices displayed by works containing it; or
+
+ c) Prohibiting misrepresentation of the origin of that material, or
+ requiring that modified versions of such material be marked in
+ reasonable ways as different from the original version; or
+
+ d) Limiting the use for publicity purposes of names of licensors or
+ authors of the material; or
+
+ e) Declining to grant rights under trademark law for use of some
+ trade names, trademarks, or service marks; or
+
+ f) Requiring indemnification of licensors and authors of that
+ material by anyone who conveys the material (or modified versions of
+ it) with contractual assumptions of liability to the recipient, for
+ any liability that these contractual assumptions directly impose on
+ those licensors and authors.
+
+ All other non-permissive additional terms are considered "further
+restrictions" within the meaning of section 10. If the Program as you
+received it, or any part of it, contains a notice stating that it is
+governed by this License along with a term that is a further
+restriction, you may remove that term. If a license document contains
+a further restriction but permits relicensing or conveying under this
+License, you may add to a covered work material governed by the terms
+of that license document, provided that the further restriction does
+not survive such relicensing or conveying.
+
+ If you add terms to a covered work in accord with this section, you
+must place, in the relevant source files, a statement of the
+additional terms that apply to those files, or a notice indicating
+where to find the applicable terms.
+
+ Additional terms, permissive or non-permissive, may be stated in the
+form of a separately written license, or stated as exceptions;
+the above requirements apply either way.
+
+ 8. Termination.
+
+ You may not propagate or modify a covered work except as expressly
+provided under this License. Any attempt otherwise to propagate or
+modify it is void, and will automatically terminate your rights under
+this License (including any patent licenses granted under the third
+paragraph of section 11).
+
+ However, if you cease all violation of this License, then your
+license from a particular copyright holder is reinstated (a)
+provisionally, unless and until the copyright holder explicitly and
+finally terminates your license, and (b) permanently, if the copyright
+holder fails to notify you of the violation by some reasonable means
+prior to 60 days after the cessation.
+
+ Moreover, your license from a particular copyright holder is
+reinstated permanently if the copyright holder notifies you of the
+violation by some reasonable means, this is the first time you have
+received notice of violation of this License (for any work) from that
+copyright holder, and you cure the violation prior to 30 days after
+your receipt of the notice.
+
+ Termination of your rights under this section does not terminate the
+licenses of parties who have received copies or rights from you under
+this License. If your rights have been terminated and not permanently
+reinstated, you do not qualify to receive new licenses for the same
+material under section 10.
+
+ 9. Acceptance Not Required for Having Copies.
+
+ You are not required to accept this License in order to receive or
+run a copy of the Program. Ancillary propagation of a covered work
+occurring solely as a consequence of using peer-to-peer transmission
+to receive a copy likewise does not require acceptance. However,
+nothing other than this License grants you permission to propagate or
+modify any covered work. These actions infringe copyright if you do
+not accept this License. Therefore, by modifying or propagating a
+covered work, you indicate your acceptance of this License to do so.
+
+ 10. Automatic Licensing of Downstream Recipients.
+
+ Each time you convey a covered work, the recipient automatically
+receives a license from the original licensors, to run, modify and
+propagate that work, subject to this License. You are not responsible
+for enforcing compliance by third parties with this License.
+
+ An "entity transaction" is a transaction transferring control of an
+organization, or substantially all assets of one, or subdividing an
+organization, or merging organizations. If propagation of a covered
+work results from an entity transaction, each party to that
+transaction who receives a copy of the work also receives whatever
+licenses to the work the party's predecessor in interest had or could
+give under the previous paragraph, plus a right to possession of the
+Corresponding Source of the work from the predecessor in interest, if
+the predecessor has it or can get it with reasonable efforts.
+
+ You may not impose any further restrictions on the exercise of the
+rights granted or affirmed under this License. For example, you may
+not impose a license fee, royalty, or other charge for exercise of
+rights granted under this License, and you may not initiate litigation
+(including a cross-claim or counterclaim in a lawsuit) alleging that
+any patent claim is infringed by making, using, selling, offering for
+sale, or importing the Program or any portion of it.
+
+ 11. Patents.
+
+ A "contributor" is a copyright holder who authorizes use under this
+License of the Program or a work on which the Program is based. The
+work thus licensed is called the contributor's "contributor version".
+
+ A contributor's "essential patent claims" are all patent claims
+owned or controlled by the contributor, whether already acquired or
+hereafter acquired, that would be infringed by some manner, permitted
+by this License, of making, using, or selling its contributor version,
+but do not include claims that would be infringed only as a
+consequence of further modification of the contributor version. For
+purposes of this definition, "control" includes the right to grant
+patent sublicenses in a manner consistent with the requirements of
+this License.
+
+ Each contributor grants you a non-exclusive, worldwide, royalty-free
+patent license under the contributor's essential patent claims, to
+make, use, sell, offer for sale, import and otherwise run, modify and
+propagate the contents of its contributor version.
+
+ In the following three paragraphs, a "patent license" is any express
+agreement or commitment, however denominated, not to enforce a patent
+(such as an express permission to practice a patent or covenant not to
+sue for patent infringement). To "grant" such a patent license to a
+party means to make such an agreement or commitment not to enforce a
+patent against the party.
+
+ If you convey a covered work, knowingly relying on a patent license,
+and the Corresponding Source of the work is not available for anyone
+to copy, free of charge and under the terms of this License, through a
+publicly available network server or other readily accessible means,
+then you must either (1) cause the Corresponding Source to be so
+available, or (2) arrange to deprive yourself of the benefit of the
+patent license for this particular work, or (3) arrange, in a manner
+consistent with the requirements of this License, to extend the patent
+license to downstream recipients. "Knowingly relying" means you have
+actual knowledge that, but for the patent license, your conveying the
+covered work in a country, or your recipient's use of the covered work
+in a country, would infringe one or more identifiable patents in that
+country that you have reason to believe are valid.
+
+ If, pursuant to or in connection with a single transaction or
+arrangement, you convey, or propagate by procuring conveyance of, a
+covered work, and grant a patent license to some of the parties
+receiving the covered work authorizing them to use, propagate, modify
+or convey a specific copy of the covered work, then the patent license
+you grant is automatically extended to all recipients of the covered
+work and works based on it.
+
+ A patent license is "discriminatory" if it does not include within
+the scope of its coverage, prohibits the exercise of, or is
+conditioned on the non-exercise of one or more of the rights that are
+specifically granted under this License. You may not convey a covered
+work if you are a party to an arrangement with a third party that is
+in the business of distributing software, under which you make payment
+to the third party based on the extent of your activity of conveying
+the work, and under which the third party grants, to any of the
+parties who would receive the covered work from you, a discriminatory
+patent license (a) in connection with copies of the covered work
+conveyed by you (or copies made from those copies), or (b) primarily
+for and in connection with specific products or compilations that
+contain the covered work, unless you entered into that arrangement,
+or that patent license was granted, prior to 28 March 2007.
+
+ Nothing in this License shall be construed as excluding or limiting
+any implied license or other defenses to infringement that may
+otherwise be available to you under applicable patent law.
+
+ 12. No Surrender of Others' Freedom.
+
+ If conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot convey a
+covered work so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you may
+not convey it at all. For example, if you agree to terms that obligate you
+to collect a royalty for further conveying from those to whom you convey
+the Program, the only way you could satisfy both those terms and this
+License would be to refrain entirely from conveying the Program.
+
+ 13. Remote Network Interaction; Use with the GNU General Public License.
+
+ Notwithstanding any other provision of this License, if you modify the
+Program, your modified version must prominently offer all users
+interacting with it remotely through a computer network (if your version
+supports such interaction) an opportunity to receive the Corresponding
+Source of your version by providing access to the Corresponding Source
+from a network server at no charge, through some standard or customary
+means of facilitating copying of software. This Corresponding Source
+shall include the Corresponding Source for any work covered by version 3
+of the GNU General Public License that is incorporated pursuant to the
+following paragraph.
+
+ Notwithstanding any other provision of this License, you have
+permission to link or combine any covered work with a work licensed
+under version 3 of the GNU General Public License into a single
+combined work, and to convey the resulting work. The terms of this
+License will continue to apply to the part which is the covered work,
+but the work with which it is combined will remain governed by version
+3 of the GNU General Public License.
+
+ 14. Revised Versions of this License.
+
+ The Free Software Foundation may publish revised and/or new versions of
+the GNU Affero General Public License from time to time. Such new versions
+will be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+ Each version is given a distinguishing version number. If the
+Program specifies that a certain numbered version of the GNU Affero General
+Public License "or any later version" applies to it, you have the
+option of following the terms and conditions either of that numbered
+version or of any later version published by the Free Software
+Foundation. If the Program does not specify a version number of the
+GNU Affero General Public License, you may choose any version ever published
+by the Free Software Foundation.
+
+ If the Program specifies that a proxy can decide which future
+versions of the GNU Affero General Public License can be used, that proxy's
+public statement of acceptance of a version permanently authorizes you
+to choose that version for the Program.
+
+ Later license versions may give you additional or different
+permissions. However, no additional obligations are imposed on any
+author or copyright holder as a result of your choosing to follow a
+later version.
+
+ 15. Disclaimer of Warranty.
+
+ THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
+APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
+HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
+OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
+THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
+IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
+ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+ 16. Limitation of Liability.
+
+ IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
+THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
+GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
+USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
+DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
+PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
+EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
+SUCH DAMAGES.
+
+ 17. Interpretation of Sections 15 and 16.
+
+ If the disclaimer of warranty and limitation of liability provided
+above cannot be given local legal effect according to their terms,
+reviewing courts shall apply local law that most closely approximates
+an absolute waiver of all civil liability in connection with the
+Program, unless a warranty or assumption of liability accompanies a
+copy of the Program in return for a fee.
+
+ END OF TERMS AND CONDITIONS
+
+ How to Apply These Terms to Your New Programs
+
+ If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+ To do so, attach the following notices to the program. It is safest
+to attach them to the start of each source file to most effectively
+state the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+ <one line to give the program's name and a brief idea of what it does.>
+ Copyright (C) <year> <name of author>
+
+ This program is free software: you can redistribute it and/or modify
+ it under the terms of the GNU Affero General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU Affero General Public License for more details.
+
+ You should have received a copy of the GNU Affero General Public License
+ along with this program. If not, see <http://www.gnu.org/licenses/>.
+
+Also add information on how to contact you by electronic and paper mail.
+
+ If your software can interact with users remotely through a computer
+network, you should also make sure that it provides a way for users to
+get its source. For example, if your program is a web application, its
+interface could display a "Source" link that leads users to an archive
+of the code. There are many ways you could offer source, and different
+solutions will be better for different programs; see section 13 for the
+specific requirements.
+
+ You should also get your employer (if you work as a programmer) or school,
+if any, to sign a "copyright disclaimer" for the program, if necessary.
+For more information on this, and how to apply and follow the GNU AGPL, see
+<http://www.gnu.org/licenses/>. \ No newline at end of file
diff --git a/README.md b/README.md
index 718f6863..0c276354 100644
--- a/README.md
+++ b/README.md
@@ -8,8 +8,8 @@ HackMD
[![version][github-version-badge]][github-release-page]
-HackMD lets you create realtime collaborative markdown notes on all platforms.
-Inspired by Hackpad, with more focus on speed and flexibility.
+HackMD lets you create realtime collaborative markdown notes on all platforms.
+Inspired by Hackpad, with more focus on speed and flexibility.
Still in the early stage, feel free to fork or contribute to HackMD.
Thanks for using! :smile:
@@ -41,11 +41,11 @@ Thanks for using! :smile:
# Browsers Requirement
-- Chrome >= 47, Chrome for Android >= 47
-- Safari >= 9, iOS Safari >= 8.4
-- Firefox >= 44
-- IE >= 9, Edge >= 12
-- Opera >= 34, Opera Mini not supported
+- ![Chrome](http://browserbadge.com/chrome/47/18px) Chrome >= 47, Chrome for Android >= 47
+- ![Safari](http://browserbadge.com/safari/9/18px) Safari >= 9, iOS Safari >= 8.4
+- ![Firefox](http://browserbadge.com/firefox/44/18px) Firefox >= 44
+- ![IE](http://browserbadge.com/ie/9/18px) IE >= 9, Edge >= 12
+- ![Opera](http://browserbadge.com/opera/34/18px) Opera >= 34, Opera Mini not supported
- Android Browser >= 4.4
# Installation
@@ -74,11 +74,26 @@ You can quickly setup a sample heroku hackmd application by clicking the button
[![Deploy](https://www.herokucdn.com/deploy/button.svg)](https://heroku.com/deploy)
## HackMD by docker container
+[![Try in PWD](https://cdn.rawgit.com/play-with-docker/stacks/cff22438/assets/images/button.png)](http://play-with-docker.com?stack=https://github.com/hackmdio/docker-hackmd/raw/master/docker-compose.yml&stack_name=hackmd)
-[docker-hackmd](https://github.com/hackmdio/docker-hackmd)
-Before you go too far, here is the great docker repo for HackMD.
-With docker, you can deploy a server in minutes without any downtime.
+**Debian-based version:**
+
+[![latest](https://images.microbadger.com/badges/version/hackmdio/hackmd.svg)](https://microbadger.com/images/hackmdio/hackmd "Get your own version badge on microbadger.com") [![](https://images.microbadger.com/badges/image/hackmdio/hackmd.svg)](https://microbadger.com/images/hackmdio/hackmd "Get your own image badge on microbadger.com")
+
+
+**Alpine-based version:**
+
+[![latest-alpine](https://images.microbadger.com/badges/version/hackmdio/hackmd:latest-alpine.svg)](https://microbadger.com/images/hackmdio/hackmd:latest-alpine "Get your own version badge on microbadger.com") [![](https://images.microbadger.com/badges/image/hackmdio/hackmd:latest-alpine.svg)](https://microbadger.com/images/hackmdio/hackmd:latest-alpine "Get your own image badge on microbadger.com")
+
+The easiest way to setup HackMD using docker are using the following three commands:
+
+```console
+git clone https://github.com/hackmdio/docker-hackmd.git
+cd docker-hackmd
+docker-compose up
+```
+Read more about it in the [docker repository…](https://github.com/hackmdio/docker-hackmd)
# Upgrade
@@ -97,13 +112,13 @@ If you are upgrading HackMD from an older version, follow these steps:
* [migration-to-0.5.0](https://github.com/hackmdio/migration-to-0.5.0)
-We don't use LZString to compress socket.io data and DB data after version 0.5.0.
+We don't use LZString to compress socket.io data and DB data after version 0.5.0.
Please run the migration tool if you're upgrading from the old version.
* [migration-to-0.4.0](https://github.com/hackmdio/migration-to-0.4.0)
-We've dropped MongoDB after version 0.4.0.
-So here is the migration tool for you to transfer the old DB data to the new DB.
+We've dropped MongoDB after version 0.4.0.
+So here is the migration tool for you to transfer the old DB data to the new DB.
This tool is also used for official service.
# Configuration
@@ -141,6 +156,9 @@ There are some configs you need to change in the files below
| HMD_GITLAB_BASEURL | no example | GitLab authentication endpoint, set to use other endpoint than GitLab.com (optional) |
| HMD_GITLAB_CLIENTID | no example | GitLab API client id |
| HMD_GITLAB_CLIENTSECRET | no example | GitLab API client secret |
+| HMD_MATTERMOST_BASEURL | no example | Mattermost authentication endpoint |
+| HMD_MATTERMOST_CLIENTID | no example | Mattermost API client id |
+| HMD_MATTERMOST_CLIENTSECRET | no example | Mattermost API client secret |
| HMD_DROPBOX_CLIENTID | no example | Dropbox API client id |
| HMD_DROPBOX_CLIENTSECRET | no example | Dropbox API client secret |
| HMD_GOOGLE_CLIENTID | no example | Google API client id |
@@ -151,9 +169,19 @@ There are some configs you need to change in the files below
| HMD_LDAP_TOKENSECRET | `supersecretkey` | secret used for generating access/refresh tokens |
| HMD_LDAP_SEARCHBASE | `o=users,dc=example,dc=com` | LDAP directory to begin search from |
| HMD_LDAP_SEARCHFILTER | `(uid={{username}})` | LDAP filter to search with |
-| HMD_LDAP_SEARCHATTRIBUTES | no example | LDAP attributes to search with |
+| HMD_LDAP_SEARCHATTRIBUTES | `displayName, mail` | LDAP attributes to search with (use comma to separate) |
| HMD_LDAP_TLS_CA | `server-cert.pem, root.pem` | Root CA for LDAP TLS in PEM format (use comma to separate) |
| HMD_LDAP_PROVIDERNAME | `My institution` | Optional name to be displayed at login form indicating the LDAP provider |
+| HMD_SAML_IDPSSOURL | `https://idp.example.com/sso` | authentication endpoint of IdP. for details, see [guide](docs/guides/auth.md#saml-onelogin). |
+| HMD_SAML_IDPCERT | `/path/to/cert.pem` | certificate file path of IdP in PEM format |
+| HMD_SAML_ISSUER | no example | identity of the service provider (optional, default: serverurl)" |
+| HMD_SAML_IDENTIFIERFORMAT | no example | name identifier format (optional, default: `urn:oasis:names:tc:SAML:1.1:nameid-format:emailAddress`) |
+| HMD_SAML_GROUPATTRIBUTE | `memberOf` | attribute name for group list (optional) |
+| HMD_SAML_REQUIREDGROUPS | `Hackmd-users` | group names that allowed (use vertical bar to separate) (optional) |
+| HMD_SAML_EXTERNALGROUPS | `Temporary-staff` | group names that not allowed (use vertical bar to separate) (optional) |
+| HMD_SAML_ATTRIBUTE_ID | `sAMAccountName` | attribute map for `id` (optional, default: NameID of SAML response) |
+| HMD_SAML_ATTRIBUTE_USERNAME | `mailNickname` | attribute map for `username` (optional, default: NameID of SAML response) |
+| HMD_SAML_ATTRIBUTE_EMAIL | `mail` | attribute map for `email` (optional, default: NameID of SAML response if `HMD_SAML_IDENTIFIERFORMAT` is default) |
| HMD_IMGUR_CLIENTID | no example | Imgur API client id |
| HMD_EMAIL | `true` or `false` | set to allow email signin |
| HMD_ALLOW_PDF_EXPORT | `true` or `false` | Enable or disable PDF exports |
@@ -216,8 +244,8 @@ There are some configs you need to change in the files below
| service | settings location | description |
| ------- | --------- | ----------- |
-| facebook, twitter, github, gitlab, dropbox, google, ldap | environment variables or `config.json` | for signin |
-| imgur | environment variables or `config.json` | for image upload |
+| facebook, twitter, github, gitlab, mattermost, dropbox, google, ldap, saml | environment variables or `config.json` | for signin |
+| imgur, s3 | environment variables or `config.json` | for image upload |
| google drive(`google/apiKey`, `google/clientID`), dropbox(`dropbox/appKey`) | `config.json` | for export and import |
## Third-party integration oauth callback urls
@@ -228,8 +256,10 @@ There are some configs you need to change in the files below
| twitter | `/auth/twitter/callback` |
| github | `/auth/github/callback` |
| gitlab | `/auth/gitlab/callback` |
+| mattermost | `/auth/mattermost/callback` |
| dropbox | `/auth/dropbox/callback` |
| google | `/auth/google/callback` |
+| saml | `/auth/saml/callback` |
# Developer Notes
@@ -249,16 +279,16 @@ hackmd/
## Operational Transformation
-From 0.3.2, we started supporting operational transformation.
-It makes concurrent editing safe and will not break up other users' operations.
-Additionally, now can show other clients' selections.
+From 0.3.2, we started supporting operational transformation.
+It makes concurrent editing safe and will not break up other users' operations.
+Additionally, now can show other clients' selections.
See more at [http://operational-transformation.github.io/](http://operational-transformation.github.io/)
# License
-**License under MIT.**
+**License under AGPL.**
[gitter-image]: https://badges.gitter.im/Join%20Chat.svg
[gitter-url]: https://gitter.im/hackmdio/hackmd?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge
diff --git a/app.json b/app.json
index d1804c52..b2116eb6 100644
--- a/app.json
+++ b/app.json
@@ -100,6 +100,18 @@
"description": "GitLab API client scope (optional)",
"required": false
},
+ "HMD_MATTERMOST_BASEURL": {
+ "description": "Mattermost authentication endpoint",
+ "required": false
+ },
+ "HMD_MATTERMOST_CLIENTID": {
+ "description": "Mattermost API client id",
+ "required": false
+ },
+ "HMD_MATTERMOST_CLIENTSECRET": {
+ "description": "Mattermost API client secret",
+ "required": false
+ },
"HMD_DROPBOX_CLIENTID": {
"description": "Dropbox API client id",
"required": false
diff --git a/config.json.example b/config.json.example
index e2d774c7..63eb443f 100644
--- a/config.json.example
+++ b/config.json.example
@@ -48,6 +48,11 @@
"clientSecret": "change this",
"scope": "use 'read_user' scope for auth user only or remove this property if you need gitlab snippet import/export support (will result to be default scope 'api')"
},
+ "mattermost": {
+ "baseURL": "change this",
+ "clientID": "change this",
+ "clientSecret": "change this"
+ },
"dropbox": {
"clientID": "change this",
"clientSecret": "change this",
@@ -65,13 +70,33 @@
"tokenSecret": "change this",
"searchBase": "change this",
"searchFilter": "change this",
- "searchAttributes": "change this",
+ "searchAttributes": ["change this"],
"tlsOptions": {
"changeme": "See https://nodejs.org/api/tls.html#tls_tls_connect_options_callback"
}
},
+ "saml": {
+ "idpSsoUrl": "change: authentication endpoint of IdP",
+ "idpCert": "change: certificate file path of IdP in PEM format",
+ "issuer": "change or delete: identity of the service provider (default: serverurl)",
+ "identifierFormat": "change or delete: name identifier format (default: 'urn:oasis:names:tc:SAML:1.1:nameid-format:emailAddress')",
+ "groupAttribute": "change or delete: attribute name for group list (ex: memberOf)",
+ "requiredGroups": [ "change or delete: group names that allowed" ],
+ "externalGroups": [ "change or delete: group names that not allowed" ],
+ "attribute": {
+ "id": "change or delete this: attribute map for `id` (default: NameID)",
+ "username": "change or delete this: attribute map for `username` (default: NameID)",
+ "email": "change or delete this: attribute map for `email` (default: NameID)"
+ }
+ },
"imgur": {
"clientID": "change this"
- }
+ },
+ "s3": {
+ "accessKeyId": "change this",
+ "secretAccessKey": "change this",
+ "region": "change this"
+ },
+ "s3bucket": "change this"
}
}
diff --git a/contribute/developer-certificate-of-origin b/contribute/developer-certificate-of-origin
new file mode 100644
index 00000000..a6bbb984
--- /dev/null
+++ b/contribute/developer-certificate-of-origin
@@ -0,0 +1,35 @@
+Developer Certificate of Origin
+Version 1.1
+
+Copyright (C) 2004, 2006 The Linux Foundation and its contributors.
+660 York Street, Suite 102,
+San Francisco, CA 94110 USA
+
+Everyone is permitted to copy and distribute verbatim copies of this
+license document, but changing it is not allowed.
+
+Developer's Certificate of Origin 1.1
+
+By making a contribution to this project, I certify that:
+
+(a) The contribution was created in whole or in part by me and I
+ have the right to submit it under the open source license
+ indicated in the file; or
+
+(b) The contribution is based upon previous work that, to the best
+ of my knowledge, is covered under an appropriate open source
+ license and I have the right under that license to submit that
+ work with modifications, whether created in whole or in part
+ by me, under the same open source license (unless I am
+ permitted to submit under a different license), as indicated
+ in the file; or
+
+(c) The contribution was provided directly to me by some other
+ person who certified (a), (b) or (c) and I have not modified
+ it.
+
+(d) I understand and agree that this project and the contribution
+ are public and that a record of the contribution (including all
+ personal information I submit with it, including my sign-off) is
+ maintained indefinitely and may be redistributed consistent with
+ this project or the open source license(s) involved.
diff --git a/docs/guides/auth.md b/docs/guides/auth.md
new file mode 100644
index 00000000..4f9ce445
--- /dev/null
+++ b/docs/guides/auth.md
@@ -0,0 +1,212 @@
+# Guide - Authentication
+
+### Twitter
+1. Sign-in or sign-up for a Twitter account
+2. Go to the Twitter Application management page [here](https://apps.twitter.com/)
+3. Click on the **Create New App** button to create a new Twitter app:
+
+![create-twitter-app](images/auth/create-twitter-app.png)
+
+4. Fill out the create application form, check the developer agreement box, and click **Create Your Twitter Application**
+
+![register-twitter-application](images/auth/register-twitter-application.png)
+
+*Note: you may have to register your phone number with Twitter to create a Twitter application*
+
+To do this Click your profile icon --> Settings and privacy --> Mobile --> Select Country/region --> Enter phone number --> Click Continue
+
+5. After you receive confirmation that the Twitter application was created, click **Keys and Access Tokens**
+
+![twitter-app-confirmation](images/auth/twitter-app-confirmation.png)
+
+6. Obtain your Twitter Consumer Key and Consumer Secret
+
+![twitter-app-keys](images/auth/twitter-app-keys.png)
+
+7. Add your Consumer Key and Consumer Secret to your config.json file or pass them as environment variables:
+ * config.json:
+ ````javascript
+ {
+ "production": {
+ "twitter": {
+ "consumerKey": "esTCJFXXXXXXXXXXXXXXXXXXX",
+ "consumerSecret": "zpCs4tU86pRVXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX"
+ }
+ }
+ }
+ ````
+ * environment variables:
+ ````
+ HMD_TWITTER_CONSUMERKEY=esTCJFXXXXXXXXXXXXXXXXXXX
+ HMD_TWITTER_CONSUMERSECRET=zpCs4tU86pRVXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
+ ````
+
+### GitHub
+1. Sign-in or sign-up for a GitHub account
+2. Navigate to developer settings in your GitHub account [here](https://github.com/settings/developers) and select the "OAuth Apps" tab
+3. Click on the **New OAuth App** button, to create a new OAuth App:
+
+![create-oauth-app](images/auth/create-oauth-app.png)
+
+4. Fill out the new OAuth application registration form, and click **Register Application**
+
+![register-oauth-application-form](images/auth/register-oauth-application-form.png)
+
+*Note: The callback URL is <your-hackmd-url>/auth/github/callback*
+
+5. After successfully registering the application, you'll receive the Client ID and Client Secret for the application
+
+![application-page](images/auth/application-page.png)
+
+6. Add the Client ID and Client Secret to your config.json file or pass them as environment variables
+ * config.json:
+ ````javascript
+ {
+ "production": {
+ "github": {
+ "clientID": "3747d30eaccXXXXXXXXX",
+ "clientSecret": "2a8e682948eee0c580XXXXXXXXXXXXXXXXXXXXXX"
+ }
+ }
+ }
+ ````
+ * environment variables:
+ ````
+ HMD_GITHUB_CLIENTID=3747d30eaccXXXXXXXXX
+ HMD_GITHUB_CLIENTSECRET=2a8e682948eee0c580XXXXXXXXXXXXXXXXXXXXXX
+ ````
+
+### SAML (OneLogin)
+1. Sign-in or sign-up for an OneLogin account. (available free trial for 2 weeks)
+2. Go to the administration page.
+3. Select the **APPS** menu and click on the **Add Apps**.
+
+![onelogin-add-app](images/auth/onelogin-add-app.png)
+
+4. Find "SAML Test Connector (SP)" for template of settings and select it.
+
+![onelogin-select-template](images/auth/onelogin-select-template.png)
+
+5. Edit display name and icons for OneLogin dashboard as you want, and click **SAVE**.
+
+![onelogin-edit-app-name](images/auth/onelogin-edit-app-name.png)
+
+6. After that other tabs will appear, click the **Configuration**, and fill out the below items, and click **SAVE**.
+ * RelayState: The base URL of your hackmd, which is issuer. (last slash is not needed)
+ * ACS (Consumer) URL Validator: The callback URL of your hackmd. (serverurl + /auth/saml/callback)
+ * ACS (Consumer) URL: same as above.
+ * Login URL: login URL(SAML requester) of your hackmd. (serverurl + /auth/saml)
+
+![onelogin-edit-sp-metadata](images/auth/onelogin-edit-sp-metadata.png)
+
+7. The registration is completed. Next, click **SSO** and copy or download the items below.
+ * X.509 Certificate: Click **View Details** and **DOWNLOAD** or copy the content of certificate ....(A)
+ * SAML 2.0 Endpoint (HTTP): Copy the URL ....(B)
+
+![onelogin-copy-idp-metadata](images/auth/onelogin-copy-idp-metadata.png)
+
+8. In your hackmd server, create IdP certificate file from (A)
+9. Add the IdP URL (B) and the Idp certificate file path to your config.json file or pass them as environment variables.
+ * config.json:
+ ````javascript
+ {
+ "production": {
+ "saml": {
+ "idpSsoUrl": "https://*******.onelogin.com/trust/saml2/http-post/sso/******",
+ "idpCert": "/path/to/idp_cert.pem"
+ }
+ }
+ }
+ ````
+ * environment variables
+ ````
+ HMD_SAML_IDPSSOURL=https://*******.onelogin.com/trust/saml2/http-post/sso/******
+ HMD_SAML_IDPCERT=/path/to/idp_cert.pem
+ ````
+10. Try sign-in with SAML from your hackmd sign-in button or OneLogin dashboard (like the screenshot below).
+
+![onelogin-use-dashboard](images/auth/onelogin-use-dashboard.png)
+
+### SAML (Other cases)
+The basic procedure is the same as the case of OneLogin which is mentioned above. If you want to match your IdP, you can use more configurations as below.
+
+* If your IdP accepts metadata XML of the service provider to ease configuraion, use this url to download metadata XML.
+ * {{your-serverurl}}/auth/saml/metadata
+ * _Note: If not accessable from IdP, download to local once and upload to IdP._
+* Change the value of `issuer`, `identifierFormat` to match your IdP.
+ * `issuer`: A unique id to identify the application to the IdP, which is the base URL of your HackMD as default
+ * `identifierFormat`: A format of unique id to identify the user of IdP, which is the format based on email address as default. It is recommend that you use as below.
+ * urn:oasis:names:tc:SAML:1.1:nameid-format:emailAddress (default)
+ * urn:oasis:names:tc:SAML:1.1:nameid-format:unspecified
+ * config.json:
+ ````javascript
+ {
+ "production": {
+ "saml": {
+ /* omitted */
+ "issuer": "myhackmd"
+ "identifierFormat": "urn:oasis:names:tc:SAML:1.1:nameid-format:unspecified"
+ }
+ }
+ }
+ ````
+ * environment variables
+ ````
+ HMD_SAML_ISSUER=myhackmd
+ HMD_SAML_IDENTIFIERFORMAT=urn:oasis:names:tc:SAML:1.1:nameid-format:unspecified
+ ````
+
+* Change mapping of attribute names to customize the displaying user name and email address to match your IdP.
+ * `attribute`: A dictionary to map attribute names
+ * `attribute.id`: A primary key of user table for your HackMD
+ * `attribute.username`: Attribute name of displaying user name on HackMD
+ * `attribute.email`: Attribute name of email address, which will be also used for Gravatar
+ * _Note: Default value of all attributes is NameID of SAML response, which is email address if `idfentifierFormat` is default._
+ * config.json:
+ ````javascript
+ {
+ "production": {
+ "saml": {
+ /* omitted */
+ "attribute": {
+ "id": "sAMAccountName",
+ "username": "displayName",
+ "email": "mail"
+ }
+ }
+ }
+ }
+ ````
+ * environment variables
+ ````
+ HMD_SAML_ATTRIBUTE_ID=sAMAccountName
+ HMD_SAML_ATTRIBUTE_USERNAME=nickName
+ HMD_SAML_ATTRIBUTE_EMAIL=mail
+ ````
+
+* If you want to controll permission by group membership, add group attribute name and required group (allowed) or external group (not allowed).
+ * `groupAttribute`: An attribute name of group membership
+ * `requiredGroups`: Group names array for allowed access to HackMD. Use vertical bar to separate for environment variables.
+ * `externalGroups`: Group names array for not allowed access to HackMD. Use vertical bar to separate for environment variables.
+ * _Note: Evaluates `externalGroups` first_
+ * config.json:
+ ````javascript
+ {
+ "production": {
+ "saml": {
+ /* omitted */
+ "groupAttribute": "memberOf",
+ "requiredGroups": [ "hackmd-users", "board-members" ],
+ "externalGroups": [ "temporary-staff" ]
+ }
+ }
+ }
+ ````
+ * environment variables
+ ````
+ HMD_SAML_GROUPATTRIBUTE=memberOf
+ HMD_SAML_REQUIREDGROUPS=hackmd-users|board-members
+ HMD_SAML_EXTERNALGROUPS=temporary-staff
+ ````
+
+
diff --git a/docs/guides/images/auth/application-page.png b/docs/guides/images/auth/application-page.png
new file mode 100644
index 00000000..a57da017
--- /dev/null
+++ b/docs/guides/images/auth/application-page.png
Binary files differ
diff --git a/docs/guides/images/auth/create-oauth-app.png b/docs/guides/images/auth/create-oauth-app.png
new file mode 100644
index 00000000..07d0b511
--- /dev/null
+++ b/docs/guides/images/auth/create-oauth-app.png
Binary files differ
diff --git a/docs/guides/images/auth/create-twitter-app.png b/docs/guides/images/auth/create-twitter-app.png
new file mode 100644
index 00000000..c555464e
--- /dev/null
+++ b/docs/guides/images/auth/create-twitter-app.png
Binary files differ
diff --git a/docs/guides/images/auth/onelogin-add-app.png b/docs/guides/images/auth/onelogin-add-app.png
new file mode 100644
index 00000000..356bb852
--- /dev/null
+++ b/docs/guides/images/auth/onelogin-add-app.png
Binary files differ
diff --git a/docs/guides/images/auth/onelogin-copy-idp-metadata.png b/docs/guides/images/auth/onelogin-copy-idp-metadata.png
new file mode 100644
index 00000000..7185f537
--- /dev/null
+++ b/docs/guides/images/auth/onelogin-copy-idp-metadata.png
Binary files differ
diff --git a/docs/guides/images/auth/onelogin-edit-app-name.png b/docs/guides/images/auth/onelogin-edit-app-name.png
new file mode 100644
index 00000000..634d1916
--- /dev/null
+++ b/docs/guides/images/auth/onelogin-edit-app-name.png
Binary files differ
diff --git a/docs/guides/images/auth/onelogin-edit-sp-metadata.png b/docs/guides/images/auth/onelogin-edit-sp-metadata.png
new file mode 100644
index 00000000..111580b1
--- /dev/null
+++ b/docs/guides/images/auth/onelogin-edit-sp-metadata.png
Binary files differ
diff --git a/docs/guides/images/auth/onelogin-select-template.png b/docs/guides/images/auth/onelogin-select-template.png
new file mode 100644
index 00000000..13401816
--- /dev/null
+++ b/docs/guides/images/auth/onelogin-select-template.png
Binary files differ
diff --git a/docs/guides/images/auth/onelogin-use-dashboard.png b/docs/guides/images/auth/onelogin-use-dashboard.png
new file mode 100644
index 00000000..ea9038ff
--- /dev/null
+++ b/docs/guides/images/auth/onelogin-use-dashboard.png
Binary files differ
diff --git a/docs/guides/images/auth/register-oauth-application-form.png b/docs/guides/images/auth/register-oauth-application-form.png
new file mode 100644
index 00000000..bd27fd70
--- /dev/null
+++ b/docs/guides/images/auth/register-oauth-application-form.png
Binary files differ
diff --git a/docs/guides/images/auth/register-twitter-application.png b/docs/guides/images/auth/register-twitter-application.png
new file mode 100644
index 00000000..442eb680
--- /dev/null
+++ b/docs/guides/images/auth/register-twitter-application.png
Binary files differ
diff --git a/docs/guides/images/auth/twitter-app-confirmation.png b/docs/guides/images/auth/twitter-app-confirmation.png
new file mode 100644
index 00000000..e24e9d56
--- /dev/null
+++ b/docs/guides/images/auth/twitter-app-confirmation.png
Binary files differ
diff --git a/docs/guides/images/auth/twitter-app-keys.png b/docs/guides/images/auth/twitter-app-keys.png
new file mode 100644
index 00000000..e13be110
--- /dev/null
+++ b/docs/guides/images/auth/twitter-app-keys.png
Binary files differ
diff --git a/lib/config/default.js b/lib/config/default.js
index e7e2e4b3..d04485ce 100644
--- a/lib/config/default.js
+++ b/lib/config/default.js
@@ -74,6 +74,11 @@ module.exports = {
clientSecret: undefined,
scope: undefined
},
+ mattermost: {
+ baseURL: undefined,
+ clientID: undefined,
+ clientSecret: undefined
+ },
dropbox: {
clientID: undefined,
clientSecret: undefined
@@ -93,6 +98,20 @@ module.exports = {
searchAttributes: undefined,
tlsca: undefined
},
+ saml: {
+ idpSsoUrl: undefined,
+ idpCert: undefined,
+ issuer: undefined,
+ identifierFormat: 'urn:oasis:names:tc:SAML:1.1:nameid-format:emailAddress',
+ groupAttribute: undefined,
+ externalGroups: [],
+ requiredGroups: [],
+ attribute: {
+ id: undefined,
+ username: undefined,
+ email: undefined
+ }
+ },
email: true,
allowemailregister: true,
allowpdfexport: true
diff --git a/lib/config/dockerSecret.js b/lib/config/dockerSecret.js
index eea2fafd..ac54fd19 100644
--- a/lib/config/dockerSecret.js
+++ b/lib/config/dockerSecret.js
@@ -38,6 +38,10 @@ if (fs.existsSync(basePath)) {
clientID: getSecret('gitlab_clientID'),
clientSecret: getSecret('gitlab_clientSecret')
},
+ mattermost: {
+ clientID: getSecret('mattermost_clientID'),
+ clientSecret: getSecret('mattermost_clientSecret')
+ },
dropbox: {
clientID: getSecret('dropbox_clientID'),
clientSecret: getSecret('dropbox_clientSecret')
diff --git a/lib/config/environment.js b/lib/config/environment.js
index 6f33d140..45a2b0e0 100644
--- a/lib/config/environment.js
+++ b/lib/config/environment.js
@@ -1,6 +1,6 @@
'use strict'
-const {toBooleanConfig} = require('./utils')
+const {toBooleanConfig, toArrayConfig} = require('./utils')
module.exports = {
domain: process.env.HMD_DOMAIN,
@@ -15,7 +15,7 @@ module.exports = {
preload: toBooleanConfig(process.env.HMD_HSTS_PRELOAD)
},
protocolusessl: toBooleanConfig(process.env.HMD_PROTOCOL_USESSL),
- alloworigin: process.env.HMD_ALLOW_ORIGIN ? process.env.HMD_ALLOW_ORIGIN.split(',') : undefined,
+ alloworigin: toArrayConfig(process.env.HMD_ALLOW_ORIGIN),
usecdn: toBooleanConfig(process.env.HMD_USECDN),
allowanonymous: toBooleanConfig(process.env.HMD_ALLOW_ANONYMOUS),
allowfreeurl: toBooleanConfig(process.env.HMD_ALLOW_FREEURL),
@@ -49,6 +49,11 @@ module.exports = {
clientSecret: process.env.HMD_GITLAB_CLIENTSECRET,
scope: process.env.HMD_GITLAB_SCOPE
},
+ mattermost: {
+ baseURL: process.env.HMD_MATTERMOST_BASEURL,
+ clientID: process.env.HMD_MATTERMOST_CLIENTID,
+ clientSecret: process.env.HMD_MATTERMOST_CLIENTSECRET
+ },
dropbox: {
clientID: process.env.HMD_DROPBOX_CLIENTID,
clientSecret: process.env.HMD_DROPBOX_CLIENTSECRET
@@ -65,9 +70,23 @@ module.exports = {
tokenSecret: process.env.HMD_LDAP_TOKENSECRET,
searchBase: process.env.HMD_LDAP_SEARCHBASE,
searchFilter: process.env.HMD_LDAP_SEARCHFILTER,
- searchAttributes: process.env.HMD_LDAP_SEARCHATTRIBUTES,
+ searchAttributes: toArrayConfig(process.env.HMD_LDAP_SEARCHATTRIBUTES),
tlsca: process.env.HMD_LDAP_TLS_CA
},
+ saml: {
+ idpSsoUrl: process.env.HMD_SAML_IDPSSOURL,
+ idpCert: process.env.HMD_SAML_IDPCERT,
+ issuer: process.env.HMD_SAML_ISSUER,
+ identifierFormat: process.env.HMD_SAML_IDENTIFIERFORMAT,
+ groupAttribute: process.env.HMD_SAML_GROUPATTRIBUTE,
+ externalGroups: toArrayConfig(process.env.HMD_SAML_EXTERNALGROUPS, '|', []),
+ requiredGroups: toArrayConfig(process.env.HMD_SAML_REQUIREDGROUPS, '|', []),
+ attribute: {
+ id: process.env.HMD_SAML_ATTRIBUTE_ID,
+ username: process.env.HMD_SAML_ATTRIBUTE_USERNAME,
+ email: process.env.HMD_SAML_ATTRIBUTE_EMAIL
+ }
+ },
email: toBooleanConfig(process.env.HMD_EMAIL),
allowemailregister: toBooleanConfig(process.env.HMD_ALLOW_EMAIL_REGISTER),
allowpdfexport: toBooleanConfig(process.env.HMD_ALLOW_PDF_EXPORT)
diff --git a/lib/config/index.js b/lib/config/index.js
index dfad28ed..3ac3de53 100644
--- a/lib/config/index.js
+++ b/lib/config/index.js
@@ -90,7 +90,9 @@ config.isTwitterEnable = config.twitter.consumerKey && config.twitter.consumerSe
config.isEmailEnable = config.email
config.isGitHubEnable = config.github.clientID && config.github.clientSecret
config.isGitLabEnable = config.gitlab.clientID && config.gitlab.clientSecret
+config.isMattermostEnable = config.mattermost.clientID && config.mattermost.clientSecret
config.isLDAPEnable = config.ldap.url
+config.isSAMLEnable = config.saml.idpSsoUrl
config.isPDFExportEnable = config.allowpdfexport
// generate correct path
diff --git a/lib/config/utils.js b/lib/config/utils.js
index 11bbd8cb..9ff2f96d 100644
--- a/lib/config/utils.js
+++ b/lib/config/utils.js
@@ -6,3 +6,10 @@ exports.toBooleanConfig = function toBooleanConfig (configValue) {
}
return configValue
}
+
+exports.toArrayConfig = function toArrayConfig (configValue, separator = ',', fallback) {
+ if (configValue && typeof configValue === 'string') {
+ return (configValue.split(separator).map(arrayItem => arrayItem.trim()))
+ }
+ return fallback
+}
diff --git a/lib/models/user.js b/lib/models/user.js
index e59b86cc..f421fe43 100644
--- a/lib/models/user.js
+++ b/lib/models/user.js
@@ -111,6 +111,15 @@ module.exports = function (sequelize, DataTypes) {
photo = letterAvatars(profile.username)
}
break
+ case 'mattermost':
+ photo = profile.avatarUrl
+ if (photo) {
+ if (bigger) photo = photo.replace(/(\?s=)\d*$/i, '$1400')
+ else photo = photo.replace(/(\?s=)\d*$/i, '$196')
+ } else {
+ photo = letterAvatars(profile.username)
+ }
+ break
case 'dropbox':
// no image api provided, use gravatar
photo = 'https://www.gravatar.com/avatar/' + md5(profile.emails[0].value)
@@ -134,6 +143,15 @@ module.exports = function (sequelize, DataTypes) {
photo = letterAvatars(profile.username)
}
break
+ case 'saml':
+ if (profile.emails[0]) {
+ photo = 'https://www.gravatar.com/avatar/' + md5(profile.emails[0])
+ if (bigger) photo += '?s=400'
+ else photo += '?s=96'
+ } else {
+ photo = letterAvatars(profile.username)
+ }
+ break
}
return photo
},
diff --git a/lib/response.js b/lib/response.js
index 9e39ffb5..9f3d5a44 100755
--- a/lib/response.js
+++ b/lib/response.js
@@ -64,9 +64,11 @@ function showIndex (req, res, next) {
twitter: config.isTwitterEnable,
github: config.isGitHubEnable,
gitlab: config.isGitLabEnable,
+ mattermost: config.isMattermostEnable,
dropbox: config.isDropboxEnable,
google: config.isGoogleEnable,
ldap: config.isLDAPEnable,
+ saml: config.isSAMLEnable,
email: config.isEmailEnable,
allowemailregister: config.allowemailregister,
allowpdfexport: config.allowpdfexport,
@@ -95,9 +97,11 @@ function responseHackMD (res, note) {
twitter: config.isTwitterEnable,
github: config.isGitHubEnable,
gitlab: config.isGitLabEnable,
+ mattermost: config.isMattermostEnable,
dropbox: config.isDropboxEnable,
google: config.isGoogleEnable,
ldap: config.isLDAPEnable,
+ saml: config.isSAMLEnable,
email: config.isEmailEnable,
allowemailregister: config.allowemailregister,
allowpdfexport: config.allowpdfexport
diff --git a/lib/web/auth/index.js b/lib/web/auth/index.js
index b5ca8434..db5ff11d 100644
--- a/lib/web/auth/index.js
+++ b/lib/web/auth/index.js
@@ -33,9 +33,11 @@ if (config.isFacebookEnable) authRouter.use(require('./facebook'))
if (config.isTwitterEnable) authRouter.use(require('./twitter'))
if (config.isGitHubEnable) authRouter.use(require('./github'))
if (config.isGitLabEnable) authRouter.use(require('./gitlab'))
+if (config.isMattermostEnable) authRouter.use(require('./mattermost'))
if (config.isDropboxEnable) authRouter.use(require('./dropbox'))
if (config.isGoogleEnable) authRouter.use(require('./google'))
if (config.isLDAPEnable) authRouter.use(require('./ldap'))
+if (config.isSAMLEnable) authRouter.use(require('./saml'))
if (config.isEmailEnable) authRouter.use(require('./email'))
// logout
diff --git a/lib/web/auth/mattermost/index.js b/lib/web/auth/mattermost/index.js
new file mode 100644
index 00000000..9ccf3de5
--- /dev/null
+++ b/lib/web/auth/mattermost/index.js
@@ -0,0 +1,49 @@
+'use strict'
+
+const Router = require('express').Router
+const passport = require('passport')
+const Mattermost = require('mattermost')
+const OAuthStrategy = require('passport-oauth2').Strategy
+const config = require('../../../config')
+const {setReturnToFromReferer, passportGeneralCallback} = require('../utils')
+
+const mattermost = new Mattermost.Client()
+
+let mattermostAuth = module.exports = Router()
+
+let mattermostStrategy = new OAuthStrategy({
+ authorizationURL: config.mattermost.baseURL + '/oauth/authorize',
+ tokenURL: config.mattermost.baseURL + '/oauth/access_token',
+ clientID: config.mattermost.clientID,
+ clientSecret: config.mattermost.clientSecret,
+ callbackURL: config.serverurl + '/auth/mattermost/callback'
+}, passportGeneralCallback)
+
+mattermostStrategy.userProfile = (accessToken, done) => {
+ mattermost.setUrl(config.mattermost.baseURL)
+ mattermost.token = accessToken
+ mattermost.useHeaderToken()
+ mattermost.getMe(
+ (data) => {
+ done(null, data)
+ },
+ (err) => {
+ done(err)
+ }
+ )
+}
+
+passport.use(mattermostStrategy)
+
+mattermostAuth.get('/auth/mattermost', function (req, res, next) {
+ setReturnToFromReferer(req)
+ passport.authenticate('oauth2')(req, res, next)
+})
+
+// mattermost auth callback
+mattermostAuth.get('/auth/mattermost/callback',
+ passport.authenticate('oauth2', {
+ successReturnToOrRedirect: config.serverurl + '/',
+ failureRedirect: config.serverurl + '/'
+ })
+)
diff --git a/lib/web/auth/saml/index.js b/lib/web/auth/saml/index.js
new file mode 100644
index 00000000..386293ae
--- /dev/null
+++ b/lib/web/auth/saml/index.js
@@ -0,0 +1,95 @@
+'use strict'
+
+const Router = require('express').Router
+const passport = require('passport')
+const SamlStrategy = require('passport-saml').Strategy
+const config = require('../../../config')
+const models = require('../../../models')
+const logger = require('../../../logger')
+const {urlencodedParser} = require('../../utils')
+const fs = require('fs')
+const intersection = function (array1, array2) { return array1.filter((n) => array2.includes(n)) }
+
+let samlAuth = module.exports = Router()
+
+passport.use(new SamlStrategy({
+ callbackUrl: config.serverurl + '/auth/saml/callback',
+ entryPoint: config.saml.idpSsoUrl,
+ issuer: config.saml.issuer || config.serverurl,
+ cert: fs.readFileSync(config.saml.idpCert, 'utf-8'),
+ identifierFormat: config.saml.identifierFormat
+}, function (user, done) {
+ // check authorization if needed
+ if (config.saml.externalGroups && config.saml.grouptAttribute) {
+ var externalGroups = intersection(config.saml.externalGroups, user[config.saml.groupAttribute])
+ if (externalGroups.length > 0) {
+ logger.error('saml permission denied: ' + externalGroups.join(', '))
+ return done('Permission denied', null)
+ }
+ }
+ if (config.saml.requiredGroups && config.saml.grouptAttribute) {
+ if (intersection(config.saml.requiredGroups, user[config.saml.groupAttribute]).length === 0) {
+ logger.error('saml permission denied')
+ return done('Permission denied', null)
+ }
+ }
+ // user creation
+ var uuid = user[config.saml.attribute.id] || user.nameID
+ var profile = {
+ provider: 'saml',
+ id: 'SAML-' + uuid,
+ username: user[config.saml.attribute.username] || user.nameID,
+ emails: user[config.saml.attribute.email] ? [user[config.saml.attribute.email]] : []
+ }
+ if (profile.emails.length === 0 && config.saml.identifierFormat === 'urn:oasis:names:tc:SAML:1.1:nameid-format:emailAddress') {
+ profile.emails.push(user.nameID)
+ }
+ var stringifiedProfile = JSON.stringify(profile)
+ models.User.findOrCreate({
+ where: {
+ profileid: profile.id.toString()
+ },
+ defaults: {
+ profile: stringifiedProfile
+ }
+ }).spread(function (user, created) {
+ if (user) {
+ var needSave = false
+ if (user.profile !== stringifiedProfile) {
+ user.profile = stringifiedProfile
+ needSave = true
+ }
+ if (needSave) {
+ user.save().then(function () {
+ if (config.debug) { logger.debug('user login: ' + user.id) }
+ return done(null, user)
+ })
+ } else {
+ if (config.debug) { logger.debug('user login: ' + user.id) }
+ return done(null, user)
+ }
+ }
+ }).catch(function (err) {
+ logger.error('saml auth failed: ' + err)
+ return done(err, null)
+ })
+}))
+
+samlAuth.get('/auth/saml',
+ passport.authenticate('saml', {
+ successReturnToOrRedirect: config.serverurl + '/',
+ failureRedirect: config.serverurl + '/'
+ })
+)
+
+samlAuth.post('/auth/saml/callback', urlencodedParser,
+ passport.authenticate('saml', {
+ successReturnToOrRedirect: config.serverurl + '/',
+ failureRedirect: config.serverurl + '/'
+ })
+)
+
+samlAuth.get('/auth/saml/metadata', function (req, res) {
+ res.type('application/xml')
+ res.send(passport._strategy('saml').generateServiceProviderMetadata())
+})
diff --git a/locales/en.json b/locales/en.json
index 2ab286bb..6b2a2066 100644
--- a/locales/en.json
+++ b/locales/en.json
@@ -29,6 +29,8 @@
"Import from browser": "Import from browser",
"Releases": "Releases",
"Are you sure?": "Are you sure?",
+ "Do you really want to delete this note?": "Do you really want to delete this note?",
+ "All users will lose their connection.": "All users will lose their connection.",
"Cancel": "Cancel",
"Yes, do it!": "Yes, do it!",
"Choose method": "Choose method",
diff --git a/locales/nl.json b/locales/nl.json
index 03e3abfc..325c273d 100644
--- a/locales/nl.json
+++ b/locales/nl.json
@@ -29,6 +29,8 @@
"Import from browser": "Importeer van browser",
"Releases": "Releases",
"Are you sure?": "Weet je het zeker?",
+ "Do you really want to delete this note?": "Will je deze notitie echt verwijderen?",
+ "All users will lose their connection.": "Alle gebruikers zullen hun verbinding verliezen.",
"Cancel": "Stoppen",
"Yes, do it!": "Ja, doe het!",
"Choose method": "Kies methode",
diff --git a/package.json b/package.json
index a78bb2bd..43668883 100644
--- a/package.json
+++ b/package.json
@@ -3,7 +3,7 @@
"version": "0.5.1",
"description": "Realtime collaborative markdown notes on all platforms.",
"main": "app.js",
- "license": "MIT",
+ "license": "AGPL-3.0",
"scripts": {
"test": "npm run-script standard && npm run-script jsonlint",
"jsonlint": "find . -not -path './node_modules/*' -type f -name '*.json' | while read json; do echo $json ; jsonlint -q $json; done",
@@ -76,6 +76,7 @@
"markdown-pdf": "^7.0.0",
"mathjax": "~2.7.0",
"mermaid": "~7.1.0",
+ "mattermost": "^3.4.0",
"meta-marked": "^0.4.2",
"method-override": "^2.3.7",
"moment": "^2.17.1",
@@ -91,7 +92,9 @@
"passport-google-oauth20": "^1.0.0",
"passport-ldapauth": "^0.6.0",
"passport-local": "^1.0.0",
+ "passport-oauth2": "^1.4.0",
"passport-twitter": "^1.0.4",
+ "passport-saml": "^0.31.0",
"passport.socketio": "^3.7.0",
"pdfobject": "^2.0.201604172",
"pg": "^6.1.2",
diff --git a/public/js/extra.js b/public/js/extra.js
index d36592d9..13b8924c 100644
--- a/public/js/extra.js
+++ b/public/js/extra.js
@@ -373,22 +373,19 @@ export function finishView (view) {
var $value = $(value)
const $ele = $(value).closest('pre')
- let mermaidError = null
- window.mermaid.parseError = (err, hash) => {
- mermaidError = err
+ window.mermaid.mermaidAPI.parse($value.text())
+ $ele.addClass('mermaid')
+ $ele.html($value.text())
+ window.mermaid.init(undefined, $ele)
+ } catch (err) {
+ var errormessage = err
+ if (err.str) {
+ errormessage = err.str
}
- if (window.mermaidAPI.parse($value.text())) {
- $ele.addClass('mermaid')
- $ele.html($value.text())
- window.mermaid.init(undefined, $ele)
- } else {
- throw new Error(mermaidError)
- }
- } catch (err) {
$value.unwrap()
- $value.parent().append('<div class="alert alert-warning">' + err + '</div>')
- console.warn(err)
+ $value.parent().append('<div class="alert alert-warning">' + errormessage + '</div>')
+ console.warn(errormessage)
}
})
// abc.js
diff --git a/public/js/render.js b/public/js/render.js
index e2574b5f..46489247 100644
--- a/public/js/render.js
+++ b/public/js/render.js
@@ -18,7 +18,7 @@ whiteList['style'] = []
// allow kbd tag
whiteList['kbd'] = []
// allow ifram tag with some safe attributes
-whiteList['iframe'] = ['allowfullscreen', 'name', 'referrerpolicy', 'sandbox', 'src', 'srcdoc', 'width', 'height']
+whiteList['iframe'] = ['allowfullscreen', 'name', 'referrerpolicy', 'sandbox', 'src', 'width', 'height']
// allow summary tag
whiteList['summary'] = []
diff --git a/public/views/hackmd/body.ejs b/public/views/hackmd/body.ejs
index ad1e748b..91343ef6 100644
--- a/public/views/hackmd/body.ejs
+++ b/public/views/hackmd/body.ejs
@@ -234,8 +234,8 @@
<h4 class="modal-title" id="myModalLabel"><%= __('Are you sure?') %></h4>
</div>
<div class="modal-body" style="color:black;">
- <h5 class="ui-delete-modal-msg">Do you really want to delete this note?</h5>
- <strong class="ui-delete-modal-item">All users will lost their connection.</strong>
+ <h5 class="ui-delete-modal-msg"><%= __('Do you really want to delete this note?') %></h5>
+ <strong class="ui-delete-modal-item"><%= __('All users will lose their connection.') %></strong>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-default" data-dismiss="modal"><%= __('Cancel') %></button>
diff --git a/public/views/hackmd/header.ejs b/public/views/hackmd/header.ejs
index 47b563ac..80df2c77 100644
--- a/public/views/hackmd/header.ejs
+++ b/public/views/hackmd/header.ejs
@@ -70,7 +70,7 @@
</li>
<li role="presentation"><a role="menuitem" class="ui-download-raw-html" tabindex="-1" href="#" target="_self"><i class="fa fa-file-code-o fa-fw"></i> <%= __('Raw HTML') %></a>
</li>
- <% if(allowpdfexport) {%>
+ <% if(typeof allowpdfexport !== 'undefined' && allowpdfexport) {%>
<li role="presentation"><a role="menuitem" class="ui-download-pdf-beta" tabindex="-1" href="#" target="_self"><i class="fa fa-file-pdf-o fa-fw"></i> PDF (Beta)</a>
</li>
<% } %>
@@ -171,7 +171,7 @@
</li>
<li role="presentation"><a role="menuitem" class="ui-download-raw-html" tabindex="-1" href="#" target="_self"><i class="fa fa-file-code-o fa-fw"></i> <%= __('Raw HTML') %></a>
</li>
- <% if(allowpdfexport) {%>
+ <% if(typeof allowpdfexport !== 'undefined' && allowpdfexport) {%>
<li role="presentation"><a role="menuitem" class="ui-download-pdf-beta" tabindex="-1" href="#" target="_self"><i class="fa fa-file-pdf-o fa-fw"></i> PDF (Beta)</a>
</li>
<% } %>
diff --git a/public/views/index/body.ejs b/public/views/index/body.ejs
index b9c5c426..d7b4458e 100644
--- a/public/views/index/body.ejs
+++ b/public/views/index/body.ejs
@@ -15,7 +15,7 @@
<% if(allowAnonymous) { %>
<a type="button" href="<%- url %>/new" class="btn btn-sm btn-primary"><i class="fa fa-plus"></i> <%= __('New guest note') %></a>
<% } %>
- <% if(facebook || twitter || github || gitlab || dropbox || google || ldap || email) { %>
+ <% if(facebook || twitter || github || gitlab || mattermost || dropbox || google || ldap || saml || email) { %>
<button class="btn btn-sm btn-success ui-signin" data-toggle="modal" data-target=".signin-modal"><%= __('Sign In') %></button>
<% } %>
</div>
@@ -48,7 +48,7 @@
<% if (errorMessage && errorMessage.length > 0) { %>
<div class="alert alert-danger" style="max-width: 400px; margin: 0 auto;"><%= errorMessage %></div>
<% } %>
- <% if(facebook || twitter || github || gitlab || dropbox || google || ldap || email) { %>
+ <% if(facebook || twitter || github || gitlab || mattermost || dropbox || google || ldap || saml || email) { %>
<span class="ui-signin">
<br>
<a type="button" class="btn btn-lg btn-success ui-signin" data-toggle="modal" data-target=".signin-modal" style="min-width: 200px;"><%= __('Sign In') %></a>
diff --git a/public/views/shared/signin-modal.ejs b/public/views/shared/signin-modal.ejs
index a8af62e7..7b44cfb0 100644
--- a/public/views/shared/signin-modal.ejs
+++ b/public/views/shared/signin-modal.ejs
@@ -28,6 +28,11 @@
<i class="fa fa-gitlab"></i> <%= __('Sign in via %s', 'GitLab') %>
</a>
<% } %>
+ <% if(mattermost) { %>
+ <a href="<%- url %>/auth/mattermost" class="btn btn-lg btn-block btn-social btn-soundcloud">
+ <i class="fa fa-mattermost"></i> <%= __('Sign in via %s', 'Mattermost') %>
+ </a>
+ <% } %>
<% if(dropbox) { %>
<a href="<%- url %>/auth/dropbox" class="btn btn-lg btn-block btn-social btn-dropbox">
<i class="fa fa-dropbox"></i> <%= __('Sign in via %s', 'Dropbox') %>
@@ -38,7 +43,12 @@
<i class="fa fa-google"></i> <%= __('Sign in via %s', 'Google') %>
</a>
<% } %>
- <% if((facebook || twitter || github || gitlab || dropbox || google) && ldap) { %>
+ <% if(saml) { %>
+ <a href="<%- url %>/auth/saml" class="btn btn-lg btn-block btn-social btn-success">
+ <i class="fa fa-users"></i> <%= __('Sign in via %s', 'SAML') %>
+ </a>
+ <% } %>
+ <% if((facebook || twitter || github || gitlab || mattermost || dropbox || google || saml) && ldap) { %>
<hr>
<% }%>
<% if(ldap) { %>
@@ -63,7 +73,7 @@
</div>
</form>
<% } %>
- <% if((facebook || twitter || github || gitlab || dropbox || google || ldap) && email) { %>
+ <% if((facebook || twitter || github || gitlab || mattermost || dropbox || google || ldap) && email) { %>
<hr>
<% }%>
<% if(email) { %>
diff --git a/yarn.lock b/yarn.lock
index 6b0e0ae2..d80251f5 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -1342,7 +1342,7 @@ component-emitter@1.1.2:
version "1.1.2"
resolved "https://registry.yarnpkg.com/component-emitter/-/component-emitter-1.1.2.tgz#296594f2753daa63996d2af08d15a95116c9aec3"
-component-emitter@1.2.1:
+component-emitter@1.2.1, component-emitter@~1.2.0:
version "1.2.1"
resolved "https://registry.yarnpkg.com/component-emitter/-/component-emitter-1.2.1.tgz#137918d6d78283f7df7a6b7c5a63e140e69425e6"
@@ -1458,6 +1458,10 @@ cookie@0.3.1:
version "0.3.1"
resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.3.1.tgz#e7e0a1f9ef43b4c8ba925c5c5a96e806d16873bb"
+cookiejar@2.0.6:
+ version "2.0.6"
+ resolved "https://registry.yarnpkg.com/cookiejar/-/cookiejar-2.0.6.tgz#0abf356ad00d1c5a219d88d44518046dd026acfe"
+
copy-webpack-plugin@^4.0.1:
version "4.0.1"
resolved "https://registry.yarnpkg.com/copy-webpack-plugin/-/copy-webpack-plugin-4.0.1.tgz#9728e383b94316050d0c7463958f2b85c0aa8200"
@@ -1686,9 +1690,9 @@ debug-log@^1.0.0:
version "1.0.1"
resolved "https://registry.yarnpkg.com/debug-log/-/debug-log-1.0.1.tgz#2307632d4c04382b8df8a32f70b895046d52745f"
-debug@*:
- version "3.1.0"
- resolved "https://registry.yarnpkg.com/debug/-/debug-3.1.0.tgz#5bb5a0672628b64149566ba16819e61518c67261"
+debug@*, debug@2, debug@2.6.9, debug@^2.1.1, debug@^2.1.3, debug@^2.2.0, debug@^2.6.8:
+ version "2.6.9"
+ resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
dependencies:
ms "2.0.0"
@@ -1710,12 +1714,6 @@ debug@2.6.7:
dependencies:
ms "2.0.0"
-debug@2.6.9, debug@^2.1.1, debug@^2.1.3, debug@^2.2.0, debug@^2.6.8:
- version "2.6.9"
- resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
- dependencies:
- ms "2.0.0"
-
decamelize@^1.0.0, decamelize@^1.1.1, decamelize@^1.1.2:
version "1.2.0"
resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290"
@@ -1900,20 +1898,13 @@ domutils@1.1:
dependencies:
domelementtype "1"
-domutils@1.5, domutils@1.5.1:
+domutils@1.5, domutils@1.5.1, domutils@^1.5.1:
version "1.5.1"
resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.5.1.tgz#dcd8488a26f563d61079e48c9f7b7e32373682cf"
dependencies:
dom-serializer "0"
domelementtype "1"
-domutils@^1.5.1:
- version "1.6.2"
- resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.6.2.tgz#1958cc0b4c9426e9ed367fb1c8e854891b0fa3ff"
- dependencies:
- dom-serializer "0"
- domelementtype "1"
-
dont-sniff-mimetype@1.0.0:
version "1.0.0"
resolved "https://registry.yarnpkg.com/dont-sniff-mimetype/-/dont-sniff-mimetype-1.0.0.tgz#5932890dc9f4e2f19e5eb02a20026e5e5efc8f58"
@@ -2421,6 +2412,10 @@ express@>=4.14:
utils-merge "1.0.0"
vary "~1.1.1"
+extend@3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.0.tgz#5a474353b9f3353ddd8176dfd37b91c83a46f1d4"
+
extend@^3.0.0, extend@~3.0.0:
version "3.0.1"
resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.1.tgz#a755ea7bc1adfcc5a31ce7e762dbaadc5e636444"
@@ -2667,6 +2662,14 @@ forever-agent@~0.6.1:
version "0.6.1"
resolved "https://registry.yarnpkg.com/forever-agent/-/forever-agent-0.6.1.tgz#fbc71f0c41adeb37f96c577ad1ed42d8fdacca91"
+form-data@1.0.0-rc3:
+ version "1.0.0-rc3"
+ resolved "https://registry.yarnpkg.com/form-data/-/form-data-1.0.0-rc3.tgz#d35bc62e7fbc2937ae78f948aaa0d38d90607577"
+ dependencies:
+ async "^1.4.0"
+ combined-stream "^1.0.5"
+ mime-types "^2.1.3"
+
form-data@~2.1.1:
version "2.1.4"
resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.1.4.tgz#33c183acf193276ecaa98143a69e94bfee1750d1"
@@ -2679,6 +2682,10 @@ formidable@^1.0.17:
version "1.1.1"
resolved "https://registry.yarnpkg.com/formidable/-/formidable-1.1.1.tgz#96b8886f7c3c3508b932d6bd70c4d3a88f35f1a9"
+formidable@~1.0.14:
+ version "1.0.17"
+ resolved "https://registry.yarnpkg.com/formidable/-/formidable-1.0.17.tgz#ef5491490f9433b705faa77249c99029ae348559"
+
forwarded@~0.1.0:
version "0.1.2"
resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.1.2.tgz#98c23dab1175657b8c0573e8ceccd91b0ff18c84"
@@ -4418,6 +4425,12 @@ mathjax@~2.7.0:
version "2.7.2"
resolved "https://registry.yarnpkg.com/mathjax/-/mathjax-2.7.2.tgz#97d78bbebfb65a8621ce33fb7c1f10917355a878"
+mattermost@^3.4.0:
+ version "3.4.0"
+ resolved "https://registry.yarnpkg.com/mattermost/-/mattermost-3.4.0.tgz#7e4958e1bc96c7da7bc5f179dd2c6ae5035a8857"
+ dependencies:
+ superagent "1.8.3"
+
maxmin@^1.1.0:
version "1.1.0"
resolved "https://registry.yarnpkg.com/maxmin/-/maxmin-1.1.0.tgz#71365e84a99dd8f8b3f7d5fde2f00d1e7f73be61"
@@ -4521,7 +4534,7 @@ method-override@^2.3.7:
parseurl "~1.3.2"
vary "~1.1.2"
-methods@~1.1.2:
+methods@~1.1.1, methods@~1.1.2:
version "1.1.2"
resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee"
@@ -4547,7 +4560,7 @@ micromatch@^2.1.5, micromatch@^2.3.7:
version "1.30.0"
resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.30.0.tgz#74c643da2dd9d6a45399963465b26d5ca7d71f01"
-mime-types@^2.1.12, mime-types@~2.1.11, mime-types@~2.1.15, mime-types@~2.1.16, mime-types@~2.1.7:
+mime-types@^2.1.12, mime-types@^2.1.3, mime-types@~2.1.11, mime-types@~2.1.15, mime-types@~2.1.16, mime-types@~2.1.7:
version "2.1.17"
resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.17.tgz#09d7a393f03e995a79f8af857b70a9e0ab16557a"
dependencies:
@@ -4849,11 +4862,7 @@ number-is-nan@^1.0.0:
version "1.0.1"
resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d"
-"nwmatcher@>= 1.3.9 < 2.0.0":
- version "1.4.2"
- resolved "https://registry.yarnpkg.com/nwmatcher/-/nwmatcher-1.4.2.tgz#c5e545ab40d22a56b0326531c4beaed7a888b3ea"
-
-nwmatcher@~1.3.1:
+"nwmatcher@>= 1.3.9 < 2.0.0", nwmatcher@~1.3.1:
version "1.3.9"
resolved "https://registry.yarnpkg.com/nwmatcher/-/nwmatcher-1.3.9.tgz#8bab486ff7fa3dfd086656bbe8b17116d3692d2a"
@@ -4865,7 +4874,7 @@ oauth@0.9.x:
version "0.9.15"
resolved "https://registry.yarnpkg.com/oauth/-/oauth-0.9.15.tgz#bd1fefaf686c96b75475aed5196412ff60cfb9c1"
-object-assign@4.1.0:
+object-assign@4.1.0, object-assign@^4.0.1, object-assign@^4.1.0:
version "4.1.0"
resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.0.tgz#7a3b3d0e98063d43f4c03f2e8ae6cd51a86883a0"
@@ -4873,10 +4882,6 @@ object-assign@^3.0.0:
version "3.0.0"
resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-3.0.0.tgz#9bedd5ca0897949bca47e7ff408062d549f587f2"
-object-assign@^4.0.1, object-assign@^4.1.0:
- version "4.1.1"
- resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863"
-
object-component@0.0.3:
version "0.0.3"
resolved "https://registry.yarnpkg.com/object-component/-/object-component-0.0.3.tgz#f0c69aa50efc95b866c186f400a33769cb2f1291"
@@ -5162,7 +5167,7 @@ passport-oauth1@1.x.x:
passport-strategy "1.x.x"
utils-merge "1.x.x"
-passport-oauth2@1.x.x, passport-oauth2@^1.3.0:
+passport-oauth2@1.x.x, passport-oauth2@^1.3.0, passport-oauth2@^1.4.0:
version "1.4.0"
resolved "https://registry.yarnpkg.com/passport-oauth2/-/passport-oauth2-1.4.0.tgz#f62f81583cbe12609be7ce6f160b9395a27b86ad"
dependencies:
@@ -5743,6 +5748,10 @@ q@^1.0.1, q@^1.1.2:
version "1.5.0"
resolved "https://registry.yarnpkg.com/q/-/q-1.5.0.tgz#dd01bac9d06d30e6f219aecb8253ee9ebdc308f1"
+qs@2.3.3:
+ version "2.3.3"
+ resolved "https://registry.yarnpkg.com/qs/-/qs-2.3.3.tgz#e9e85adbe75da0bbe4c8e0476a086290f863b404"
+
qs@6.5.0:
version "6.5.0"
resolved "https://registry.yarnpkg.com/qs/-/qs-6.5.0.tgz#8d04954d364def3efc55b5a0793e1e2c8b1e6e49"
@@ -5789,13 +5798,7 @@ range-parser@~1.2.0:
version "1.2.0"
resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.0.tgz#f49be6b487894ddc40dcc94a322f611092e00d5e"
-raphael@2.2.7:
- version "2.2.7"
- resolved "https://registry.yarnpkg.com/raphael/-/raphael-2.2.7.tgz#231b19141f8d086986d8faceb66f8b562ee2c810"
- dependencies:
- eve-raphael "0.5.0"
-
-"raphael@git+https://github.com/dmitrybaranovskiy/raphael":
+raphael@2.2.7, "raphael@git+https://github.com/dmitrybaranovskiy/raphael":
version "2.2.7"
resolved "git+https://github.com/dmitrybaranovskiy/raphael#fe8e591e1c86b5aeb4c252b33c08e647434504c5"
dependencies:
@@ -5868,7 +5871,16 @@ readable-stream@1.0, "readable-stream@>=1.0.33-1 <1.1.0-0":
isarray "0.0.1"
string_decoder "~0.10.x"
-readable-stream@1.1:
+readable-stream@1.0.27-1:
+ version "1.0.27-1"
+ resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-1.0.27-1.tgz#6b67983c20357cefd07f0165001a16d710d91078"
+ dependencies:
+ core-util-is "~1.0.0"
+ inherits "~2.0.1"
+ isarray "0.0.1"
+ string_decoder "~0.10.x"
+
+readable-stream@1.1, readable-stream@~1.1.9:
version "1.1.13"
resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-1.1.13.tgz#f6eef764f514c89e2b9e23146a75ba106756d23e"
dependencies:
@@ -5889,15 +5901,6 @@ readable-stream@2.3.3, readable-stream@^2.0.1, readable-stream@^2.0.2, readable-
string_decoder "~1.0.3"
util-deprecate "~1.0.1"
-readable-stream@~1.1.9:
- version "1.1.14"
- resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-1.1.14.tgz#7cf4c54ef648e3813084c636dd2079e166c081d9"
- dependencies:
- core-util-is "~1.0.0"
- inherits "~2.0.1"
- isarray "0.0.1"
- string_decoder "~0.10.x"
-
readdirp@^2.0.0:
version "2.1.0"
resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-2.1.0.tgz#4ed0ad060df3073300c48440373f72d1cc642d78"
@@ -5932,6 +5935,10 @@ redent@^1.0.0:
indent-string "^2.1.0"
strip-indent "^1.0.1"
+reduce-component@1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/reduce-component/-/reduce-component-1.0.1.tgz#e0c93542c574521bea13df0f9488ed82ab77c5da"
+
reduce-css-calc@^1.2.6:
version "1.3.0"
resolved "https://registry.yarnpkg.com/reduce-css-calc/-/reduce-css-calc-1.3.0.tgz#747c914e049614a4c9cfbba629871ad1d2927716"
@@ -6243,14 +6250,10 @@ safefs@^3.1.2:
dependencies:
graceful-fs "*"
-sax@1.2.1:
+sax@1.2.1, sax@>=0.6.0, sax@^1.2.1, sax@~1.2.1:
version "1.2.1"
resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.1.tgz#7b8e656190b228e81a66aea748480d828cd2d37a"
-sax@>=0.6.0, sax@^1.2.1, sax@~1.2.1:
- version "1.2.4"
- resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9"
-
scandirectory@^2.5.0:
version "2.5.0"
resolved "https://registry.yarnpkg.com/scandirectory/-/scandirectory-2.5.0.tgz#6ce03f54a090b668e3cbedbf20edf9e310593e72"
@@ -6283,14 +6286,10 @@ select@^1.1.2:
version "5.4.1"
resolved "https://registry.yarnpkg.com/semver/-/semver-5.4.1.tgz#e059c09d8571f0540823733433505d3a2f00b18e"
-semver@4.3.2:
+semver@4.3.2, semver@^4.1.0:
version "4.3.2"
resolved "https://registry.yarnpkg.com/semver/-/semver-4.3.2.tgz#c7a07158a80bedd052355b770d82d6640f803be7"
-semver@^4.1.0:
- version "4.3.6"
- resolved "https://registry.yarnpkg.com/semver/-/semver-4.3.6.tgz#300bc6e0e86374f7ba61068b5b1ecd57fc6532da"
-
send@0.15.6:
version "0.15.6"
resolved "https://registry.yarnpkg.com/send/-/send-0.15.6.tgz#20f23a9c925b762ab82705fe2f9db252ace47e34"
@@ -6752,6 +6751,22 @@ style-loader@^0.13.1:
dependencies:
loader-utils "^1.0.2"
+superagent@1.8.3:
+ version "1.8.3"
+ resolved "https://registry.yarnpkg.com/superagent/-/superagent-1.8.3.tgz#2b7d70fcc870eda4f2a61e619dd54009b86547c3"
+ dependencies:
+ component-emitter "~1.2.0"
+ cookiejar "2.0.6"
+ debug "2"
+ extend "3.0.0"
+ form-data "1.0.0-rc3"
+ formidable "~1.0.14"
+ methods "~1.1.1"
+ mime "1.3.4"
+ qs "2.3.3"
+ readable-stream "1.0.27-1"
+ reduce-component "1.0.1"
+
supports-color@^2.0.0:
version "2.0.0"
resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-2.0.0.tgz#535d045ce6b6363fa40117084629995e9df324c7"
@@ -7251,14 +7266,10 @@ utils-merge@1.0.1, utils-merge@1.x.x:
version "1.0.1"
resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713"
-uuid@3.0.1:
+uuid@3.0.1, uuid@^3.0.0:
version "3.0.1"
resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.0.1.tgz#6544bba2dfda8c1cf17e629a3a305e2bb1fee6c1"
-uuid@^3.0.0:
- version "3.1.0"
- resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.1.0.tgz#3dd3d3e790abc24d7b0d3a034ffababe28ebbc04"
-
uws@~0.14.1:
version "0.14.5"
resolved "https://registry.yarnpkg.com/uws/-/uws-0.14.5.tgz#67aaf33c46b2a587a5f6666d00f7691328f149dc"
@@ -7468,13 +7479,7 @@ which-module@^2.0.0:
version "2.0.0"
resolved "https://registry.yarnpkg.com/which-module/-/which-module-2.0.0.tgz#d9ef07dce77b9902b8a3a8fa4b31c3e3f7e6e87a"
-which@^1.2.12, which@^1.2.9:
- version "1.3.0"
- resolved "https://registry.yarnpkg.com/which/-/which-1.3.0.tgz#ff04bdfc010ee547d780bec38e1ac1c2777d253a"
- dependencies:
- isexe "^2.0.0"
-
-which@~1.2.10:
+which@^1.2.12, which@^1.2.9, which@~1.2.10:
version "1.2.14"
resolved "https://registry.yarnpkg.com/which/-/which-1.2.14.tgz#9a87c4378f03e827cecaf1acdf56c736c01c14e5"
dependencies: