diff --git a/README.md b/README.md index d3e6241..1f22cae 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -# kickstart-for-time-pwa +# TimeSafari.app - Crowd-Funder for Time - PWA ## Project setup @@ -13,6 +13,11 @@ npm install npm run serve ``` +### Lints and fixes files +``` +npm run lint +``` + ### Compiles and minifies for production If you are deploying in a subdirectory, add it to `publicPath` in vue.config.js, eg: `publicPath: "/app/time-tracker/",` @@ -21,41 +26,19 @@ If you are deploying in a subdirectory, add it to `publicPath` in vue.config.js, npm run build ``` -### Lints and fixes files -``` -npm run lint -``` - -## Tests +... then copy the contents of the `sw_scripts` folder to the `dist` folder - except additional_scripts.js. -### -For your own web-push tests, change the 'vapid' URL in App.vue, and install apps on the same domain. -### Test key contents - -See [this page](openssl_signing_console.rst) +## Tests ### Register new user on test server -New users require registration. This can be done with a claim payload like this -by an existing user: - -``` - const vcClaim = { - "@context": "https://schema.org", - "@type": "RegisterAction", - agent: { identifier: identity0.did }, - object: SERVICE_ID, - participant: { identifier: newIdentity.did }, - }; -``` - On the test server, User #0 has rights to register others, so you can start playing one of two ways: -- Import the keys for the test User `did:ethr:0x000Ee5654b9742f6Fe18ea970e32b97ee2247B51` by importing this seed phrase: - `seminar accuse mystery assist delay law thing deal image undo guard initial shallow wrestle list fragile borrow velvet tomorrow awake explain test offer control` +- Import the keys for the test User `did:ethr:0x0000694B58C2cC69658993A90D3840C560f2F51F` by importing this seed phrase: + `rigid shrug mobile smart veteran half all pond toilet brave review universe ship congress found yard skate elite apology jar uniform subway slender luggage` (Other test users are found [here](https://github.com/trentlarson/endorser-ch/blob/master/test/util.js).) - Alternatively, register someone else under User #0 automatically: @@ -66,14 +49,35 @@ playing one of two ways: ### Create multiple identifiers -Go to /start and create or import a new one. Then switch identifiers on the bottom of the Your Identity page. +Under the "Your Identity" screen, click "Advanced", click "Switch Identity / No Identity", then "Add Another Identity...". ### Create keys with alternate tools -See [this page](openssl_signing_console.rst) +[This page](openssl_signing_console.rst) is a tool to create a JWT from a locally-generated keypair. + +### Web-push + +For your own web-push tests, change the 'vapid' URL in App.vue, and install apps on the same domain. + +### Manual walk-through + +- Clear the browser cache for localhost for a new user. + - See that it's using the test API. + - On each page, verify the messaging. + - On the home page, see the feed without names, and see a message prompting to generate an ID. + - On the discovery page, check that they can see projects, and set a search area to see projects nearby. +- As User #0 in another browser on the test API, add a give & a project. (See User #0 details above.) + - With the new user on the home page, see the feed that shows User #0 in network but without the name. +- As the new user on the contacts page, add User #0 as a contact. + - On the home page, see the feed that shows User #0 with a name. +- Generate an ID. + - On the home page, check that it now prompts them to get registered. + - On the account page, check that they see messages on limits. +- Register the ID from User #0. + - As the new user on the home page, check that they can now record a gift. + - On the contacts page, check that they cannot register someone else yet. +- Walk through the functions on each page. -### Customize Vue configuration -See [Configuration Reference](https://cli.vuejs.org/config/). ## Scenarios @@ -88,12 +92,13 @@ See [Configuration Reference](https://cli.vuejs.org/config/). - Click on the "Registration Unknown" button and register that person to be able to make claims as them. -### Clear data & restart +### Clear/Reset data & restart -* Clear cache for localhost, then go to http://localhost:8080/start - (because it'll generate a new one automatically if you start on the `/account` page). +* Clear cache for localhost. * Unregister service worker (in Chrome, go to `chrome://serviceworker-internals/`; in Firefox, go to `about:serviceworkers` or `about:debugging`). -* Clear notifications (in Chrome, go to `chrome://settings/content/notifications`; in Firefox, go to `about:preferences` and search). +* Clear notification permission (in Chrome, go to `chrome://settings/content/notifications`; in Firefox, go to `about:preferences` and search). + + ## Other @@ -103,110 +108,10 @@ See [Configuration Reference](https://cli.vuejs.org/config/). * Notifications can be type of `toast` (self-dismiss), `info`, `success`, `warning`, and `danger`. They are done via [notiwind](https://www.npmjs.com/package/notiwind) and set up in App.vue. -``` -// reference material from https://github.com/trentlarson/endorser-mobile/blob/8dc8e0353e0cc80ffa7ed89ded15c8b0da92726b/src/utility/idUtility.ts#L83 - -// Import an existing ID -export const importAndStoreIdentifier = async (mnemonic: string, mnemonicPassword: string, toLowercase: boolean, previousIdentifiers: Array) => { - - // just to get rid of variability that might cause an error - mnemonic = mnemonic.trim().toLowerCase() - - /** - // an approach I pieced together - // requires: yarn add elliptic - // ... plus: - // const EC = require('elliptic').ec - // const secp256k1 = new EC('secp256k1') - // - const keyHex: string = bip39.mnemonicToEntropy(mnemonic) - // returns a KeyPair from the elliptic.ec library - const keyPair = secp256k1.keyFromPrivate(keyHex, 'hex') - // this code is from did-provider-eth createIdentifier - const privateHex = keyPair.getPrivate('hex') - const publicHex = keyPair.getPublic('hex') - const address = didJwt.toEthereumAddress(publicHex) - **/ - - /** - // from https://github.com/uport-project/veramo/discussions/346#discussioncomment-302234 - // ... which almost works but the didJwt.toEthereumAddress is wrong - // requires: yarn add bip32 - // ... plus: import * as bip32 from 'bip32' - // - const seed: Buffer = await bip39.mnemonicToSeed(mnemonic) - const root = bip32.fromSeed(seed) - const node = root.derivePath(UPORT_ROOT_DERIVATION_PATH) - const privateHex = node.privateKey.toString("hex") - const publicHex = node.publicKey.toString("hex") - const address = didJwt.toEthereumAddress('0x' + publicHex) - **/ - - /** - // from https://github.com/uport-project/veramo/discussions/346#discussioncomment-302234 - // requires: yarn add @ethersproject/hdnode - // ... plus: import { HDNode } from '@ethersproject/hdnode' - **/ - const hdnode: HDNode = HDNode.fromMnemonic(mnemonic) - const rootNode: HDNode = hdnode.derivePath(UPORT_ROOT_DERIVATION_PATH) - const privateHex = rootNode.privateKey.substring(2) // original starts with '0x' - const publicHex = rootNode.publicKey.substring(2) // original starts with '0x' - let address = rootNode.address - - const prevIds = previousIdentifiers || []; - - if (toLowercase) { - const foundEqual = R.find( - (id) => utility.rawAddressOfDid(id.did) === address, - prevIds - ) - if (foundEqual) { - // They're trying to create a lowercase version of one that exists in normal case. - // (We really should notify the user.) - appStore.dispatch(appSlice.actions.addLog({log: true, msg: "Will create a normal-case version of the DID since a regular version exists."})) - } else { - address = address.toLowerCase() - } - } else { - // They're not trying to convert to lowercase. - const foundLower = R.find((id) => - utility.rawAddressOfDid(id.did) === address.toLowerCase(), - prevIds - ) - if (foundLower) { - // They're trying to create a normal case version of one that exists in lowercase. - // (We really should notify the user.) - appStore.dispatch(appSlice.actions.addLog({log: true, msg: "Will create a lowercase version of the DID since a lowercase version exists."})) - address = address.toLowerCase() - } - } - - appStore.dispatch(appSlice.actions.addLog({log: false, msg: "... derived keys and address..."})) - - const newId = newIdentifier(address, publicHex, privateHex, UPORT_ROOT_DERIVATION_PATH) - appStore.dispatch(appSlice.actions.addLog({log: false, msg: "... created new ID..."})) - - // awaiting because otherwise the UI may not see that a mnemonic was created - const savedId = await storeIdentifier(newId, mnemonic, mnemonicPassword) - appStore.dispatch(appSlice.actions.addLog({log: false, msg: "... stored new ID..."})) - return savedId -} - -// Create a totally new ID -export const createAndStoreIdentifier = async (mnemonicPassword) => { - - // This doesn't give us the entropy/seed. - //const id = await agent.didManagerCreate() - - const entropy = crypto.randomBytes(32) - const mnemonic = bip39.entropyToMnemonic(entropy) - appStore.dispatch(appSlice.actions.addLog({log: false, msg: "... generated mnemonic..."})) - - return importAndStoreIdentifier(mnemonic, mnemonicPassword, false, []) -} -``` +* [Customize Vue configuration](https://cli.vuejs.org/config/). + -## Kudos +### Kudos Gifts make the world go 'round! diff --git a/openssl_signing_console.rst b/openssl_signing_console.rst index 28a272a..8b7befd 100644 --- a/openssl_signing_console.rst +++ b/openssl_signing_console.rst @@ -1,8 +1,11 @@ -Prerequisites: +JWT Creation & Verification -jq +To run this in a script, see ./openssl_signing_console.sh -You can create a JWT using a library or by encoding the header and payload base64Url and signing it with a secret using a ES256K algorithm. Here is an example of how you can create a JWT using the jq and openssl command line utilities: +Prerequisites: openssl, jq + +You can create a JWT using a library or by encoding the header and payload base64Url and signing it with a secret using +a ES256K algorithm. Here is an example of how you can create a JWT using the jq and openssl command line utilities: Here is an example of how you can use openssl to sign a JWT with the ES256K algorithm: @@ -15,20 +18,22 @@ openssl ec -in private.pem -pubout -out public.pem header='{"alg":"ES256K", "issuer": "", "typ":"JWT"}' - Next, create a payload object as a JSON object containing the claims you want to include in the JWT. For example schema.org : + Next, create a payload object as a JSON object containing the claims you want to include in the JWT. + For example schema.org : payload='{"@context": "http://schema.org", "@type": "PlanAction", "identifier": "did:ethr:0xb86913f83A867b5Ef04902419614A6FF67466c12", "name": "Test", "description": "Me"}' Encode the header and payload objects as base64Url strings. You can use the jq command line utility to do this: -header_b64=$(echo -n "$header" | jq -c -M . | tr -d '\n') -payload_b64=$(echo -n "$payload" | jq -c -M . | tr -d '\n') +header_b64=$(echo -n "$header" | jq -c -M . | tr -d '\n' | base64 | tr -d '=' | tr '+' '-' | tr '/' '_') +payload_b64=$(echo -n "$payload" | jq -c -M . | tr -d '\n' | base64 | tr -d '=' | tr '+' '-' | tr '/' '_') Concatenate the encoded header, payload, and a secret to create the signing input: signing_input="$header_b64.$payload_b64" - Create the signature by signing the signing input with a ES256K algorithm and your secret. You can use the openssl command line utility to do this: + Create the signature by signing the signing input with a ES256K algorithm and your secret. + You can use the openssl command line utility to do this: signature=$(echo -n "$signing_input" | openssl dgst -sha256 -sign private.pem) @@ -43,7 +48,7 @@ Authorization: Bearer $jwt To verify the JWT, you can use the openssl utility with the public key: -openssl dgst -sha256 -verify public.pem -signature <(echo -n "$signature") "$signing_input" - - This will verify the signature and output Verified OK if the signature is valid. If the signature is not valid, it will output an error. +echo -n "$signing_input" | openssl dgst -sha256 -verify public.pem -signature <(echo -n "$signature") + This will verify the signature and output "Verified OK" if the signature is valid. + If the signature is not valid, it will give an error response and output "Verification failure". diff --git a/openssl_signing_console.sh b/openssl_signing_console.sh index acdda68..b459dad 100755 --- a/openssl_signing_console.sh +++ b/openssl_signing_console.sh @@ -1,25 +1,39 @@ #!/bin/bash +# Generate a JWT, with signature verified using OpenSSL +# +# Prerequisites: openssl, jq +# +# Usage: source ./openssl_signing_console.sh +# +# For a more complete explanation, see ./openssl_signing_console.rst + + +# Generate a key and extract the public part openssl ecparam -name secp256k1 -genkey -noout -out private.pem openssl ec -in private.pem -pubout -out public.pem +# Use test data header='{"alg":"ES256K", "issuer": "", "typ":"JWT"}' - payload='{"@context": "http://schema.org", "@type": "PlanAction", "identifier": "did:ethr:0xb86913f83A867b5Ef04902419614A6FF67466c12", "name": "Test", "description": "Me"}' -header_b64=$(echo -n "$header" | jq -c -M . | tr -d '\n') -payload_b64=$(echo -n "$payload" | jq -c -M . | tr -d '\n') +header_b64=$(echo -n "$header" | jq -c -M . | tr -d '\n' | base64 | tr -d '=' | tr '+' '-' | tr '/' '_') +payload_b64=$(echo -n "$payload" | jq -c -M . | tr -d '\n' | base64 | tr -d '=' | tr '+' '-' | tr '/' '_') signing_input="$header_b64.$payload_b64" -echo -n "$signing_input" | openssl dgst -sha256 -sign private.pem -out signature.bin +signature=$(echo -n "$signing_input" | openssl dgst -sha256 -sign private.pem | openssl base64 -e) -# Read binary signature from file and encode it to Base64 URL-Safe format -signature_b64=$(base64 -w 0 < signature.bin | tr -d '=' | tr '+' '-' | tr '/' '_') +echo -n "$signing_input" | openssl dgst -sha256 -verify public.pem -signature <(echo -n "$signature" | openssl base64 -d) -# Construct the JWT -jwt="$signing_input.$signature_b64" -openssl dgst -sha256 -verify public.pem -signature signature.bin -out verified.txt <(echo -n "$signing_input") + +# Read binary signature and encode it to Base64 URL-Safe format +signature_b64=$(echo -n "$signature" | base64 | tr -d '=' | tr '+' '-' | tr '/' '_') + +# Construct the JWT +jwt="$signing_input.$signature_b64" + +echo Resulting JWT: $jwt diff --git a/package-lock.json b/package-lock.json index 1745080..75f460e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { - "name": "kickstart-for-time-pwa", - "version": "0.1.3", + "name": "crowd-funder-for-time-pwa", + "version": "0.1.4", "lockfileVersion": 3, "requires": true, "packages": { "": { - "name": "kickstart-for-time-pwa", - "version": "0.1.3", + "name": "crowd-funder-for-time-pwa", + "version": "0.1.4", "dependencies": { "@ethersproject/hdnode": "^5.7.0", "@fortawesome/fontawesome-svg-core": "^6.4.2", diff --git a/package.json b/package.json index c96b1bf..0abdcc4 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { - "name": "kickstart-for-time-pwa", - "version": "0.1.3", + "name": "crowd-funder-for-time-pwa", + "version": "0.1.4", "private": true, "scripts": { "serve": "vue-cli-service serve", @@ -72,13 +72,13 @@ "@vue/cli-service": "~5.0.8", "@vue/eslint-config-typescript": "^11.0.3", "autoprefixer": "^10.4.15", - "eslint": "^8.48.0", + "eslint": "^8.53.0", "eslint-config-prettier": "^9.0.0", "eslint-plugin-prettier": "^5.0.0", "eslint-plugin-vue": "^9.17.0", "leaflet": "^1.9.4", "postcss": "^8.4.29", - "prettier": "^3.0.3", + "prettier": "^3.1.0", "tailwindcss": "^3.3.3", "typescript": "~5.2.2" } diff --git a/project.task.yaml b/project.task.yaml index af1c65b..9b56584 100644 --- a/project.task.yaml +++ b/project.task.yaml @@ -1,55 +1,45 @@ tasks: +- remove hard-coded anomalistlabs.com + +- don't show "Give" & "Offer" on project screen if they don't have an identifier +- allow some gives even if they aren't registered + - in endorser-push-server - mount folder for persistent sqlite DB outside of container +- extract private_key_hex in webpush.py - 40 notifications : - push, where we trigger a ServiceWorker(?) in the app to reach out and check for new data assignee:matthew -- 01 Replace Gifted/Give in ContactsView with GiftedDialog assignee:matthew - -- 01 fix the Discovery map display to not show on top of bottom icons (and any other UI tweaks on the map flow) assignee-group:ui -- .1 add instructions for map location selection - -- 01 Show pop-up or some message confirming that settings & contacts download has been initiated/finished - +- .2 change the "claims" verbiage in feeds (eg. safari-notifications.js) +- .5 allow to manage their notifications even without an identity - 01 Ensure each action sent to the server has a confirmation - eg registration (ie a toast something that dismisses after 5-10s) +- .3 fix the Project-location-selection map display to not show on top of bottom icons (and any other UI tweaks on the map flow) assignee-group:ui -- Home Feed & Quick Give screen : - - 01 save the feed-viewed status in settings storage ("afterQuery") - - 01 quick action - send action, maybe choose via canvas tool - - SEE: https://github.com/konvajs/vue-konva - -- 24 Move to Vite assignee:matthew - -- .5 switch so DiscoverView shows anywhere by default, and no number unless search is done (and maybe a better filter UI, including "mine" to consolidate with ProjectsView) -- .2 fit as many icons as possible on home & project view screens but only going halfway down the page assignee-group:ui - .5 Add infinite scroll to gifts on the home page - .5 bug - search for "Safari" does not find the project, but if already on the "Anywhere" tab it shows all - .2 figure out why endorser-mobile search doesn't find recently created PlanAction - .1 when creating a plan, select location and then make sure you can deselect on Android -- .5 add link to further project / people when a project pays ahead -- .5 add project ID to the URL of the project-view, to make a project publicly-accessible - .5 fix where user 0 sees no txns from user 1 on contacts page but sees them on list page -- .2 on ProjectViewView, show different messages for "to" and "from" sections if none exist -- .2 fix rate limit verbiage (with the new one-per-day allowance) assignee:trent -- .1 remove the logic to exclude beforeId in list of plans after server has commit 26b25af605e715600d4f12b6416ed9fd7142d164 +- .1 remove the logic to exclude beforeId in list of plans after server has commit 26b25af605e715600d4f12b6416ed9fd7142d164 assignee:trent - .2 in SeedBackupView, don't load the mnemonic and keep it in memory; only load it when they click "show" -- .1 Make give description text box into something that expands as they type -- .1 Make contact info specific to Time Safari - rather pointing at CommunityCred.org +- fix cert generation (since it didn't happen automatically for Nov 30) - Discuss whether the remaining tasks are worthwhile before MVP release. +- .1 Make give description text box into something that expands as they type? - 04 allow user to download claims, mine + ones I can see about me from others -- 02 allow user to create new DIDs from the same seed phrase (ie. increment derivation path) -- .5 on ProjectView page, show immediate feedback when a gift is given (on list?) -- and consider the same for Home & Contacts pages - .5 customize favicon assignee-group:ui - .2 Show a warning if both giver and recipient are the same (but still allow?) - 01 Would it look better to shrink the buttons on many pages so they don't expand to the width of the screen? assignee-group:ui - .5 Display a more appealing confirmation on the map when erasing the marker -- .5 make a VC details page +- .5 make a VC details page, or link to endorser.ch - .1 Add units or different icon to the coins (to distinguish $, BTC, hours, etc) -- .5 include the hash of the latest commit on help page next to version +- .5 include the hash of the latest commit on help page next to version (maybe Trent's git-hash branch) - .5 remove references to localStorage for projectId (now that it's pulling from the path) +- bug (that is hard to reproduce) - on the second 'give' recorded on prod it showed me as the agent +- make identicons for contacts into more-memorable faces (and maybe change project identicons, too) +- allow download of each VC (to show that they can actually own their data) - contacts v+ : - 01 Import all the non-sensitive data (ie. contacts & settings). @@ -58,14 +48,17 @@ tasks: - stats v1 : - 01 show numeric stats - - 04 show different graphic for projects vs people on world + - 04 show different graphic for projects vs people (gnome?) on world - 01 link to world for specific stats - .5 don't load another instance of a bush if it already exists - maybe - allow type annotations in World.js & landmarks.js (since we get this error - "Types are not supported by current JavaScript version") - 08 convert to cleaner implementation (maybe Drie -- https://github.com/janvorisek/drie) - Release Minimum Viable Product : + - generate new webpush.db entries, data/webpush.db private_key_hex & subscription_info & vapid_claims email + - .5 deploy endorser.ch server above Dec 1 (to get plan searches by names as well as descriptions) - 08 thorough testing for errors & edge cases + - 01 ensure ability to recover server remotely, and add redundant access - Turn off stats-world or ensure it's usable (eg. cannot zoom out too far and lose world, cannot screenshot). - Add disclaimers. - Switch default server to the public server. @@ -75,7 +68,9 @@ tasks: blocks: ref:https://raw.githubusercontent.com/trentlarson/lives-of-gifts/master/project.yaml#kickstarter%20for%20time - .5 show seed phrase in a QR code for transfer to another device +- .5 on DiscoverView, switch to a filter UI (eg. just from friend +- 24 Move to Vite - 32 accept images for projects - 32 accept images for contacts @@ -85,6 +80,10 @@ tasks: - for subtasks: fulfills (is it really the same?), feeds, contributes to, supplies, boosts, advances - for blocking: blocks, precedes, comes before, is sought by -- vs follows, seeks, builds on ("contributes to" isn't specific enough, "succeeds" has different, possibly confusing meaning) +- .5 add "back" button to all screens that aren't part of the bottom tray +- .5 fit as many icons as possible on home & project view screens but only going halfway down the page assignee-group:ui +- .5 Replace Gifted/Give in ContactsView with GiftedDialog + - Stats : - 01 point out user's location on the world - 01 present a credential selected from the stats @@ -101,11 +100,10 @@ tasks: - Multiple identities -- Peer DID - -- DIDComm - -- Write to or read from a different ledger (eg. private ACDC, attest.sh) +- Support KERI AIDs +- Support Peer DIDs +- Support messaging through DIDComm +- Write to or read from a different ledger (eg. private ACDC, EAS & attest.sh) - Do we want split first name & last name? @@ -113,6 +111,7 @@ tasks: - pull, w/ scheduled runs - 01 On nearby search, if user starts changing their box but cancels and goes back to the map it is zoomed far out. Fix to fit the box better. +- 16 From the home screen, make the quick action even easier. log: - videos for multiple identities https://youtu.be/p8L87AeD76w and for adding time to contacts https://youtu.be/7Yylczevp10 done:2023-03-29 diff --git a/sample.txt b/sample.txt deleted file mode 100644 index efe20be..0000000 --- a/sample.txt +++ /dev/null @@ -1,177 +0,0 @@ - -> kickstart-for-time-pwa@0.1.0 build -> vue-cli-service build - -All browser targets in the browserslist configuration have supported ES module. -Therefore we don't build two separate bundles for differential loading. - - - WARNING Compiled with 5 warnings6:06:43 PM - -[eslint] -/home/matthew/projects/kick-starter-for-time-pwa/src/components/World/components/objects/landmarks.js - 98:11 warning Unexpected console statement no-console - 133:7 warning Unexpected console statement no-console - 144:5 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/router/index.ts - 210:3 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/AccountViewView.vue - 362:7 warning Unexpected console statement no-console - 375:7 warning Unexpected console statement no-console - 404:7 warning Unexpected console statement no-console - 516:7 warning Unexpected console statement no-console - 536:7 warning Unexpected console statement no-console - 630:5 warning Unexpected console statement no-console - 682:7 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/ContactAmountsView.vue - 206:9 warning Unexpected console statement no-console - 233:9 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/ContactGiftingView.vue - 244:9 warning Unexpected console statement no-console - 267:7 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/ContactsView.vue - 340:9 warning Unexpected console statement no-console - 577:9 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/DiscoverView.vue - 315:9 warning Unexpected console statement no-console - 343:7 warning Unexpected console statement no-console - 390:9 warning Unexpected console statement no-console - 423:7 warning Unexpected console statement no-console - 532:9 warning Unexpected console statement no-console - 575:7 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/HomeView.vue - 349:9 warning Unexpected console statement no-console - 498:9 warning Unexpected console statement no-console - 521:7 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/IdentitySwitcherView.vue - 142:7 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/ImportAccountView.vue - 123:9 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/ImportDerivedAccountView.vue - 159:7 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/NewEditProjectView.vue - 183:9 warning Unexpected console statement no-console - 215:7 warning Unexpected console statement no-console - 297:13 warning Unexpected console statement no-console - 320:11 warning Unexpected console statement no-console - 345:7 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/ProjectViewView.vue - 387:9 warning Unexpected console statement no-console - 421:7 warning Unexpected console statement no-console - 457:7 warning Unexpected console statement no-console - 552:9 warning Unexpected console statement no-console - 554:11 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/ProjectsView.vue - 131:9 warning Unexpected console statement no-console - 144:7 warning Unexpected console statement no-console - 221:9 warning Unexpected console statement no-console - 237:7 warning Unexpected console statement no-console - -/home/matthew/projects/kick-starter-for-time-pwa/src/views/SeedBackupView.vue - 94:7 warning Unexpected console statement no-console - -✖ 44 problems (0 errors, 44 warnings) - - -You may use special comments to disable some warnings. -Use // eslint-disable-next-line to ignore the next line. -Use /* eslint-disable */ to ignore all warnings in a file. - warning - -/models/lupine_plant/textures/lambert2SG_baseColor.png is 3.75 MB, and won't be precached. Configure maximumFileSizeToCacheInBytes to change this limit. - - warning - -/models/lupine_plant/textures/lambert2SG_normal.png is 4.91 MB, and won't be precached. Configure maximumFileSizeToCacheInBytes to change this limit. - - warning - -asset size limit: The following asset(s) exceed the recommended size limit (244 KiB). -This can impact web performance. -Assets: - js/project.44f30c9f.js (318 KiB) - js/statistics.8a97010a.js (586 KiB) - js/chunk-vendors.a4845bfb.js (411 KiB) - js/705.f6a6ce2a.js (252 KiB) - img/textures/leafy-autumn-forest-floor.jpg (705 KiB) - models/lupine_plant/textures/lambert2SG_baseColor.png (3.58 MiB) - models/lupine_plant/textures/lambert2SG_normal.png (4.69 MiB) - - warning - -entrypoint size limit: The following entrypoint(s) combined asset size exceeds the recommended limit (244 KiB). This can impact web performance. -Entrypoints: - app (447 KiB) - js/chunk-vendors.a4845bfb.js - css/app.8f21529c.css - js/app.8833cebc.js - - - File Size Gzipped - - dist/js/statistics.8a97010a.js 585.72 KiB 148.80 KiB - dist/js/chunk-vendors.a4845bfb.js 411.44 KiB 137.82 KiB - dist/js/project.44f30c9f.js 317.61 KiB 78.67 KiB - dist/js/705.f6a6ce2a.js 251.66 KiB 87.12 KiB - dist/js/891.33615e4f.js 147.32 KiB 42.09 KiB - dist/js/153.e2c8e249.js 146.26 KiB 42.21 KiB - dist/js/820.13565d16.js 66.10 KiB 18.33 KiB - dist/js/contact-qr.e170ec33.js 54.85 KiB 15.63 KiB - dist/js/772.7b4c53a7.js 30.29 KiB 7.21 KiB - dist/js/361.898a4525.js 27.40 KiB 8.19 KiB - dist/js/account.77d86130.js 17.51 KiB 5.93 KiB - dist/js/app.8833cebc.js 17.31 KiB 5.84 KiB - dist/js/contacts.3fc90ff8.js 16.94 KiB 5.52 KiB - dist/js/discover.24106939.js 15.30 KiB 5.22 KiB - dist/js/536.3bb13201.js 15.23 KiB 4.84 KiB - dist/workbox-5b385ed2.js 14.11 KiB 4.93 KiB - dist/js/home.218b99dd.js 13.89 KiB 4.97 KiB - dist/js/help.50d3117b.js 12.49 KiB 4.38 KiB - dist/js/projects.417a6cb7.js 8.71 KiB 3.00 KiB - dist/js/contact-amounts.a32b0ccd.js 8.44 KiB 3.25 KiB - dist/js/229.120e09bf.js 7.99 KiB 2.72 KiB - dist/js/identity-switcher.c7937333.js 7.44 KiB 2.52 KiB - dist/js/new-edit-project.0552181b.js 7.36 KiB 3.11 KiB - dist/js/300.dcaeb2a3.js 6.56 KiB 3.24 KiB - dist/js/seed-backup.76a0f7b3.js 3.99 KiB 1.97 KiB - dist/js/import-derive.c688d4b8.js 3.81 KiB 1.82 KiB - dist/js/import-account.c3fa35fd.js 3.54 KiB 1.66 KiB - dist/js/new-edit-account.bb763be2.js 3.39 KiB 1.51 KiB - dist/js/431.5a6d64e0.js 3.38 KiB 2.56 KiB - dist/service-worker.js 3.37 KiB 1.38 KiB - dist/js/scan-contact.46be989a.js 2.79 KiB 1.18 KiB - dist/js/start.091a7740.js 2.70 KiB 1.30 KiB - dist/js/new-identifier.bb379420.js 2.12 KiB 1.18 KiB - dist/js/93.b873dbbf.js 2.08 KiB 1.61 KiB - dist/js/new-edit-commitment.9248d367.j 1.96 KiB 1.05 KiB - s - dist/js/confirm-contact.02004d1d.js 1.89 KiB 1.04 KiB - dist/js/858.ae4c08ec.js 0.97 KiB 0.78 KiB - dist/css/app.8f21529c.css 18.41 KiB 4.39 KiB - dist/css/discover.73ee9bd3.css 14.77 KiB 6.25 KiB - dist/css/new-edit-project.73ee9bd3.css 14.77 KiB 6.25 KiB - dist/css/contacts.abb5e493.css 0.40 KiB 0.23 KiB - dist/css/contact-amounts.5b26ccd4.css 0.31 KiB 0.20 KiB - dist/css/home.828bc66e.css 0.25 KiB 0.19 KiB - dist/css/project.828bc66e.css 0.25 KiB 0.19 KiB - dist/css/statistics.828bc66e.css 0.25 KiB 0.19 KiB - - Images and other types of assets omitted. - Build at: 2023-09-07T10:06:43.972Z - Hash: 2b39fcd4d0e78263 - Time: 32016ms - - DONE Build complete. The dist directory is ready to be deployed. - INFO Check out deployment instructions at https://cli.vuejs.org/guide/deployment.html - diff --git a/src/App.vue b/src/App.vue index 2486884..585cdcc 100644 --- a/src/App.vue +++ b/src/App.vue @@ -262,6 +262,28 @@ + + diff --git a/src/libs/endorserServer.ts b/src/libs/endorserServer.ts index 628d5ba..8db51eb 100644 --- a/src/libs/endorserServer.ts +++ b/src/libs/endorserServer.ts @@ -59,17 +59,40 @@ export interface GiveServerRecord { unit: string; } +export interface OfferServerRecord { + amount: number; + amountGiven: number; + offeredByDid: string; + recipientDid: string; + requirementsMet: boolean; + unit: string; + validThrough: string; +} + export interface GiveVerifiableCredential { "@context"?: string; // optional when embedded, eg. in an Agree - "@type": string; + "@type": "GiveAction"; agent?: { identifier: string }; description?: string; - fulfills?: { "@type": string; identifier: string }; + fulfills?: { "@type": string; identifier?: string; lastClaimId?: string }; identifier?: string; object?: { amountOfThisGood: number; unitCode: string }; recipient?: { identifier: string }; } +export interface OfferVerifiableCredential { + "@context"?: string; // optional when embedded, eg. in an Agree + "@type": "Offer"; + description?: string; + includesObject?: { amountOfThisGood: number; unitCode: string }; + itemOffered?: { + description?: string; + isPartOf?: { identifier?: string; lastClaimId?: string; "@type"?: string }; + }; + offeredBy?: { identifier: string }; + validThrough?: string; +} + export interface PlanVerifiableCredential { "@context": "https://schema.org"; "@type": "PlanAction"; @@ -152,7 +175,7 @@ export interface ErrorResult { error: InternalError; } -export type CreateAndSubmitGiveResult = SuccessResult | ErrorResult; +export type CreateAndSubmitClaimResult = SuccessResult | ErrorResult; /** * For result, see https://api.endorser.ch/api-docs/#/claims/post_api_v2_claim @@ -172,20 +195,81 @@ export async function createAndSubmitGive( description?: string, hours?: number, fulfillsProjectHandleId?: string, -): Promise { - try { - const vcClaim: GiveVerifiableCredential = { - "@context": "https://schema.org", - "@type": "GiveAction", - recipient: toDid ? { identifier: toDid } : undefined, - agent: fromDid ? { identifier: fromDid } : undefined, - description: description || undefined, - object: hours ? { amountOfThisGood: hours, unitCode: "HUR" } : undefined, - fulfills: fulfillsProjectHandleId - ? { "@type": "PlanAction", identifier: fulfillsProjectHandleId } - : undefined, +): Promise { + const vcClaim: GiveVerifiableCredential = { + "@context": "https://schema.org", + "@type": "GiveAction", + recipient: toDid ? { identifier: toDid } : undefined, + agent: fromDid ? { identifier: fromDid } : undefined, + description: description || undefined, + object: hours ? { amountOfThisGood: hours, unitCode: "HUR" } : undefined, + fulfills: fulfillsProjectHandleId + ? { "@type": "PlanAction", identifier: fulfillsProjectHandleId } + : undefined, + }; + return createAndSubmitClaim( + vcClaim as GenericClaim, + identity, + apiServer, + axios, + ); +} + +/** + * For result, see https://api.endorser.ch/api-docs/#/claims/post_api_v2_claim + * + * @param identity + * @param description may be null; should have this or hours + * @param hours may be null; should have this or description + * @param expirationDate ISO 8601 date string YYYY-MM-DD (may be null) + * @param fulfillsProjectHandleId ID of project to which this contributes (may be null) + */ +export async function createAndSubmitOffer( + axios: Axios, + apiServer: string, + identity: IIdentifier, + description?: string, + hours?: number, + expirationDate?: string, + fulfillsProjectHandleId?: string, +): Promise { + const vcClaim: OfferVerifiableCredential = { + "@context": "https://schema.org", + "@type": "Offer", + offeredBy: { identifier: identity.did }, + validThrough: expirationDate || undefined, + }; + if (hours) { + vcClaim.includesObject = { + amountOfThisGood: hours, + unitCode: "HUR", }; + } + if (description) { + vcClaim.itemOffered = { description }; + } + if (fulfillsProjectHandleId) { + vcClaim.itemOffered = vcClaim.itemOffered || {}; + vcClaim.itemOffered.isPartOf = { + "@type": "PlanAction", + identifier: fulfillsProjectHandleId, + }; + } + return createAndSubmitClaim( + vcClaim as GenericClaim, + identity, + apiServer, + axios, + ); +} +export async function createAndSubmitClaim( + vcClaim: GenericClaim, + identity: IIdentifier, + apiServer: string, + axios: Axios, +): Promise { + try { const vcPayload = { vc: { "@context": ["https://www.w3.org/2018/credentials/v1"], @@ -226,15 +310,11 @@ export async function createAndSubmitGive( }); return { type: "success", response }; - } catch (error: unknown) { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + } catch (error: any) { + console.log("Error creating claim:", error); const errorMessage: string = - error === null - ? "Null error" - : error instanceof Error - ? error.message - : typeof error === "object" && error !== null && "message" in error - ? (error as { message: string }).message - : "Unknown error"; + error.response?.data?.error?.message || error.message || "Unknown error"; return { type: "error", diff --git a/src/registerServiceWorker.ts b/src/registerServiceWorker.ts index 5b60bb7..22ca9b9 100644 --- a/src/registerServiceWorker.ts +++ b/src/registerServiceWorker.ts @@ -3,7 +3,7 @@ import { register } from "register-service-worker"; if (process.env.NODE_ENV === "production") { - register(`${process.env.BASE_URL}service-worker.js`, { + register("/additional-scripts.js", { ready() { console.log( "App is being served from cache by a service worker.\n" + diff --git a/src/router/index.ts b/src/router/index.ts index 8868f60..49b3376 100644 --- a/src/router/index.ts +++ b/src/router/index.ts @@ -168,6 +168,14 @@ const routes: Array = [ /* webpackChunkName: "scan-contact" */ "../views/ContactScanView.vue" ), }, + { + path: "/search-area", + name: "search-area", + component: () => + import( + /* webpackChunkName: "search-area" */ "../views/SearchAreaView.vue" + ), + }, { path: "/seed-backup", name: "seed-backup", diff --git a/src/views/AccountViewView.vue b/src/views/AccountViewView.vue index 08083fb..e33d0d2 100644 --- a/src/views/AccountViewView.vue +++ b/src/views/AccountViewView.vue @@ -186,8 +186,9 @@

You have done {{ limits.doneRegistrationsThisMonth }} registrations - out of {{ limits.maxRegistrationsPerMonth }} for this month. Your - registrations counter resets at + out of {{ limits.maxRegistrationsPerMonth }} for this month. (You can + register nobody on your first day, and after that only one a day in + your first month.) Your registration counter resets at {{ readableTime(limits.nextMonthBeginDateTime) }}

@@ -700,11 +701,11 @@ export default class AccountViewView extends Vue { this.$notify( { group: "alert", - type: "toast", + type: "success", title: "Download Started", text: "See your downloads directory for the backup.", }, - 5000, + -1, ); } diff --git a/src/views/ContactAmountsView.vue b/src/views/ContactAmountsView.vue index 385a285..897505c 100644 --- a/src/views/ContactAmountsView.vue +++ b/src/views/ContactAmountsView.vue @@ -2,8 +2,11 @@
-
-

+
+

+ +

+ Given with {{ contact?.name }} +

-

- Given with {{ contact?.name }} -

(Only 50 most recent) @@ -358,7 +362,10 @@ export default class ContactsView extends Vue {