summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--AUTHORS104
-rw-r--r--CONTRIBUTING.md29
-rw-r--r--CONTRIBUTORS902
-rw-r--r--LICENSE673
-rw-r--r--README.md22
-rw-r--r--config.json.example17
-rw-r--r--contribute/developer-certificate-of-origin35
-rw-r--r--docs/guides/auth.md135
-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--lib/config/default.js15
-rw-r--r--lib/config/environment.js21
-rw-r--r--lib/config/index.js1
-rw-r--r--lib/config/utils.js7
-rw-r--r--lib/migrations/20171009121200-longtext-for-mysql.js16
-rw-r--r--lib/models/note.js2
-rw-r--r--lib/models/revision.js6
-rw-r--r--lib/models/user.js9
-rw-r--r--lib/realtime.js6
-rw-r--r--[-rwxr-xr-x]lib/response.js2
-rw-r--r--lib/web/auth/index.js1
-rw-r--r--lib/web/auth/ldap/index.js8
-rw-r--r--lib/web/auth/saml/index.js95
-rw-r--r--locales/en.json3
-rw-r--r--package.json3
-rw-r--r--public/css/extra.css31
-rw-r--r--public/css/github-extract.css41
-rw-r--r--public/css/index.css74
-rw-r--r--public/css/markdown.css38
-rw-r--r--[-rwxr-xr-x]public/docs/release-notes.md0
-rw-r--r--public/js/extra.js6
-rw-r--r--public/js/index.js15
-rw-r--r--public/js/lib/appState.js3
-rw-r--r--public/js/lib/editor/index.js6
-rw-r--r--public/js/lib/editor/ui-elements.js1
-rw-r--r--[-rwxr-xr-x]public/js/reveal-markdown.js0
-rw-r--r--public/views/hackmd/header.ejs5
-rw-r--r--public/views/index/body.ejs4
-rw-r--r--public/views/shared/signin-modal.ejs7
-rw-r--r--yarn.lock72
44 files changed, 2319 insertions, 96 deletions
diff --git a/AUTHORS b/AUTHORS
index b155d3fa..ef33e970 100644
--- 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 100644
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 a80deef0..cb8924ec 100644
--- a/README.md
+++ b/README.md
@@ -6,7 +6,7 @@ HackMD
[![Join the chat at https://gitter.im/hackmdio/hackmd][gitter-image]][gitter-url]
[![build status][travis-image]][travis-url]
[![version][github-version-badge]][github-release-page]
-
+[![Help Contribute to Open Source][codetriage-image]][codetriage-url]
HackMD lets you create realtime collaborative markdown notes on all platforms.
Inspired by Hackpad, with more focus on speed and flexibility.
@@ -169,9 +169,20 @@ 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_USERNAMEFIELD | `uid` | The LDAP field which is used as the username on HackMD |
| 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 |
@@ -234,7 +245,7 @@ There are some configs you need to change in the files below
| service | settings location | description |
| ------- | --------- | ----------- |
-| facebook, twitter, github, gitlab, mattermost, dropbox, google, ldap | environment variables or `config.json` | for signin |
+| 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 |
@@ -249,6 +260,7 @@ There are some configs you need to change in the files below
| mattermost | `/auth/mattermost/callback` |
| dropbox | `/auth/dropbox/callback` |
| google | `/auth/google/callback` |
+| saml | `/auth/saml/callback` |
# Developer Notes
@@ -277,7 +289,7 @@ See more at [http://operational-transformation.github.io/](http://operational-tr
# 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
@@ -287,3 +299,5 @@ See more at [http://operational-transformation.github.io/](http://operational-tr
[github-release-page]: https://github.com/hackmdio/hackmd/releases
[standardjs-image]: https://cdn.rawgit.com/feross/standard/master/badge.svg
[standardjs-url]: https://github.com/feross/standard
+[codetriage-image]: https://www.codetriage.com/hackmdio/hackmd/badges/users.svg
+[codetriage-url]: https://www.codetriage.com/hackmdio/hackmd
diff --git a/config.json.example b/config.json.example
index bd7ab043..c2c270c3 100644
--- a/config.json.example
+++ b/config.json.example
@@ -70,11 +70,26 @@
"tokenSecret": "change this",
"searchBase": "change this",
"searchFilter": "change this",
- "searchAttributes": "change this",
+ "searchAttributes": ["change this"],
+ "usernameField": "change this e.g. uid",
"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"
},
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
index 37b89004..4f9ce445 100644
--- a/docs/guides/auth.md
+++ b/docs/guides/auth.md
@@ -75,3 +75,138 @@ To do this Click your profile icon --> Settings and privacy --> Mobile --> Sele
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/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/lib/config/default.js b/lib/config/default.js
index 273bad02..8d36db02 100644
--- a/lib/config/default.js
+++ b/lib/config/default.js
@@ -96,8 +96,23 @@ module.exports = {
searchBase: undefined,
searchFilter: undefined,
searchAttributes: undefined,
+ usernameField: 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/environment.js b/lib/config/environment.js
index 0c272f05..27e63591 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),
@@ -70,9 +70,24 @@ 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),
+ usernameField: process.env.HMD_LDAP_USERNAMEFIELD,
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 cf6f2ada..a14f3197 100644
--- a/lib/config/index.js
+++ b/lib/config/index.js
@@ -92,6 +92,7 @@ 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/migrations/20171009121200-longtext-for-mysql.js b/lib/migrations/20171009121200-longtext-for-mysql.js
new file mode 100644
index 00000000..61b409ca
--- /dev/null
+++ b/lib/migrations/20171009121200-longtext-for-mysql.js
@@ -0,0 +1,16 @@
+'use strict'
+module.exports = {
+ up: function (queryInterface, Sequelize) {
+ queryInterface.changeColumn('Notes', 'content', {type: Sequelize.TEXT('long')})
+ queryInterface.changeColumn('Revisions', 'patch', {type: Sequelize.TEXT('long')})
+ queryInterface.changeColumn('Revisions', 'content', {type: Sequelize.TEXT('long')})
+ queryInterface.changeColumn('Revisions', 'latContent', {type: Sequelize.TEXT('long')})
+ },
+
+ down: function (queryInterface, Sequelize) {
+ queryInterface.changeColumn('Notes', 'content', {type: Sequelize.TEXT})
+ queryInterface.changeColumn('Revisions', 'patch', {type: Sequelize.TEXT})
+ queryInterface.changeColumn('Revisions', 'content', {type: Sequelize.TEXT})
+ queryInterface.changeColumn('Revisions', 'latContent', {type: Sequelize.TEXT})
+ }
+}
diff --git a/lib/models/note.js b/lib/models/note.js
index c0ef1374..33dde80d 100644
--- a/lib/models/note.js
+++ b/lib/models/note.js
@@ -60,7 +60,7 @@ module.exports = function (sequelize, DataTypes) {
}
},
content: {
- type: DataTypes.TEXT,
+ type: DataTypes.TEXT('long'),
get: function () {
return sequelize.processData(this.getDataValue('content'), '')
},
diff --git a/lib/models/revision.js b/lib/models/revision.js
index 225a95d4..32c57d03 100644
--- a/lib/models/revision.js
+++ b/lib/models/revision.js
@@ -58,7 +58,7 @@ module.exports = function (sequelize, DataTypes) {
defaultValue: Sequelize.UUIDV4
},
patch: {
- type: DataTypes.TEXT,
+ type: DataTypes.TEXT('long'),
get: function () {
return sequelize.processData(this.getDataValue('patch'), '')
},
@@ -67,7 +67,7 @@ module.exports = function (sequelize, DataTypes) {
}
},
lastContent: {
- type: DataTypes.TEXT,
+ type: DataTypes.TEXT('long'),
get: function () {
return sequelize.processData(this.getDataValue('lastContent'), '')
},
@@ -76,7 +76,7 @@ module.exports = function (sequelize, DataTypes) {
}
},
content: {
- type: DataTypes.TEXT,
+ type: DataTypes.TEXT('long'),
get: function () {
return sequelize.processData(this.getDataValue('content'), '')
},
diff --git a/lib/models/user.js b/lib/models/user.js
index 27566def..f421fe43 100644
--- a/lib/models/user.js
+++ b/lib/models/user.js
@@ -143,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/realtime.js b/lib/realtime.js
index 361bbf09..e03e2d0b 100644
--- a/lib/realtime.js
+++ b/lib/realtime.js
@@ -709,7 +709,7 @@ function connection (socket) {
return failConnection(404, 'note id not found', socket)
}
- if (isDuplicatedInSocketQueue(socket, connectionSocketQueue)) return
+ if (isDuplicatedInSocketQueue(connectionSocketQueue, socket)) return
// store noteId in this socket session
socket.noteId = noteId
@@ -723,8 +723,8 @@ function connection (socket) {
var maxrandomcount = 10
var found = false
do {
- Object.keys(notes[noteId].users).forEach(function (user) {
- if (user.color === color) {
+ Object.keys(notes[noteId].users).forEach(function (userId) {
+ if (notes[noteId].users[userId].color === color) {
found = true
}
})
diff --git a/lib/response.js b/lib/response.js
index 61ce5747..9f3d5a44 100755..100644
--- a/lib/response.js
+++ b/lib/response.js
@@ -68,6 +68,7 @@ function showIndex (req, res, next) {
dropbox: config.isDropboxEnable,
google: config.isGoogleEnable,
ldap: config.isLDAPEnable,
+ saml: config.isSAMLEnable,
email: config.isEmailEnable,
allowemailregister: config.allowemailregister,
allowpdfexport: config.allowpdfexport,
@@ -100,6 +101,7 @@ function responseHackMD (res, note) {
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 4b618101..db5ff11d 100644
--- a/lib/web/auth/index.js
+++ b/lib/web/auth/index.js
@@ -37,6 +37,7 @@ 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/ldap/index.js b/lib/web/auth/ldap/index.js
index 9a63578a..cc0d29ad 100644
--- a/lib/web/auth/ldap/index.js
+++ b/lib/web/auth/ldap/index.js
@@ -24,9 +24,15 @@ passport.use(new LDAPStrategy({
}
}, function (user, done) {
var uuid = user.uidNumber || user.uid || user.sAMAccountName
+ var username = uuid
+
+ if (config.ldap.usernameField && user[config.ldap.usernameField]) {
+ username = user[config.ldap.usernameField]
+ }
+
var profile = {
id: 'LDAP-' + uuid,
- username: uuid,
+ username: username,
displayName: user.displayName,
emails: user.mail ? [user.mail] : [],
avatarUrl: null,
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 6b2a2066..49e93a83 100644
--- a/locales/en.json
+++ b/locales/en.json
@@ -102,5 +102,6 @@
"Select From Available Snippets": "Select From Available Snippets",
"OR": "OR",
"Export to Snippet": "Export to Snippet",
- "Select Visibility Level": "Select Visibility Level"
+ "Select Visibility Level": "Select Visibility Level",
+ "Night Theme": "Night Theme"
}
diff --git a/package.json b/package.json
index b4942c5a..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",
@@ -94,6 +94,7 @@
"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/css/extra.css b/public/css/extra.css
index 169a1a5a..1b132901 100644
--- a/public/css/extra.css
+++ b/public/css/extra.css
@@ -179,6 +179,11 @@
border-left: 1px solid black;
}
+.night .ui-toc-dropdown .nav>li>a:focus, .night .ui-toc-dropdown .nav>li>a:hover{
+ color: white;
+ border-left-color: white;
+}
+
.ui-toc-dropdown[dir='rtl'] .nav>li>a:focus,.ui-toc-dropdown[dir='rtl'] .nav>li>a:hover {
padding-right: 19px;
border-left: none;
@@ -192,6 +197,10 @@
background-color: transparent;
border-left: 2px solid black;
}
+.night .ui-toc-dropdown .nav>.active:focus>a,.night .ui-toc-dropdown .nav>.active:hover>a,.night .ui-toc-dropdown .nav>.active>a {
+ color: white;
+ border-left: 2px solid white;
+}
.ui-toc-dropdown[dir='rtl'] .nav>.active:focus>a,.ui-toc-dropdown[dir='rtl'] .nav>.active:hover>a,.ui-toc-dropdown[dir='rtl'] .nav>.active>a {
padding-right: 18px;
@@ -216,6 +225,10 @@
font-weight: 400;
}
+.night .ui-toc-dropdown .nav > li > a{
+ color: #aaa;
+}
+
.ui-toc-dropdown[dir='rtl'] .nav .nav>li>a {
padding-right: 30px;
}
@@ -350,13 +363,23 @@ small .dropdown a:focus, small .dropdown a:hover {
}
.unselectable {
- -moz-user-select: none;
- -khtml-user-select: none;
- -webkit-user-select: none;
+ -moz-user-select: none;
+ -khtml-user-select: none;
+ -webkit-user-select: none;
-o-user-select: none;
user-select: none;
}
+.night .navbar{
+ background: #333;
+ border-bottom-color: #333;
+ color: #eee;
+}
+
+.night .navbar a{
+ color: #eee;
+}
+
@media print {
div, table, img, pre, blockquote {
page-break-inside: avoid !important;
@@ -364,4 +387,4 @@ small .dropdown a:focus, small .dropdown a:hover {
a[href]:after {
font-size: 12px !important;
}
-} \ No newline at end of file
+}
diff --git a/public/css/github-extract.css b/public/css/github-extract.css
index 4d2650d4..7f7058a0 100644
--- a/public/css/github-extract.css
+++ b/public/css/github-extract.css
@@ -68,6 +68,9 @@
color: #777;
border-left: 0.25em solid #ddd;
}
+.night .markdown-body blockquote{
+ color: #bcbcbc;
+}
.markdown-body blockquote>:first-child {
margin-top: 0;
@@ -107,6 +110,15 @@
line-height: 1.25;
}
+.night .markdown-body h1,
+.night .markdown-body h2,
+.night .markdown-body h3,
+.night .markdown-body h4,
+.night .markdown-body h5,
+.night .markdown-body h6 {
+ color: #ddd;
+}
+
.markdown-body h1 .octicon-link,
.markdown-body h2 .octicon-link,
.markdown-body h3 .octicon-link,
@@ -118,6 +130,15 @@
visibility: hidden;
}
+.night .markdown-body h1 .octicon-link,
+.night .markdown-body h2 .octicon-link,
+.night .markdown-body h3 .octicon-link,
+.night .markdown-body h4 .octicon-link,
+.night .markdown-body h5 .octicon-link,
+.night .markdown-body h6 .octicon-link {
+ color: #fff;
+}
+
.markdown-body h1:hover .anchor,
.markdown-body h2:hover .anchor,
.markdown-body h3:hover .anchor,
@@ -180,6 +201,8 @@
color: #777
}
+
+
.markdown-body ul,
.markdown-body ol {
padding-left: 2em
@@ -246,11 +269,19 @@
background-color: #fff;
border-top: 1px solid #ccc;
}
+.night .markdown-body table tr {
+ background-color: #5f5f5f;
+}
.markdown-body table tr:nth-child(2n) {
background-color: #f8f8f8;
}
+.night .markdown-body table tr:nth-child(2n){
+
+ background-color: #4f4f4f;
+}
+
.markdown-body img {
max-width: 100%;
box-sizing: content-box;
@@ -370,6 +401,14 @@
border-radius: 3px;
}
+.night .markdown-body code,
+.night .markdown-body tt {
+
+ color: #eee;
+ background-color: rgba(230, 230, 230, 0.36);
+
+}
+
.markdown-body code::before,
.markdown-body code::after,
.markdown-body tt::before,
@@ -512,4 +551,4 @@
margin: 0.31em 0 0.2em -1.3em !important;
vertical-align: middle;
cursor: default !important;
-} \ No newline at end of file
+}
diff --git a/public/css/index.css b/public/css/index.css
index 8f483aa7..b00eba41 100644
--- a/public/css/index.css
+++ b/public/css/index.css
@@ -10,6 +10,16 @@ body {
padding-top: 51px;
/*overflow: hidden;*/
}
+
+.night a,
+.night .open-files-container li.selected a {
+ color: #5EB7E0;
+}
+
+body.night{
+ background: #333 !important;
+}
+
.CodeMirror {
font-family: "Source Code Pro", Consolas, monaco, monospace;
letter-spacing: 0.025em;
@@ -117,6 +127,11 @@ body {
margin-left: 0;
margin-right: 0;
}
+
+.night .ui-content{
+ background-color: #333;
+}
+
.ui-edit-area {
height: 100%;
/*padding-left: 15px;*/
@@ -144,6 +159,12 @@ body {
.ui-edit-area .ui-sync-toggle:active {
box-shadow: inset 0 3px 5px rgba(0,0,0,.125), 2px 0px 2px #e7e7e7;
}
+
+.night .ui-edit-area .ui-resizable-handle.ui-resizable-e{
+ background: #3c3c3c;
+ box-shadow: 3px 0px 6px #353535;
+}
+
.ui-view-area {
/*overflow-y: scroll;*/
-webkit-overflow-scrolling: touch;
@@ -154,6 +175,13 @@ body {
padding-right: 15px;
}
}
+
+.night .ui-view-area{
+ background: #333;
+ color: #ededed;
+}
+
+
.ui-scrollable {
height: 100%;
overflow-x: hidden;
@@ -238,12 +266,32 @@ body {
.navbar-nav > li > a {
cursor: pointer;
}
+
+.night .navbar-default .navbar-nav > li > a:focus,
+.night .navbar-default .navbar-nav > li > a:hover,
+.night .navbar-default .navbar-brand:focus,
+.night .navbar-default .navbar-brand:hover{
+ color: #fff;
+}
+
+.night .navbar-default .navbar-nav > .open > a,
+.night .navbar-default .navbar-nav > .open > a:focus,
+.night .navbar-default .navbar-nav > .open > a:hover {
+ color: white;
+ background: #000;
+
+}
.dropdown-menu > li > a {
cursor: pointer;
text-overflow: ellipsis;
max-width: calc(100vw - 30px);
overflow: hidden;
}
+
+.night .dropdown-menu{
+ background: #222;
+}
+
.dropdown-menu.CodeMirror-other-cursor {
transition: none;
}
@@ -276,8 +324,8 @@ div[contenteditable]:empty:not(:focus):before{
max-height: 40vh;
overflow: auto;
}
-.dropdown-menu.list::-webkit-scrollbar {
- display: none;
+.dropdown-menu.list::-webkit-scrollbar {
+ display: none;
}
.dropdown-menu .emoji {
margin-bottom: 0 !important;
@@ -292,6 +340,16 @@ div[contenteditable]:empty:not(:focus):before{
background: inherit;
}
+.night .navbar .btn-default{
+ background-color: #333;
+ border-color: #565656;
+ color: #eee;
+}
+
+.night .btn.btn-default.ui-view.active{
+ background: #202020;
+}
+
.btn-file {
position: relative;
overflow: hidden;
@@ -312,6 +370,12 @@ div[contenteditable]:empty:not(:focus):before{
display: block;
}
+.night .btn.focus,
+.night .btn:focus,
+.night .btn:hover{
+ color: #fff;
+}
+
.info-label {
width: 36%;
text-align: right;
@@ -481,8 +545,8 @@ div[contenteditable]:empty:not(:focus):before{
border: 1px solid #2893ef;
}
-.status-bar .indent-width-input::-webkit-inner-spin-button,
-.status-bar .indent-width-input::-webkit-outer-spin-button {
+.status-bar .indent-width-input::-webkit-inner-spin-button,
+.status-bar .indent-width-input::-webkit-outer-spin-button {
-webkit-appearance: none;
margin: 0;
}
@@ -524,4 +588,4 @@ div[contenteditable]:empty:not(:focus):before{
.CodeMirror {
height: auto !important;
}
-} \ No newline at end of file
+}
diff --git a/public/css/markdown.css b/public/css/markdown.css
index ad3a655f..6741729d 100644
--- a/public/css/markdown.css
+++ b/public/css/markdown.css
@@ -69,6 +69,12 @@
border-collapse: inherit !important;
}
+.night .markdown-body .gist table tr:nth-child(2n){
+
+ background-color: #ddd;
+
+}
+
.markdown-body code[data-gist-id] {
background: none;
padding: 0;
@@ -93,6 +99,7 @@
.markdown-body code[data-gist-id] table tr {
background: unset;
+
}
/*fixed style for rtl in pre and code*/
@@ -121,6 +128,16 @@
white-space: inherit;
}
+.night .markdown-body pre.graphviz .graph > polygon{
+ fill: #333;
+}
+
+.night .markdown-body pre.mermaid .titleText,
+.night .markdown-body pre.mermaid text,
+.night .markdown-body pre.mermaid .sectionTitle{
+ fill: white;
+}
+
.markdown-body pre.flow-chart > code,
.markdown-body pre.sequence-diagram > code,
.markdown-body pre.graphviz > code,
@@ -138,6 +155,27 @@
height: 100%;
}
+.night .markdown-body .abc path{
+ fill: #eee;
+}
+
+.night .markdown-body .abc path.note_selected{
+ fill: ##4DD0E1;
+}
+
+.night tspan{
+ fill: #fefefe;
+}
+
+.night pre rect{
+ fill: transparent;
+}
+
+.night pre.flow-chart rect,
+.night pre.flow-chart path{
+ stroke: white;
+}
+
.markdown-body pre > code.wrap {
white-space: pre-wrap; /* Since CSS 2.1 */
white-space: -moz-pre-wrap; /* Mozilla, since 1999 */
diff --git a/public/docs/release-notes.md b/public/docs/release-notes.md
index 2e0a71c6..2e0a71c6 100755..100644
--- a/public/docs/release-notes.md
+++ b/public/docs/release-notes.md
diff --git a/public/js/extra.js b/public/js/extra.js
index 13b8924c..75aa29af 100644
--- a/public/js/extra.js
+++ b/public/js/extra.js
@@ -156,7 +156,11 @@ export function renderTags (view) {
}
function slugifyWithUTF8 (text) {
- let newText = S(text.toLowerCase()).trim().stripTags().dasherize().s
+ // remove html tags and trim spaces
+ let newText = S(text).trim().stripTags().s
+ // replace all spaces in between to dashes
+ newText = newText.replace(/\s+/g, '-')
+ // slugify string to make it valid for attribute
newText = newText.replace(/([!"#$%&'()*+,./:;<=>?@[\\\]^`{|}~])/g, '')
return newText
}
diff --git a/public/js/index.js b/public/js/index.js
index b336af90..5ff716fd 100644
--- a/public/js/index.js
+++ b/public/js/index.js
@@ -1633,6 +1633,10 @@ ui.toolbar.view.click(function () {
ui.toolbar.both.click(function () {
changeMode(modeType.both)
})
+
+ui.toolbar.night.click(function () {
+ toggleNightMode()
+})
// permission
// freely
ui.infobar.permission.freely.click(function () {
@@ -1666,6 +1670,17 @@ $('.ui-delete-modal-confirm').click(function () {
socket.emit('delete')
})
+function toggleNightMode () {
+ var $body = $('body')
+ var isActive = ui.toolbar.night.hasClass('active')
+ if (isActive) {
+ $body.removeClass('night')
+ appState.nightMode = false
+ } else {
+ $body.addClass('night')
+ appState.nightMode = true
+ }
+}
function emitPermission (_permission) {
if (_permission !== permission) {
socket.emit('permission', _permission)
diff --git a/public/js/lib/appState.js b/public/js/lib/appState.js
index fb8030e1..87aaf737 100644
--- a/public/js/lib/appState.js
+++ b/public/js/lib/appState.js
@@ -2,7 +2,8 @@ import modeType from './modeType'
let state = {
syncscroll: true,
- currentMode: modeType.view
+ currentMode: modeType.view,
+ nightMode: false
}
export default state
diff --git a/public/js/lib/editor/index.js b/public/js/lib/editor/index.js
index 33c1e0d4..003b32b7 100644
--- a/public/js/lib/editor/index.js
+++ b/public/js/lib/editor/index.js
@@ -171,13 +171,13 @@ export default class Editor {
this.statusLength.text('Length ' + docLength)
if (docLength > (config.docmaxlength * 0.95)) {
this.statusLength.css('color', 'red')
- this.statusLength.attr('title', 'Your almost reach note max length limit.')
+ this.statusLength.attr('title', 'You have almost reached the limit for this document.')
} else if (docLength > (config.docmaxlength * 0.8)) {
this.statusLength.css('color', 'orange')
- this.statusLength.attr('title', 'You nearly fill the note, consider to make more pieces.')
+ this.statusLength.attr('title', 'This document is nearly full, consider splitting it or creating a new one.')
} else {
this.statusLength.css('color', 'white')
- this.statusLength.attr('title', 'You could write up to ' + config.docmaxlength + ' characters in this note.')
+ this.statusLength.attr('title', 'You can write up to ' + config.docmaxlength + ' characters in this document.')
}
}
diff --git a/public/js/lib/editor/ui-elements.js b/public/js/lib/editor/ui-elements.js
index 0d330d77..88a1e3ca 100644
--- a/public/js/lib/editor/ui-elements.js
+++ b/public/js/lib/editor/ui-elements.js
@@ -37,6 +37,7 @@ export const getUIElements = () => ({
edit: $('.ui-edit'),
view: $('.ui-view'),
both: $('.ui-both'),
+ night: $('.ui-night'),
uploadImage: $('.ui-upload-image')
},
infobar: {
diff --git a/public/js/reveal-markdown.js b/public/js/reveal-markdown.js
index d15b5ebd..d15b5ebd 100755..100644
--- a/public/js/reveal-markdown.js
+++ b/public/js/reveal-markdown.js
diff --git a/public/views/hackmd/header.ejs b/public/views/hackmd/header.ejs
index 80df2c77..b87f21fa 100644
--- a/public/views/hackmd/header.ejs
+++ b/public/views/hackmd/header.ejs
@@ -96,6 +96,11 @@
<input type="radio" name="mode" autocomplete="off"><i class="fa fa-pencil"></i>
</label>
</div>
+ <div class="btn-group" data-toggle="buttons">
+ <label class="btn ui-night" title="<%= __('Night Theme') %>">
+ <input type="checkbox" name="night"><i class="fa fa-moon-o"></i>
+ </label>
+ </div>
<span class="btn btn-link btn-file ui-help" title="<%= __('Help') %>" data-toggle="modal" data-target=".help-modal">
<i class="fa fa-question-circle"></i>
</span>
diff --git a/public/views/index/body.ejs b/public/views/index/body.ejs
index 230eb117..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 || mattermost || 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 || mattermost || 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 89b542e9..7b44cfb0 100644
--- a/public/views/shared/signin-modal.ejs
+++ b/public/views/shared/signin-modal.ejs
@@ -43,7 +43,12 @@
<i class="fa fa-google"></i> <%= __('Sign in via %s', 'Google') %>
</a>
<% } %>
- <% if((facebook || twitter || github || gitlab || mattermost || 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) { %>
diff --git a/yarn.lock b/yarn.lock
index d80251f5..b9f9e9d7 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -282,6 +282,12 @@ async@^2.1.4:
dependencies:
lodash "^4.14.0"
+async@^2.1.5:
+ version "2.6.0"
+ resolved "https://registry.yarnpkg.com/async/-/async-2.6.0.tgz#61a29abb6fcc026fea77e56d1c6ec53a795951f4"
+ dependencies:
+ lodash "^4.14.0"
+
async@~0.2.6:
version "0.2.10"
resolved "https://registry.yarnpkg.com/async/-/async-0.2.10.tgz#b6bbe0b0674b9d719708ca38de8c237cb526c3d1"
@@ -1972,7 +1978,7 @@ ejs-loader@^0.3.0:
loader-utils "^0.2.7"
lodash "^3.6.0"
-ejs@^2.5.5:
+ejs@^2.5.5, ejs@^2.5.6:
version "2.5.7"
resolved "https://registry.yarnpkg.com/ejs/-/ejs-2.5.7.tgz#cc872c168880ae3c7189762fd5ffc00896c9518a"
@@ -4734,6 +4740,10 @@ node-dir@^0.1.10:
dependencies:
minimatch "^3.0.2"
+node-forge@^0.7.0:
+ version "0.7.1"
+ resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-0.7.1.tgz#9da611ea08982f4b94206b3beb4cc9665f20c300"
+
node-libs-browser@^0.7.0:
version "0.7.0"
resolved "https://registry.yarnpkg.com/node-libs-browser/-/node-libs-browser-0.7.0.tgz#3e272c0819e308935e26674408d7af0e1491b83b"
@@ -5183,7 +5193,19 @@ passport-oauth@^1.0.0:
passport-oauth1 "1.x.x"
passport-oauth2 "1.x.x"
-passport-strategy@1.x.x:
+passport-saml@^0.31.0:
+ version "0.31.0"
+ resolved "https://registry.yarnpkg.com/passport-saml/-/passport-saml-0.31.0.tgz#e4d654cab30f018bfd39056efe7bcfa770aab463"
+ dependencies:
+ passport-strategy "*"
+ q "^1.5.0"
+ xml-crypto "^0.10.1"
+ xml-encryption "^0.11.0"
+ xml2js "0.4.x"
+ xmlbuilder "^9.0.4"
+ xmldom "0.1.x"
+
+passport-strategy@*, passport-strategy@1.x.x:
version "1.0.0"
resolved "https://registry.yarnpkg.com/passport-strategy/-/passport-strategy-1.0.0.tgz#b5539aa8fc225a3d1ad179476ddf236b440f52e4"
@@ -5748,6 +5770,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"
+q@^1.5.0:
+ version "1.5.1"
+ resolved "https://registry.yarnpkg.com/q/-/q-1.5.1.tgz#7e32f75b41381291d04611f1bf14109ac00651d7"
+
qs@2.3.3:
version "2.3.3"
resolved "https://registry.yarnpkg.com/qs/-/qs-2.3.3.tgz#e9e85adbe75da0bbe4c8e0476a086290f863b404"
@@ -7565,6 +7591,23 @@ xml-char-classes@^1.0.0:
version "1.0.0"
resolved "https://registry.yarnpkg.com/xml-char-classes/-/xml-char-classes-1.0.0.tgz#64657848a20ffc5df583a42ad8a277b4512bbc4d"
+xml-crypto@^0.10.1:
+ version "0.10.1"
+ resolved "https://registry.yarnpkg.com/xml-crypto/-/xml-crypto-0.10.1.tgz#f832f74ccf56f24afcae1163a1fcab44d96774a8"
+ dependencies:
+ xmldom "=0.1.19"
+ xpath.js ">=0.0.3"
+
+xml-encryption@^0.11.0:
+ version "0.11.0"
+ resolved "https://registry.yarnpkg.com/xml-encryption/-/xml-encryption-0.11.0.tgz#458c2cb7d0300ff62d304c74eb3ded08ca97456b"
+ dependencies:
+ async "^2.1.5"
+ ejs "^2.5.6"
+ node-forge "^0.7.0"
+ xmldom "~0.1.15"
+ xpath "0.0.24"
+
xml-name-validator@^2.0.1:
version "2.0.1"
resolved "https://registry.yarnpkg.com/xml-name-validator/-/xml-name-validator-2.0.1.tgz#4d8b8f1eccd3419aa362061becef515e1e559635"
@@ -7576,16 +7619,31 @@ xml2js@0.4.17:
sax ">=0.6.0"
xmlbuilder "^4.1.0"
+xml2js@0.4.x:
+ version "0.4.19"
+ resolved "https://registry.yarnpkg.com/xml2js/-/xml2js-0.4.19.tgz#686c20f213209e94abf0d1bcf1efaa291c7827a7"
+ dependencies:
+ sax ">=0.6.0"
+ xmlbuilder "~9.0.1"
+
xmlbuilder@4.2.1, xmlbuilder@^4.1.0:
version "4.2.1"
resolved "https://registry.yarnpkg.com/xmlbuilder/-/xmlbuilder-4.2.1.tgz#aa58a3041a066f90eaa16c2f5389ff19f3f461a5"
dependencies:
lodash "^4.0.0"
-xmldom@0.1.x:
+xmlbuilder@^9.0.4, xmlbuilder@~9.0.1:
+ version "9.0.4"
+ resolved "https://registry.yarnpkg.com/xmlbuilder/-/xmlbuilder-9.0.4.tgz#519cb4ca686d005a8420d3496f3f0caeecca580f"
+
+xmldom@0.1.x, xmldom@~0.1.15:
version "0.1.27"
resolved "https://registry.yarnpkg.com/xmldom/-/xmldom-0.1.27.tgz#d501f97b3bdb403af8ef9ecc20573187aadac0e9"
+xmldom@=0.1.19:
+ version "0.1.19"
+ resolved "https://registry.yarnpkg.com/xmldom/-/xmldom-0.1.19.tgz#631fc07776efd84118bf25171b37ed4d075a0abc"
+
xmlhttprequest-ssl@1.5.3:
version "1.5.3"
resolved "https://registry.yarnpkg.com/xmlhttprequest-ssl/-/xmlhttprequest-ssl-1.5.3.tgz#185a888c04eca46c3e4070d99f7b49de3528992d"
@@ -7594,6 +7652,14 @@ xmlhttprequest@>=1.5.0:
version "1.8.0"
resolved "https://registry.yarnpkg.com/xmlhttprequest/-/xmlhttprequest-1.8.0.tgz#67fe075c5c24fef39f9d65f5f7b7fe75171968fc"
+xpath.js@>=0.0.3:
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/xpath.js/-/xpath.js-1.1.0.tgz#3816a44ed4bb352091083d002a383dd5104a5ff1"
+
+xpath@0.0.24:
+ version "0.0.24"
+ resolved "https://registry.yarnpkg.com/xpath/-/xpath-0.0.24.tgz#1ade162e1cc523c8d39fc7d06afc16ea216f29fb"
+
xregexp@^2.0.0:
version "2.0.0"
resolved "https://registry.yarnpkg.com/xregexp/-/xregexp-2.0.0.tgz#52a63e56ca0b84a7f3a5f3d61872f126ad7a5943"