From 27bea8182fdd36a76df1632c64155bff56551f34 Mon Sep 17 00:00:00 2001 From: SimonFJ20 Date: Mon, 13 May 2024 23:50:54 +0200 Subject: [PATCH] tdd hawd >~< --- crypto.ts | 26 +++++++ database.ts | 22 ++++++ deno.jsonc | 5 ++ deno.lock | 190 +++++++++++++++++++++++++++++++++++++++++++++ main.ts | 19 +++++ mockdb.ts | 75 ++++++++++++++++++ public/favicon.ico | Bin 0 -> 11454 bytes public/index.html | 9 +++ sessions.ts | 47 +++++++++++ sessions_test.ts | 85 ++++++++++++++++++++ users.ts | 75 ++++++++++++++++++ users_test.ts | 132 +++++++++++++++++++++++++++++++ utils.ts | 16 ++++ 13 files changed, 701 insertions(+) create mode 100644 crypto.ts create mode 100644 database.ts create mode 100644 deno.jsonc create mode 100644 deno.lock create mode 100644 main.ts create mode 100644 mockdb.ts create mode 100644 public/favicon.ico create mode 100644 public/index.html create mode 100644 sessions.ts create mode 100644 sessions_test.ts create mode 100644 users.ts create mode 100644 users_test.ts create mode 100644 utils.ts diff --git a/crypto.ts b/crypto.ts new file mode 100644 index 0000000..f2f8236 --- /dev/null +++ b/crypto.ts @@ -0,0 +1,26 @@ +import * as bcrypt from "https://deno.land/x/bcrypt@v0.4.1/mod.ts"; + +export interface Crypto { + hash(value: string): Promise; + compare(value: string, hash: string): Promise; +} + +export class BcryptCrypto implements Crypto { + public hash(value: string): Promise { + return bcrypt.hash(value); + } + public compare(value: string, hash: string): Promise { + return bcrypt.compare(value, hash); + } +} + +export class MockCrypto implements Crypto { + // deno-lint-ignore require-await + public async hash(value: string): Promise { + return bcrypt.hashSync(value, bcrypt.genSaltSync(4)); + } + // deno-lint-ignore require-await + public async compare(value: string, hash: string): Promise { + return bcrypt.compareSync(value, hash); + } +} diff --git a/database.ts b/database.ts new file mode 100644 index 0000000..918c7d7 --- /dev/null +++ b/database.ts @@ -0,0 +1,22 @@ +import { Option, Result } from "./utils.ts"; + +export type User = { + id: number; + username: string; + passwordHash: string; +}; + +export type Session = { + id: number; + userId: number; +}; + +export interface Database { + createUser(init: Omit): Promise>; + userWithId(id: number): Promise, string>>; + userWithUsername(username: string): Promise, string>>; + userWithUsernameExists(username: string): Promise>; + + createSession(init: Omit): Promise>; + sessionWithId(id: number): Promise, string>>; +} diff --git a/deno.jsonc b/deno.jsonc new file mode 100644 index 0000000..3fecf5f --- /dev/null +++ b/deno.jsonc @@ -0,0 +1,5 @@ +{ + "fmt": { + "indentWidth": 4 + } +} diff --git a/deno.lock b/deno.lock new file mode 100644 index 0000000..ba9c7ea --- /dev/null +++ b/deno.lock @@ -0,0 +1,190 @@ +{ + "version": "3", + "redirects": { + "https://deno.land/std/assert/mod.ts": "https://deno.land/std@0.224.0/assert/mod.ts" + }, + "remote": { + "https://deno.land/std@0.200.0/_util/os.ts": "d932f56d41e4f6a6093d56044e29ce637f8dcc43c5a90af43504a889cf1775e3", + "https://deno.land/std@0.200.0/assert/assert.ts": "9a97dad6d98c238938e7540736b826440ad8c1c1e54430ca4c4e623e585607ee", + "https://deno.land/std@0.200.0/assert/assertion_error.ts": "4d0bde9b374dfbcbe8ac23f54f567b77024fb67dbb1906a852d67fe050d42f56", + "https://deno.land/std@0.200.0/async/deferred.ts": "42790112f36a75a57db4a96d33974a936deb7b04d25c6084a9fa8a49f135def8", + "https://deno.land/std@0.200.0/bytes/bytes_list.ts": "31d664f4d42fa922066405d0e421c56da89d751886ee77bbe25a88bf0310c9d0", + "https://deno.land/std@0.200.0/bytes/concat.ts": "d26d6f3d7922e6d663dacfcd357563b7bf4a380ce5b9c2bbe0c8586662f25ce2", + "https://deno.land/std@0.200.0/bytes/copy.ts": "939d89e302a9761dcf1d9c937c7711174ed74c59eef40a1e4569a05c9de88219", + "https://deno.land/std@0.200.0/bytes/ends_with.ts": "4228811ebc71615d27f065c54b5e815ec1972538772b0f413c0efe05245b472e", + "https://deno.land/std@0.200.0/bytes/equals.ts": "fc190cce412b2136979181b163ec7e05f7e7a947e39102eee4b8c0d62519ddf9", + "https://deno.land/std@0.200.0/bytes/includes_needle.ts": "76a8163126fb2f8bf86fd7f22192c3bb04bf6a20b987a095127c2ca08adf3ba6", + "https://deno.land/std@0.200.0/bytes/index_of_needle.ts": "9c06610e9611b5647ac25952e71a22e09227c9f1b8cbeeb33399bf8bf8a7f649", + "https://deno.land/std@0.200.0/bytes/last_index_of_needle.ts": "f1602f221c3b678bc4f1e1c88a70a15ab7da32c21751dbbc6c957c956951d784", + "https://deno.land/std@0.200.0/bytes/mod.ts": "e869bba1e7a2e3a9cc6c2d55471888429a544e70a840c087672e656e7ba21815", + "https://deno.land/std@0.200.0/bytes/repeat.ts": "6f5e490d8d72bcbf8d84a6bb04690b9b3eb5822c5a11687bca73a2318a842294", + "https://deno.land/std@0.200.0/bytes/starts_with.ts": "3e607a70c9c09f5140b7a7f17a695221abcc7244d20af3eb47ccbb63f5885135", + "https://deno.land/std@0.200.0/crypto/keystack.ts": "877ab0f19eb7d37ad6495190d3c3e39f58e9c52e0b6a966f82fd6df67ca55f90", + "https://deno.land/std@0.200.0/crypto/timing_safe_equal.ts": "7b0a4d2ef1c17590e0ad6c0cb1776369d2ba80cd99e945005e117851690507fe", + "https://deno.land/std@0.200.0/encoding/base64.ts": "144ae6234c1fbe5b68666c711dc15b1e9ee2aef6d42b3b4345bf9a6c91d70d0d", + "https://deno.land/std@0.200.0/encoding/base64url.ts": "2ed4ba122b20fedf226c5d337cf22ee2024fa73a8f85d915d442af7e9ce1fae1", + "https://deno.land/std@0.200.0/http/_negotiation/common.ts": "14d1a52427ab258a4b7161cd80e1d8a207b7cc64b46e911780f57ead5f4323c6", + "https://deno.land/std@0.200.0/http/_negotiation/encoding.ts": "ff747d107277c88cb7a6a62a08eeb8d56dad91564cbcccb30694d5dc126dcc53", + "https://deno.land/std@0.200.0/http/_negotiation/language.ts": "7bcddd8db3330bdb7ce4fc00a213c5547c1968139864201efd67ef2d0d51887d", + "https://deno.land/std@0.200.0/http/_negotiation/media_type.ts": "58847517cd549384ad677c0fe89e0a4815be36fe7a303ea63cee5f6a1d7e1692", + "https://deno.land/std@0.200.0/http/cookie_map.ts": "64601025a7d24c3ebd80a169ccc99145bdbfc60606935348e1d4366d0bf9010d", + "https://deno.land/std@0.200.0/http/etag.ts": "807382795850cde5c437c74bcc09392bc0fc56de348fc1271f383f4b28935b9f", + "https://deno.land/std@0.200.0/http/http_errors.ts": "bbda34819060af86537cecc9dc8e045f877130808b7e7acde4197c5328e852d0", + "https://deno.land/std@0.200.0/http/http_status.ts": "8a7bcfe3ac025199ad804075385e57f63d055b2aed539d943ccc277616d6f932", + "https://deno.land/std@0.200.0/http/method.ts": "e66c2a015cb46c21ab0bb3589aa4fca43143a506cb324ffdfd42d2edef7bc0c4", + "https://deno.land/std@0.200.0/http/negotiation.ts": "46e74a6bad4b857333a58dc5b50fe8e5a4d5267e97292293ea65f980bd918086", + "https://deno.land/std@0.200.0/http/server_sent_event.ts": "29f707c1afa5278ac0315ac115ee679d6b93596d5af3fad5ef33f04254ca76c1", + "https://deno.land/std@0.200.0/http/user_agent.ts": "35d3c70d0926b0e121b8c1bbc324b3522479158acaa4f0c43928362b7bf4e6f4", + "https://deno.land/std@0.200.0/io/buf_reader.ts": "2bccff0878537ef201c5051fc0db0ce8196388c5ea69d2be6be1900fe48c5f4b", + "https://deno.land/std@0.200.0/io/buf_writer.ts": "48c33c8f00b61dcbc7958706741cec8e59810bd307bc6a326cbd474fe8346dfd", + "https://deno.land/std@0.200.0/io/buffer.ts": "4d6883daeb2e698579c4064170515683d69f40f3de019bfe46c5cf31e74ae793", + "https://deno.land/std@0.200.0/io/copy_n.ts": "c055296297b9d4897d90d1ac056b072dc02614e60c67f438e23fbce052ea4c69", + "https://deno.land/std@0.200.0/io/limited_reader.ts": "6c9a216f8eef39c1ee2a6b37a29372c8fc63455b2eeb91f06d9646f8f759fc8b", + "https://deno.land/std@0.200.0/io/mod.ts": "2665bcccc1fd6e8627cca167c3e92aaecbd9897556b6f69e6d258070ef63fd9b", + "https://deno.land/std@0.200.0/io/multi_reader.ts": "9c2a0a31686c44b277e16da1d97b4686a986edcee48409b84be25eedbc39b271", + "https://deno.land/std@0.200.0/io/read_delim.ts": "c02b93cc546ae8caad8682ae270863e7ace6daec24c1eddd6faabc95a9d876a3", + "https://deno.land/std@0.200.0/io/read_int.ts": "7cb8bcdfaf1107586c3bacc583d11c64c060196cb070bb13ae8c2061404f911f", + "https://deno.land/std@0.200.0/io/read_lines.ts": "c526c12a20a9386dc910d500f9cdea43cba974e853397790bd146817a7eef8cc", + "https://deno.land/std@0.200.0/io/read_long.ts": "f0aaa420e3da1261c5d33c5e729f09922f3d9fa49f046258d4ff7a00d800c71e", + "https://deno.land/std@0.200.0/io/read_range.ts": "46a2263d0f8369b6d9abb0b25d99ceb65ff08d621fc57bcc53832e6979295043", + "https://deno.land/std@0.200.0/io/read_short.ts": "805cb329574b850b84bf14a92c052c59b5977a492cd780c41df8ad40826c1a20", + "https://deno.land/std@0.200.0/io/read_string_delim.ts": "5dc9f53bdf78e7d4ee1e56b9b60352238ab236a71c3e3b2a713c3d78472a53ce", + "https://deno.land/std@0.200.0/io/slice_long_to_bytes.ts": "48d9bace92684e880e46aa4a2520fc3867f9d7ce212055f76ecc11b22f9644b7", + "https://deno.land/std@0.200.0/io/string_reader.ts": "da0f68251b3d5b5112485dfd4d1b1936135c9b4d921182a7edaf47f74c25cc8f", + "https://deno.land/std@0.200.0/io/string_writer.ts": "8a03c5858c24965a54c6538bed15f32a7c72f5704a12bda56f83a40e28e5433e", + "https://deno.land/std@0.200.0/media_types/_db.ts": "7606d83e31f23ce1a7968cbaee852810c2cf477903a095696cdc62eaab7ce570", + "https://deno.land/std@0.200.0/media_types/_util.ts": "916efbd30b6148a716f110e67a4db29d6949bf4048997b754415dd7e42c52378", + "https://deno.land/std@0.200.0/media_types/content_type.ts": "ad98a5aa2d95f5965b2796072284258710a25e520952376ed432b0937ce743bc", + "https://deno.land/std@0.200.0/media_types/extension.ts": "a7cd28c9417143387cdfed27d4e8607ebcf5b1ec27eb8473d5b000144689fe65", + "https://deno.land/std@0.200.0/media_types/extensions_by_type.ts": "43806d6a52a0d6d965ada9d20e60a982feb40bc7a82268178d94edb764694fed", + "https://deno.land/std@0.200.0/media_types/format_media_type.ts": "f5e1073c05526a6f5a516ac5c5587a1abd043bf1039c71cde1166aa4328c8baf", + "https://deno.land/std@0.200.0/media_types/get_charset.ts": "18b88274796fda5d353806bf409eb1d2ddb3f004eb4bd311662c4cdd8ac173db", + "https://deno.land/std@0.200.0/media_types/mod.ts": "d3f0b99f85053bc0b98ecc24eaa3546dfa09b856dc0bbaf60d8956d2cdd710c8", + "https://deno.land/std@0.200.0/media_types/parse_media_type.ts": "8cb0144385c555c9ce81881b7cee3fbb746f23b4af988fecdb7bd01ef8cc35b1", + "https://deno.land/std@0.200.0/media_types/type_by_extension.ts": "daa801eb0f11cdf199445d0f1b656cf116d47dcf9e5b85cc1e6b4469f5ee0432", + "https://deno.land/std@0.200.0/media_types/vendor/mime-db.v1.52.0.ts": "6925bbcae81ca37241e3f55908d0505724358cda3384eaea707773b2c7e99586", + "https://deno.land/std@0.200.0/path/_basename.ts": "057d420c9049821f983f784fd87fa73ac471901fb628920b67972b0f44319343", + "https://deno.land/std@0.200.0/path/_constants.ts": "e49961f6f4f48039c0dfed3c3f93e963ca3d92791c9d478ac5b43183413136e0", + "https://deno.land/std@0.200.0/path/_dirname.ts": "355e297236b2218600aee7a5301b937204c62e12da9db4b0b044993d9e658395", + "https://deno.land/std@0.200.0/path/_extname.ts": "eaaa5aae1acf1f03254d681bd6a8ce42a9cb5b7ff2213a9d4740e8ab31283664", + "https://deno.land/std@0.200.0/path/_format.ts": "4a99270d6810f082e614309164fad75d6f1a483b68eed97c830a506cc589f8b4", + "https://deno.land/std@0.200.0/path/_from_file_url.ts": "7e4e5626089785adddb061f1b9f4932d6b21c7df778e7449531a11e32048245c", + "https://deno.land/std@0.200.0/path/_interface.ts": "6471159dfbbc357e03882c2266d21ef9afdb1e4aa771b0545e90db58a0ba314b", + "https://deno.land/std@0.200.0/path/_is_absolute.ts": "05dac10b5e93c63198b92e3687baa2be178df5321c527dc555266c0f4f51558c", + "https://deno.land/std@0.200.0/path/_join.ts": "fd78555bc34d5f188918fc7018dfe8fe2df5bbad94a3b30a433666c03934d77f", + "https://deno.land/std@0.200.0/path/_normalize.ts": "a19ec8706b2707f9dd974662a5cd89fad438e62ab1857e08b314a8eb49a34d81", + "https://deno.land/std@0.200.0/path/_parse.ts": "0f9b0ff43682dd9964eb1c4398610c4e165d8db9d3ac9d594220217adf480cfa", + "https://deno.land/std@0.200.0/path/_relative.ts": "27bdeffb5311a47d85be26d37ad1969979359f7636c5cd9fcf05dcd0d5099dc5", + "https://deno.land/std@0.200.0/path/_resolve.ts": "7a3616f1093735ed327e758313b79c3c04ea921808ca5f19ddf240cb68d0adf6", + "https://deno.land/std@0.200.0/path/_to_file_url.ts": "739bfda583598790b2e77ce227f2bb618f6ebdb939788cea47555b43970ec58c", + "https://deno.land/std@0.200.0/path/_to_namespaced_path.ts": "0d5f4caa2ed98ef7a8786286df6af804b50e38859ae897b5b5b4c8c5930a75c8", + "https://deno.land/std@0.200.0/path/_util.ts": "4e191b1bac6b3bf0c31aab42e5ca2e01a86ab5a0d2e08b75acf8585047a86221", + "https://deno.land/std@0.200.0/path/basename.ts": "6f08fbb90dbfcf320765b3abb01f995b1723f75e2534acfd5380e202c802a3aa", + "https://deno.land/std@0.200.0/path/common.ts": "ee7505ab01fd22de3963b64e46cff31f40de34f9f8de1fff6a1bd2fe79380000", + "https://deno.land/std@0.200.0/path/dirname.ts": "098996822a31b4c46e1eb52a19540d3c6f9f54b772fc8a197939eeabc29fca2f", + "https://deno.land/std@0.200.0/path/extname.ts": "9b83c62fd16505739541f7a3ab447d8972da39dbf668d47af2f93206c2480893", + "https://deno.land/std@0.200.0/path/format.ts": "cb22f95cc7853d590b87708cc9441785e760d711188facff3d225305a8213aca", + "https://deno.land/std@0.200.0/path/from_file_url.ts": "a6221cfc928928ec4d9786d767dfac98fa2ab746af0786446c9834a07b98817e", + "https://deno.land/std@0.200.0/path/glob.ts": "d479e0a695621c94d3fd7fe7abd4f9499caf32a8de13f25073451c6ef420a4e1", + "https://deno.land/std@0.200.0/path/is_absolute.ts": "6b3d36352eb7fa29edb53f9e7b09b1aeb022a3c5465764f6cc5b8c41f9736197", + "https://deno.land/std@0.200.0/path/join.ts": "4a2867ff2f3c81ffc9eb3d56dade16db6f8bd3854f269306d23dad4115089c84", + "https://deno.land/std@0.200.0/path/mod.ts": "7765507696cb321994cdacfc19ee3ba61e8e3ebf4bd98fa75a276cf5dc18ce2a", + "https://deno.land/std@0.200.0/path/normalize.ts": "7d992cd262b2deefa842d93a8ba2ed51f3949ba595b1d07f627ac2cddbc74808", + "https://deno.land/std@0.200.0/path/parse.ts": "031fe488b3497fb8312fc1dc3c3d6c2d80707edd9c661e18ee9fd20f95edf322", + "https://deno.land/std@0.200.0/path/posix.ts": "0a1c1952d132323a88736d03e92bd236f3ed5f9f079e5823fae07c8d978ee61b", + "https://deno.land/std@0.200.0/path/relative.ts": "7db80c5035016174267da16321a742d76e875215c317859a383b12f413c6f5d6", + "https://deno.land/std@0.200.0/path/resolve.ts": "103b62207726a27f28177f397008545804ecb20aaf00623af1f622b18cd80b9f", + "https://deno.land/std@0.200.0/path/separator.ts": "0fb679739d0d1d7bf45b68dacfb4ec7563597a902edbaf3c59b50d5bcadd93b1", + "https://deno.land/std@0.200.0/path/to_file_url.ts": "dd32f7a01bbf3b15b5df46796659984b372973d9b2d7d59bcf0eb990763a0cb5", + "https://deno.land/std@0.200.0/path/to_namespaced_path.ts": "4e643ab729bf49ccdc166ad48d2de262ff462938fcf2a44a4425588f4a0bd690", + "https://deno.land/std@0.200.0/path/win32.ts": "8b3f80ef7a462511d5e8020ff490edcaa0a0d118f1b1e9da50e2916bdd73f9dd", + "https://deno.land/std@0.200.0/streams/_common.ts": "f45cba84f0d813de3326466095539602364a9ba521f804cc758f7a475cda692d", + "https://deno.land/std@0.200.0/streams/buffer.ts": "6cd773d22cf21bb988a98cc551b5abfc4c3b03516f93eaa3fb6f2f6e16032deb", + "https://deno.land/std@0.200.0/streams/byte_slice_stream.ts": "c46d7c74836fc8c1a9acd9fe211cbe1bbaaee1b36087c834fb03af4991135c3a", + "https://deno.land/std@0.200.0/streams/copy.ts": "75cbc795ff89291df22ddca5252de88b2e16d40c85d02840593386a8a1454f71", + "https://deno.land/std@0.200.0/streams/delimiter_stream.ts": "f69e849b3d1f59f02424497273f411105a6f76a9f13da92aeeb9a2d554236814", + "https://deno.land/std@0.200.0/streams/early_zip_readable_streams.ts": "4005fa74162b943f79899e5d7cb96adcbc0a6b867f9144974ed12d30e0a556e1", + "https://deno.land/std@0.200.0/streams/iterate_reader.ts": "bbec1d45c2df2c0c5920bad0549351446fdc8e0886d99e95959b259dbcdb6072", + "https://deno.land/std@0.200.0/streams/limited_bytes_transform_stream.ts": "05dc592ffaab83257494d22dd53917e56243c26e5e3129b3f13ddbbbc4785048", + "https://deno.land/std@0.200.0/streams/limited_transform_stream.ts": "d69ab790232c1b86f53621ad41ef03c235f2abb4b7a1cd51960ad6e12ee55e38", + "https://deno.land/std@0.200.0/streams/merge_readable_streams.ts": "dc2db0cbf1b14d999aa2aa2a2a1ba24ce58953878f29845ed9319321d0a01fab", + "https://deno.land/std@0.200.0/streams/mod.ts": "c07ec010e700b9ea887dc36ca08729828bc7912f711e4054e24d33fd46282252", + "https://deno.land/std@0.200.0/streams/read_all.ts": "ee319772fb0fd28302f97343cc48dfcf948f154fd0d755d8efe65814b70533be", + "https://deno.land/std@0.200.0/streams/readable_stream_from_iterable.ts": "a355e97ba8671a4611ae9671c1f33c4a7e6a1b42fbdc9a399338ac3d6f875364", + "https://deno.land/std@0.200.0/streams/readable_stream_from_reader.ts": "bfc416c4576a30aac6b9af22c9dc292c20c6742141ee7c55b5e85460beb0c54e", + "https://deno.land/std@0.200.0/streams/reader_from_iterable.ts": "55f68110dce3f8f2c87b834d95f153bc904257fc65175f9f2abe78455cb8047c", + "https://deno.land/std@0.200.0/streams/reader_from_stream_reader.ts": "fa4971e5615a010e49492c5d1688ca1a4d17472a41e98b498ab89a64ebd7ac73", + "https://deno.land/std@0.200.0/streams/text_delimiter_stream.ts": "20e680ab8b751390e359288ce764f9c47d164af11a263870746eeca4bc7d976b", + "https://deno.land/std@0.200.0/streams/text_line_stream.ts": "0f2c4b33a5fdb2476f2e060974cba1347cefe99a4af33c28a57524b1a34750fa", + "https://deno.land/std@0.200.0/streams/to_transform_stream.ts": "89fd367cafb3b6d80d61e2f4f1fcf66cc75723ecee8d474b495f022264ec6c3b", + "https://deno.land/std@0.200.0/streams/writable_stream_from_writer.ts": "56fff5c82fb736fdd669b567cc0b2bbbe0351002cd13254eae26c366e2bed89a", + "https://deno.land/std@0.200.0/streams/write_all.ts": "aec90152978581ea62d56bb53a5cbf487e6a89c902f87c5969681ffbdf32b998", + "https://deno.land/std@0.200.0/streams/writer_from_stream_writer.ts": "07c7ee025151a190f37fc42cbb01ff93afc949119ebddc6e0d0df14df1bf6950", + "https://deno.land/std@0.200.0/streams/zip_readable_streams.ts": "a9d81aa451240f79230add674809dbee038d93aabe286e2d9671e66591fc86ca", + "https://deno.land/std@0.224.0/assert/_constants.ts": "a271e8ef5a573f1df8e822a6eb9d09df064ad66a4390f21b3e31f820a38e0975", + "https://deno.land/std@0.224.0/assert/assert.ts": "09d30564c09de846855b7b071e62b5974b001bb72a4b797958fe0660e7849834", + "https://deno.land/std@0.224.0/assert/assert_almost_equals.ts": "9e416114322012c9a21fa68e187637ce2d7df25bcbdbfd957cd639e65d3cf293", + "https://deno.land/std@0.224.0/assert/assert_array_includes.ts": "14c5094471bc8e4a7895fc6aa5a184300d8a1879606574cb1cd715ef36a4a3c7", + "https://deno.land/std@0.224.0/assert/assert_equals.ts": "3bbca947d85b9d374a108687b1a8ba3785a7850436b5a8930d81f34a32cb8c74", + "https://deno.land/std@0.224.0/assert/assert_exists.ts": "43420cf7f956748ae6ed1230646567b3593cb7a36c5a5327269279c870c5ddfd", + "https://deno.land/std@0.224.0/assert/assert_false.ts": "3e9be8e33275db00d952e9acb0cd29481a44fa0a4af6d37239ff58d79e8edeff", + "https://deno.land/std@0.224.0/assert/assert_greater.ts": "5e57b201fd51b64ced36c828e3dfd773412c1a6120c1a5a99066c9b261974e46", + "https://deno.land/std@0.224.0/assert/assert_greater_or_equal.ts": "9870030f997a08361b6f63400273c2fb1856f5db86c0c3852aab2a002e425c5b", + "https://deno.land/std@0.224.0/assert/assert_instance_of.ts": "e22343c1fdcacfaea8f37784ad782683ec1cf599ae9b1b618954e9c22f376f2c", + "https://deno.land/std@0.224.0/assert/assert_is_error.ts": "f856b3bc978a7aa6a601f3fec6603491ab6255118afa6baa84b04426dd3cc491", + "https://deno.land/std@0.224.0/assert/assert_less.ts": "60b61e13a1982865a72726a5fa86c24fad7eb27c3c08b13883fb68882b307f68", + "https://deno.land/std@0.224.0/assert/assert_less_or_equal.ts": "d2c84e17faba4afe085e6c9123a63395accf4f9e00150db899c46e67420e0ec3", + "https://deno.land/std@0.224.0/assert/assert_match.ts": "ace1710dd3b2811c391946954234b5da910c5665aed817943d086d4d4871a8b7", + "https://deno.land/std@0.224.0/assert/assert_not_equals.ts": "78d45dd46133d76ce624b2c6c09392f6110f0df9b73f911d20208a68dee2ef29", + "https://deno.land/std@0.224.0/assert/assert_not_instance_of.ts": "3434a669b4d20cdcc5359779301a0588f941ffdc2ad68803c31eabdb4890cf7a", + "https://deno.land/std@0.224.0/assert/assert_not_match.ts": "df30417240aa2d35b1ea44df7e541991348a063d9ee823430e0b58079a72242a", + "https://deno.land/std@0.224.0/assert/assert_not_strict_equals.ts": "37f73880bd672709373d6dc2c5f148691119bed161f3020fff3548a0496f71b8", + "https://deno.land/std@0.224.0/assert/assert_object_match.ts": "411450fd194fdaabc0089ae68f916b545a49d7b7e6d0026e84a54c9e7eed2693", + "https://deno.land/std@0.224.0/assert/assert_rejects.ts": "4bee1d6d565a5b623146a14668da8f9eb1f026a4f338bbf92b37e43e0aa53c31", + "https://deno.land/std@0.224.0/assert/assert_strict_equals.ts": "b4f45f0fd2e54d9029171876bd0b42dd9ed0efd8f853ab92a3f50127acfa54f5", + "https://deno.land/std@0.224.0/assert/assert_string_includes.ts": "496b9ecad84deab72c8718735373feb6cdaa071eb91a98206f6f3cb4285e71b8", + "https://deno.land/std@0.224.0/assert/assert_throws.ts": "c6508b2879d465898dab2798009299867e67c570d7d34c90a2d235e4553906eb", + "https://deno.land/std@0.224.0/assert/assertion_error.ts": "ba8752bd27ebc51f723702fac2f54d3e94447598f54264a6653d6413738a8917", + "https://deno.land/std@0.224.0/assert/equal.ts": "bddf07bb5fc718e10bb72d5dc2c36c1ce5a8bdd3b647069b6319e07af181ac47", + "https://deno.land/std@0.224.0/assert/fail.ts": "0eba674ffb47dff083f02ced76d5130460bff1a9a68c6514ebe0cdea4abadb68", + "https://deno.land/std@0.224.0/assert/mod.ts": "48b8cb8a619ea0b7958ad7ee9376500fe902284bb36f0e32c598c3dc34cbd6f3", + "https://deno.land/std@0.224.0/assert/unimplemented.ts": "8c55a5793e9147b4f1ef68cd66496b7d5ba7a9e7ca30c6da070c1a58da723d73", + "https://deno.land/std@0.224.0/assert/unreachable.ts": "5ae3dbf63ef988615b93eb08d395dda771c96546565f9e521ed86f6510c29e19", + "https://deno.land/std@0.224.0/fmt/colors.ts": "508563c0659dd7198ba4bbf87e97f654af3c34eb56ba790260f252ad8012e1c5", + "https://deno.land/std@0.224.0/internal/diff.ts": "6234a4b493ebe65dc67a18a0eb97ef683626a1166a1906232ce186ae9f65f4e6", + "https://deno.land/std@0.224.0/internal/format.ts": "0a98ee226fd3d43450245b1844b47003419d34d210fa989900861c79820d21c2", + "https://deno.land/std@0.224.0/internal/mod.ts": "534125398c8e7426183e12dc255bb635d94e06d0f93c60a297723abe69d3b22e", + "https://deno.land/x/bcrypt@v0.4.1/mod.ts": "ff09bdae282583cf5f7d87efe37ddcecef7f14f6d12e8b8066a3058db8c6c2f7", + "https://deno.land/x/bcrypt@v0.4.1/src/bcrypt/base64.ts": "b8266450a4f1eb6960f60f2f7986afc4dde6b45bd2d7ee7ba10789e67e17b9f7", + "https://deno.land/x/bcrypt@v0.4.1/src/bcrypt/bcrypt.ts": "ec221648cc6453ea5e3803bc817c01157dada06aa6f7a0ba6b9f87aae32b21e2", + "https://deno.land/x/bcrypt@v0.4.1/src/main.ts": "08d201b289c8d9c46f8839c69cd6625b213863db29775c7a200afc3b540e64f8", + "https://deno.land/x/bcrypt@v0.4.1/src/worker.ts": "5a73bdfee9c9e622f47c9733d374b627dce52fb3ec1e74c8226698b3fc57ffac", + "https://deno.land/x/oak@v12.6.1/application.ts": "3028d3f6fa5ee743de013881550d054372c11d83c45099c2d794033786d27008", + "https://deno.land/x/oak@v12.6.1/body.ts": "1899761b97fc9d776f3710b2637fb047ba29b968609afc6c0e5219b1108e703c", + "https://deno.land/x/oak@v12.6.1/buf_reader.ts": "26640736541598dbd9f2b84a9d0595756afff03c9ca55b66eef1911f7798b56d", + "https://deno.land/x/oak@v12.6.1/content_disposition.ts": "8b8c3cb2fba7138cd5b7f82fc3b5ea39b33db924a824b28261659db7e164621e", + "https://deno.land/x/oak@v12.6.1/context.ts": "895a2d40186b89c28ba3947bf08a9335f1a11fc33133f760082536b74c53d1ca", + "https://deno.land/x/oak@v12.6.1/deps.ts": "267ef76c25592101fe1f6c6d7730664015a9179c974da4f7441297d9367a9514", + "https://deno.land/x/oak@v12.6.1/etag.ts": "32e47726b41698aefdd71faac5aaf2907c9bdd41ef18a7693863be4f8fee115d", + "https://deno.land/x/oak@v12.6.1/forwarded.ts": "e656f96a85574e2a6ee54dc35efc9f72d543c9ae0923760ef426ee7369eef01c", + "https://deno.land/x/oak@v12.6.1/headers.ts": "769fd042d34fbcd5667cbd745b5c65d335cc8430e822dbf1f87d65313cab4b47", + "https://deno.land/x/oak@v12.6.1/helpers.ts": "6b03c6a2be06ec775d54449e442a2bac234aa952948ca758356eab6dc87af618", + "https://deno.land/x/oak@v12.6.1/http_server_native.ts": "98e12c50a959553cfc144bc00999c969fa69ca781cbd96bec563f55691ab82db", + "https://deno.land/x/oak@v12.6.1/http_server_native_request.ts": "552b174b5e13e92de8897d5b6f716b1e5a53543115481d65a651a41e4ca29ec9", + "https://deno.land/x/oak@v12.6.1/isMediaType.ts": "62d638abcf837ece3a8f07a4b7ca59794135cb0d4b73194c7d5837efd4161005", + "https://deno.land/x/oak@v12.6.1/mediaTyper.ts": "042b853fc8e9c3f6c628dd389e03ef481552bf07242efc3f8a1af042102a6105", + "https://deno.land/x/oak@v12.6.1/middleware.ts": "c7f7a0424a6dd99a00e4b8d7d6e131efc0facc8dea781845d713b63df8ef1862", + "https://deno.land/x/oak@v12.6.1/middleware/proxy.ts": "6f2799cf60d926e7a8d13ff757a59d7f0f930407db7ee9b81e7c064138eb89ff", + "https://deno.land/x/oak@v12.6.1/mod.ts": "f6aa47ad1b6099470c9a884cccad9d3ac0fd242ba940896291ab76cd26cf554b", + "https://deno.land/x/oak@v12.6.1/multipart.ts": "1484e01b98f5135f2aa09f7d0ce1e7be39109bf9f045ac660e941619d04e3d29", + "https://deno.land/x/oak@v12.6.1/range.ts": "1ca15fc1ac21c650c34e6997a75af2af9d9d8eb6fe2d5d1dadeac3dfd4a9c152", + "https://deno.land/x/oak@v12.6.1/request.ts": "32409827e285ee65889b22bbaaea5d6b280258124c2e9a4f724baa8e6d6375b7", + "https://deno.land/x/oak@v12.6.1/response.ts": "094d950a5158f5b3446ca8a7b6e975dd23afb42b38c38517cc2f41dc75b16b4c", + "https://deno.land/x/oak@v12.6.1/router.ts": "0f53d6249f9e8f89f2522b2b810b9302d0f22593c184b16b24b03bf2b7d42ea1", + "https://deno.land/x/oak@v12.6.1/send.ts": "5ec49f106294593f468317a0c885da4f3274bf6d0fe9e16a7304391730b4f4fb", + "https://deno.land/x/oak@v12.6.1/structured_clone.ts": "c3888b14d1eec558345bfbf13d0993d59bd45aaa8588444e35dd558c3a921cd8", + "https://deno.land/x/oak@v12.6.1/testing.ts": "37d684d57bb8e5150fb5eb2677e66b04dcb422709cf2c5a74c1df94d52aa02e2", + "https://deno.land/x/oak@v12.6.1/util.ts": "0a3fdffb114859c2de84e1783efa3a544af4d2af8c6f08e0d25655de9d3e69bb", + "https://deno.land/x/path_to_regexp@v6.2.1/index.ts": "894060567837bae8fc9c5cbd4d0a05e9024672083d5883b525c031eea940e556" + } +} diff --git a/main.ts b/main.ts new file mode 100644 index 0000000..008e12a --- /dev/null +++ b/main.ts @@ -0,0 +1,19 @@ +import { Application, Router } from "https://deno.land/x/oak@v12.6.1/mod.ts"; + +const port = 8000; + +const app = new Application(); + +const router = new Router(); + +app.use((ctx) => { + return ctx.send({ + root: "public", + index: "index.html", + }); +}); + +app.use(router.routes()); + +console.log(`Listening on port ${port}`); +await app.listen({ port }); diff --git a/mockdb.ts b/mockdb.ts new file mode 100644 index 0000000..2a79f0c --- /dev/null +++ b/mockdb.ts @@ -0,0 +1,75 @@ +// deno-lint-ignore-file require-await +import { Database, Session, User } from "./database.ts"; +import { Err, None, Ok, Option, Result, Some } from "./utils.ts"; + +export class MockDb implements Database { + public inErronousState = false; + public users: User[] = []; + public sessions: Session[] = []; + + public async createUser( + init: Omit, + ): Promise> { + if (this.inErronousState) return Err("error"); + const user: User = { + ...init, + id: this.users.length, + }; + this.users.push(user); + return Ok(user); + } + + public async userWithId( + id: number, + ): Promise, string>> { + if (this.inErronousState) return Err("error"); + const found = this.users.find((user) => user.id === id); + if (found === undefined) { + return Ok(None); + } + return Ok(Some(found)); + } + + public async userWithUsername( + username: string, + ): Promise, string>> { + if (this.inErronousState) return Err("error"); + const found = this.users.find((user) => user.username === username); + if (found === undefined) { + return Ok(None); + } + return Ok(Some(found)); + } + + public async userWithUsernameExists( + username: string, + ): Promise> { + if (this.inErronousState) return Err("error"); + return Ok( + this.users.find((user) => user.username === username) !== undefined, + ); + } + + public async createSession( + init: Omit, + ): Promise> { + if (this.inErronousState) return Err("error"); + const session: Session = { + ...init, + id: this.users.length, + }; + this.sessions.push(session); + return Ok(session); + } + + public async sessionWithId( + id: number, + ): Promise, string>> { + if (this.inErronousState) return Err("error"); + const found = this.sessions.find((session) => session.id === id); + if (found === undefined) { + return Ok(None); + } + return Ok(Some(found)); + } +} diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..a9ebba4c1f73d2af822b7f50a1a8c83e67292d78 GIT binary patch literal 11454 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 Fe*su&e&PTC literal 0 HcmV?d00001 diff --git a/public/index.html b/public/index.html new file mode 100644 index 0000000..9c8a078 --- /dev/null +++ b/public/index.html @@ -0,0 +1,9 @@ + + + + + + +

grill blog

+ + diff --git a/sessions.ts b/sessions.ts new file mode 100644 index 0000000..e7cc23e --- /dev/null +++ b/sessions.ts @@ -0,0 +1,47 @@ +import { BcryptCrypto, Crypto } from "./crypto.ts"; +import { Database } from "./database.ts"; +import { Err, Ok, Result } from "./utils.ts"; + +export type LoginReq = { + username: string; + password: string; +}; + +export type LoginRes = { + sessionId: number; +}; + +export type LoginError = + | "internal" + | "wrong username/password" + | "account banned"; + +export class Sessions { + public constructor( + private db: Database, + private crypto: Crypto = new BcryptCrypto(), + ) {} + + public async login(req: LoginReq): Promise> { + const findUserResult = await this.db.userWithUsername(req.username); + if (!findUserResult.ok) { + return Err("internal"); + } + if (!findUserResult.value.some) { + return Err("wrong username/password"); + } + const user = findUserResult.value.value; + if (!await this.crypto.compare(req.password, user.passwordHash)) { + return Err("wrong username/password"); + } + const session = await this.db.createSession({ + userId: user.id, + }); + if (!session.ok) { + return Err("internal"); + } + return Ok({ + sessionId: session.value.id, + }); + } +} diff --git a/sessions_test.ts b/sessions_test.ts new file mode 100644 index 0000000..7423667 --- /dev/null +++ b/sessions_test.ts @@ -0,0 +1,85 @@ +import { assertEquals } from "https://deno.land/std@0.224.0/assert/assert_equals.ts"; +import { MockCrypto } from "./crypto.ts"; +import { MockDb } from "./mockdb.ts"; +import { LoginReq, Sessions } from "./sessions.ts"; +import { Users } from "./users.ts"; + +const loginReq: LoginReq = { + username: "testuser", + password: "1234", +}; + +Deno.test("login: unknown user", async () => { + const db = new MockDb(); + const sessions = new Sessions(db, new MockCrypto()); + const result = await sessions.login(loginReq); + assertEquals(result.ok, false); + if (result.ok) return; + assertEquals(result.error, "wrong username/password"); +}); + +Deno.test("login: wrong username", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + const sessions = new Sessions(db, new MockCrypto()); + { + const result = await users.createUser({ + username: "testuser", + password: "1234", + }); + assertEquals(result.ok, true); + } + { + const result = await sessions.login({ + ...loginReq, + username: "wrong", + }); + assertEquals(result.ok, false); + if (result.ok) return; + assertEquals(result.error, "wrong username/password"); + } +}); + +Deno.test("login: wrong password", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + const sessions = new Sessions(db, new MockCrypto()); + { + const result = await users.createUser({ + username: "testuser", + password: "1234", + }); + assertEquals(result.ok, true); + } + { + const result = await sessions.login({ + ...loginReq, + password: "wrong", + }); + assertEquals(result.ok, false); + if (result.ok) return; + assertEquals(result.error, "wrong username/password"); + } +}); + +Deno.test("login: add session to database", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + const sessions = new Sessions(db, new MockCrypto()); + { + const result = await users.createUser({ + username: "testuser", + password: "1234", + }); + assertEquals(result.ok, true); + } + { + const result = await sessions.login(loginReq); + assertEquals(result.ok, true); + if (!result.ok) return; + const sessionResult = await db.sessionWithId(result.value.sessionId); + assertEquals(sessionResult.ok, true); + if (!sessionResult.ok) return; + assertEquals(sessionResult.value.some, true); + } +}); diff --git a/users.ts b/users.ts new file mode 100644 index 0000000..dc3b7a2 --- /dev/null +++ b/users.ts @@ -0,0 +1,75 @@ +import { BcryptCrypto, Crypto } from "./crypto.ts"; +import { Database } from "./database.ts"; +import { _, Err, Ok, Result } from "./utils.ts"; + +export type CreateUserReq = { + username: string; + password: string; +}; + +export type CreateUserError = + | "internal" + | "username empty" + | "password empty" + | "username taken"; + +export type UserInfoRes = { + id: number; + username: string; +}; +export type UserInfoError = + | "internal" + | "not found"; + +export class Users { + public constructor( + private db: Database, + private crypto: Crypto = new BcryptCrypto(), + ) {} + + public async createUser( + req: CreateUserReq, + ): Promise> { + if (req.username === "") { + return Err("username empty"); + } + if (req.password === "") { + return Err("password empty"); + } + const userExistsResult = await this.db.userWithUsernameExists( + req.username, + ); + if (!userExistsResult.ok) { + return Err("internal"); + } + if (userExistsResult.value) { + return Err("username taken"); + } + const passwordHash = await this.crypto.hash(req.password); + const createResult = await this.db.createUser({ + username: req.username, + passwordHash, + }); + if (!createResult.ok) { + return Err("internal"); + } + return Ok(_); + } + + public async userInfo( + id: number, + ): Promise> { + const queryResult = await this.db.userWithId(id); + if (!queryResult.ok) { + return Err("internal"); + } + if (!queryResult.value.some) { + return Err("not found"); + } + const user = queryResult.value.value; + return Ok({ + id: user.id, + username: user.username, + }); + } +} diff --git a/users_test.ts b/users_test.ts new file mode 100644 index 0000000..990c1f2 --- /dev/null +++ b/users_test.ts @@ -0,0 +1,132 @@ +import { assertEquals } from "https://deno.land/std@0.224.0/assert/mod.ts"; +import { CreateUserReq, Users } from "./users.ts"; +import { MockDb } from "./mockdb.ts"; +import { assertNotEquals } from "https://deno.land/std@0.224.0/assert/assert_not_equals.ts"; +import { MockCrypto } from "./crypto.ts"; + +const createUserReq: CreateUserReq = { + username: "testuser", + password: "1234", +}; + +Deno.test("createUser: no username", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + const result = await users.createUser({ + ...createUserReq, + username: "", + }); + assertEquals(result.ok, false); + if (result.ok) return; + assertEquals(result.error, "username empty"); +}); + +Deno.test("createUser: no password", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + const result = await users.createUser({ + ...createUserReq, + password: "", + }); + assertEquals(result.ok, false); + if (result.ok) return; + assertEquals(result.error, "password empty"); +}); + +Deno.test("createUser: add to database", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + { + const result = await db.userWithUsername(createUserReq.username); + assertEquals(result.ok, true); + if (!result.ok) return; + assertEquals(result.value.some, false); + } + const createResult = await users.createUser(createUserReq); + assertEquals(createResult.ok, true); + { + const result = await db.userWithUsername(createUserReq.username); + assertEquals(result.ok, true); + if (!result.ok) return; + assertEquals(result.value.some, true); + } +}); + +Deno.test("createUser: taken username", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + { + const result = await users.createUser(createUserReq); + assertEquals(result.ok, true); + } + { + const result = await users.createUser(createUserReq); + assertEquals(result.ok, false); + if (result.ok) return; + assertEquals(result.error, "username taken"); + } +}); + +Deno.test("createUser: add username to database", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + const createResult = await users.createUser(createUserReq); + assertEquals(createResult.ok, true); + const result = await db.userWithUsername(createUserReq.username); + assertEquals(result.ok, true); + if (!result.ok) return; + assertEquals(result.value.some, true); + if (!result.value.some) return; + assertEquals(result.value.value.username, createUserReq.username); +}); + +Deno.test("createUser: add password hash to database", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + const createResult = await users.createUser(createUserReq); + assertEquals(createResult.ok, true); + const result = await db.userWithUsername(createUserReq.username); + assertEquals(result.ok, true); + if (!result.ok) return; + assertEquals(result.value.some, true); + if (!result.value.some) return; + assertNotEquals(result.value.value.passwordHash, ""); + assertNotEquals(result.value.value.passwordHash, createUserReq.password); +}); + +Deno.test("createUser: database error -> internal error", async () => { + const db = new MockDb(); + db.inErronousState = true; + const users = new Users(db, new MockCrypto()); + const result = await users.createUser(createUserReq); + assertEquals(result.ok, false); + if (result.ok) return; + assertEquals(result.error, "internal"); +}); + +Deno.test("userInfo: not found", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + const result = await users.userInfo(0); + assertEquals(result.ok, false); + if (result.ok) return; + assertEquals(result.error, "not found"); +}); + +Deno.test("userInfo: find created", async () => { + const db = new MockDb(); + const users = new Users(db, new MockCrypto()); + { + const result = await users.createUser(createUserReq); + assertEquals(result.ok, true); + } + { + const result = await users.userInfo(0); + assertEquals(result.ok, true); + if (!result.ok) return; + assertEquals(result.value, { + id: 0, + username: createUserReq.username, + }); + } +}); diff --git a/utils.ts b/utils.ts new file mode 100644 index 0000000..e8d7ce4 --- /dev/null +++ b/utils.ts @@ -0,0 +1,16 @@ +export type Ok = { ok: true; value: T }; +export type Err = { ok: false; error: E }; +export type Result = Ok | Err; + +export const Ok = (value: T): Ok => ({ ok: true, value }); +export const Err = (error: E): Err => ({ ok: false, error }); + +export type Some = { some: true; value: T }; +export type None = { some: false }; +export type Option = Some | None; + +export const Some = (value: T): Some => ({ some: true, value }); +export const None: None = { some: false } as const; + +export const _ = Symbol("_"); +export type _ = typeof _;