Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Migrate to Prosopo Procaptcha #1

Closed
wants to merge 2 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions .env.example
Original file line number Diff line number Diff line change
Expand Up @@ -32,3 +32,11 @@ SMF_CONFIG_DB_PORT=5432
SMF_CONFIG_DB_USERNAME=postgres
SMF_CONFIG_DB_PASSWORD=postgres
SMF_CONFIG_DB_DATABASE_NAME=faucet

# Captcha provider details
SMF_CONFIG_CAPTCHA_PROVIDER=procaptcha
SMF_CONFIG_PROSOPO_DEFAULT_ENVIRONMENT=production
SMF_CONFIG_PROSOPO_DEFAULT_NETWORK=rococo
SMF_CONFIG_PROSOPO_DAPP_NAME=Polkadot
SMF_CONFIG_PROSOPO_SUBSTRATE_ENDPOINT=wss://rococo-contracts-rpc.polkadot.io:443
SMF_CONFIG_PROSOPO_SITE_KEY=5C4hrfjw9DjXZTzV3MwzrrAr9P1MJhSrvWGWqi1eSuyUpnhM
2 changes: 1 addition & 1 deletion .eslintignore
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
data
build
test
test
5 changes: 3 additions & 2 deletions .github/workflows/E2E.yml
Original file line number Diff line number Diff line change
Expand Up @@ -26,11 +26,12 @@ jobs:
- name: Run a local relaychain with a parachain using zombienet
run: |
export PATH=$(pwd):$PATH
npx --yes @zombienet/cli@1.3.43 \
npx --yes @zombienet/cli@1.3.86 -l text \
--provider native spawn zombienet.native.toml \
> polkadot.txt 2>&1 &
source wait_until.sh 'curl -s "127.0.0.1:9933"'
source wait_until.sh 'curl -s "127.0.0.1:9923"'
source wait_until.sh 'curl -s "127.0.0.1:9934"'
source wait_until.sh 'curl -s "127.0.0.1:9988"'
working-directory: e2e
- name: Build faucet
run: yarn build:docker
Expand Down
29 changes: 20 additions & 9 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ Example requests:
curl -X POST \
localhost:5555/drip/web \
-H "Content-Type: application/json" \
-d '{"address": "xxx", "parachain_id": "1002", "recaptcha": "captcha_token"}'
-d '{"address": "xxx", "parachain_id": "1002", "captchaResponse": "captcha_token"}'
```

In React:
Expand All @@ -76,7 +76,7 @@ const request = async () => {
const body = {
address: "xxx",
parachain_id: "1002",
recaptcha: captcha_token
captchaResponse: captcha_token
}

const fetchResult = await fetch("http://localhost:5555/drip/web", {
Expand All @@ -89,15 +89,26 @@ const request = async () => {
}
```

Where the `captcha_token` is a recaptcha token created with a `sitekey`
is matching the recaptcha secret specified in `SMF_BACKEND_RECAPTCHA_SECRET`.
Where the `captcha_token` is either a

For testing, you can use a public, testing recaptcha secret which will allow any captcha token to pass.
- JSON payload that includes a verified rococo address [procaptcha](https://prosopo.io)

```shell
# Public testing secret, will accept all tokens.
SMF_BACKEND_RECAPTCHA_SECRET="6LeIxAcTAAAAAGG-vFI1TnRWxMZNFuojJ4WifJWe"
```
or

- a recaptcha token created with a `sitekey` matching the [recaptcha](https://developers.google.com/recaptcha/) secret specified in `SMF_BACKEND_RECAPTCHA_SECRET`.

For testing, you can use

- Alice's address `5GrwvaEF5zXb26Fz9rcQpDWS57CtERHpNehXCPcNoHGKutQY` with procaptcha

or

- a public, testing captcha secret which will allow any captcha token to pass with recaptcha.

```shell
# Public testing secret, will accept all tokens.
SMF_BACKEND_RECAPTCHA_SECRET="6LeIxAcTAAAAAGG-vFI1TnRWxMZNFuojJ4WifJWe"
```

### Helm chart

Expand Down
6 changes: 5 additions & 1 deletion client/.env
Original file line number Diff line number Diff line change
@@ -1,9 +1,13 @@
PUBLIC_DEMO_MODE=
PUBLIC_CAPTCHA_KEY=6LeIxAcTAAAAAJcZVRqyHh71UMIEGNQ_MXjiZKhI
PUBLIC_RECAPTCHA_KEY=6LeIxAcTAAAAAJcZVRqyHh71UMIEGNQ_MXjiZKhI
PUBLIC_PROSOPO_SITE_KEY=5C4hrfjw9DjXZTzV3MwzrrAr9P1MJhSrvWGWqi1eSuyUpnhM

PUBLIC_FAUCET_URL=
# uncomment to direct requests to local instance
# PUBLIC_FAUCET_URL=http://localhost:5555/drip/web/

PUBLIC_ISSUE_LINK=https://github.com/paritytech/polkadot-testnet-faucet/issues/new/choose
PUBLIC_FORUM="https://forum.polkadot.network/t/experiencing-trouble-accessing-our-rococo-faucet-please-post-here/2952"
PUBLIC_CAPTCHA_PROVIDER=procaptcha


6 changes: 4 additions & 2 deletions client/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,11 @@ Two current options are to [access Matrix and contact a bot](https://wiki.polkad

## Development

To develop you need two env variables:
To develop you need three env variables:

- `PUBLIC_CAPTCHA_KEY`: The [reCaptcha v2 site key](https://www.google.com/u/0/recaptcha/admin).
- `PUBLIC_CAPTCHA_PROVIDER`: The captcha provider. Currently `procaptcha` and `recaptcha` are supported. You will then need one of the following site keys:
- `PUBLIC_PROSOPO_SITE_KEY`: The [Prosopo site key](https://prosopo.io/) which is `5HUBceb4Du6dvMA9BiwN5VzUrzUsX9Zp7z7nSR2cC1TCv5jg`.
- `PUBLIC_RECAPTCHA_KEY`: The [reCaptcha v2 site key](https://www.google.com/u/0/recaptcha/admin).
- `PUBLIC_FAUCET_URL`: The endpoint to contact the faucet backend. Keep unset to run client-side code with production backend.

The reason for which these variables have `PUBLIC_` as a prefix is a security measure to not upload any unnecessary data. [More info here](https://kit.svelte.dev/docs/modules#$env-static-public)
Expand Down
7 changes: 6 additions & 1 deletion client/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,9 @@
"dev": "vite dev",
"build": "vite build",
"preview": "vite preview",
"test": "playwright test",
"test:recaptcha": "playwright test --config playwright.config.recaptcha.ts",
"test:procaptcha": "playwright test --config playwright.config.procaptcha.ts",
"test": "yarn test:recaptcha && yarn test:procaptcha",
"check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json",
"check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch"
},
Expand All @@ -20,10 +22,13 @@
"autoprefixer": "^10.4.16",
"daisyui": "^4.4.4",
"postcss": "^8.4.31",
"prettier": "^3.0.3",
"prettier-plugin-svelte": "^3.0.3",
"svelte": "^4.2.7",
"svelte-check": "^3.6.1",
"svelte-markdown": "^0.4.0",
"svelte-meta-tags": "^3.1.0",
"svelte-preprocess": "^5.1.1",
"tailwindcss": "^3.3.5",
"tslib": "^2.6.2",
"typescript": "^5.3.2",
Expand Down
17 changes: 17 additions & 0 deletions client/playwright.config.procaptcha.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
import type { PlaywrightTestConfig } from "@playwright/test";

const config: PlaywrightTestConfig = {
webServer: {
command: "npm run build && npm run preview",
port: 4173,
env: {
PUBLIC_CAPTCHA_PROVIDER: "procaptcha",
PUBLIC_PROSOPO_SITE_KEY: "5HUBceb4Du6dvMA9BiwN5VzUrzUsX9Zp7z7nSR2cC1TCv5jg",
PUBLIC_DEMO_MODE: "",
PUBLIC_FAUCET_URL: "https://example.com/test",
},
},
testDir: "tests",
};

export default config;
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,13 @@ const config: PlaywrightTestConfig = {
command: "npm run build && npm run preview",
port: 4173,
env: {
PUBLIC_CAPTCHA_KEY: "6LeIxAcTAAAAAJcZVRqyHh71UMIEGNQ_MXjiZKhI",
PUBLIC_CAPTCHA_PROVIDER: "recaptcha",
PUBLIC_RECAPTCHA_KEY: "6LeIxAcTAAAAAJcZVRqyHh71UMIEGNQ_MXjiZKhI",
PUBLIC_DEMO_MODE: "",
PUBLIC_FAUCET_URL: "https://example.com/test",
},
},

testDir: "tests",
};

Expand Down
19 changes: 16 additions & 3 deletions client/src/app.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,14 @@ declare global {
}

declare interface Window {
grecaptcha?: Captcha;
grecaptcha?: Recaptcha;
procaptcha?: Procaptcha;
captchaLoaded: () => void;
onToken: (token: string) => void;
onExpiredToken: () => void;
}
}

interface Captcha {
interface Recaptcha {
render: (
element: string,
key: {
Expand All @@ -30,4 +30,17 @@ interface Captcha {
getResponse: () => string;
}

interface Procaptcha {
render: (
element: string,
key: {
siteKey: string;
callback?: string;
theme?: "light" | "dark";
"chalexpired-callback"?: string;
},
) => void;
default: (callback: () => void) => void;
}

export {};
59 changes: 45 additions & 14 deletions client/src/lib/components/CaptchaV2.svelte
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
<script lang="ts">
<script type="module" lang="ts">
import { createEventDispatcher, onMount } from "svelte";
import Cross from "./icons/Cross.svelte";
import { CaptchaProvider } from "$lib/utils/captcha";

export let captchaKey: string;

export let captchaProvider: string;
const dispatch = createEventDispatcher<{ token: string }>();

const captchaId = "captcha_element";
Expand All @@ -13,7 +14,7 @@
let componentMounted: boolean;

onMount(() => {
window.captchaLoaded = () => {
window.captchaLoaded = async () => {
const colorTheme =
theme === "auto"
? window.matchMedia && window.matchMedia("(prefers-color-scheme: dark)").matches
Expand All @@ -22,21 +23,42 @@
: theme;
const mobileScreen = window.innerHeight > window.innerWidth;

if (!window.grecaptcha) {
if (captchaProvider === CaptchaProvider.procaptcha) {
if (!window.procaptcha) {
captchaError = true;
throw new Error(`${captchaProvider} is undefined!`);
}
window.procaptcha.render(captchaId, {
siteKey: captchaKey,
theme: colorTheme,
callback: "onToken",
"chalexpired-callback": "onExpiredToken",
});
} else if (captchaProvider === CaptchaProvider.recaptcha) {
if (!window.grecaptcha) {
captchaError = true;
throw new Error("grecaptcha is undefined!");
}
window.grecaptcha.render(captchaId, {
sitekey: captchaKey,
theme: colorTheme,
callback: "onToken",
size: mobileScreen ? "compact" : "normal",
"expired-callback": "onExpiredToken",
});
} else {
captchaError = true;
throw new Error("grecaptcha is undefined!");
throw new Error(`Unknown captcha provider: ${captchaProvider}`);
}
window.grecaptcha.render(captchaId, {
sitekey: captchaKey,
theme: colorTheme,
callback: "onToken",
size: mobileScreen ? "compact" : "normal",
"expired-callback": "onExpiredToken",
});
};

window.onToken = (token) => {
dispatch("token", token);
// Forces a new captcha on page reload
if (captchaProvider === CaptchaProvider.procaptcha) {
window.localStorage.removeItem("@prosopo/current_account");
window.localStorage.removeItem("@prosopo/provider");
}
};

// clean the token so the form becomes invalid
Expand All @@ -52,15 +74,24 @@

<svelte:head>
{#if componentMounted}
<script src="https://www.google.com/recaptcha/api.js?onload=captchaLoaded&render=explicit" async defer></script>
{#if captchaProvider === "procaptcha"}
<script
type="module"
src="https://js.prosopo.io/js/procaptcha.bundle.js?render=implicit&onload=captchaLoaded"
async
defer
></script>
{:else}
<script src="https://www.google.com/recaptcha/api.js?onload=captchaLoaded&render=explicit" async defer></script>
{/if}
{/if}
</svelte:head>

{#if captchaError}
<div class="alert alert-error shadow-lg" data-testid="error">
<div>
<Cross />
<span>Error loading Google Captcha. Please reload the page.</span>
<span>Error loading {captchaProvider} Captcha. Please reload the page.</span>
</div>
</div>
{/if}
Expand Down
16 changes: 12 additions & 4 deletions client/src/lib/components/Form.svelte
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
<script lang="ts">
import { PUBLIC_CAPTCHA_KEY } from "$env/static/public";
import type { NetworkData } from "$lib/utils/networkData";
import { PUBLIC_CAPTCHA_PROVIDER, PUBLIC_PROSOPO_SITE_KEY, PUBLIC_RECAPTCHA_KEY } from "$env/static/public";
import { operation, testnet } from "$lib/utils/stores";
import { request as faucetRequest } from "../utils";
import CaptchaV2 from "./CaptchaV2.svelte";
import NetworkDropdown from "./NetworkDropdown.svelte";
import NetworkInput from "./NetworkInput.svelte";
import { CaptchaProvider } from "$lib/utils/captcha";
import NetworkDropdown from "./NetworkDropdown.svelte";
import type { NetworkData } from "$lib/utils/networkData";

let address: string = "";
export let network: number = -1;
Expand Down Expand Up @@ -58,7 +59,14 @@
</div>
{#if !webRequest}
<div class="grid place-items-center">
<CaptchaV2 captchaKey={PUBLIC_CAPTCHA_KEY ?? ""} on:token={onToken} theme="dark" />
<CaptchaV2
captchaKey={PUBLIC_CAPTCHA_PROVIDER === CaptchaProvider.procaptcha
? PUBLIC_PROSOPO_SITE_KEY
: PUBLIC_RECAPTCHA_KEY}
captchaProvider={PUBLIC_CAPTCHA_PROVIDER}
on:token={onToken}
theme="dark"
/>
</div>
<button class="submit-btn" type="submit" data-testid="submit-button" disabled={!formValid}>
Get some {$testnet.currency}s
Expand Down
4 changes: 4 additions & 0 deletions client/src/lib/utils/captcha.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
export enum CaptchaProvider {
procaptcha = "procaptcha",
recaptcha = "recaptcha",
}
8 changes: 4 additions & 4 deletions client/src/lib/utils/faucetRequest.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,24 +4,24 @@ import type { NetworkData } from "./networkData";

export async function request(
address: string,
recaptcha: string,
captchaResponse: string,
network: NetworkData,
parachain?: number,
): Promise<string> {
if (DEMO !== undefined && DEMO !== "") {
return await boilerplateRequest(address);
}
const chain = parachain && parachain > 0 ? parachain.toString() : undefined;
return await faucetRequest(address, recaptcha, network, chain);
return await faucetRequest(address, captchaResponse, network, chain);
}

export async function faucetRequest(
address: string,
recaptcha: string,
captchaResponse: string,
network: NetworkData,
parachain_id?: string,
): Promise<string> {
const body = { address, parachain_id, recaptcha };
const body = { address, parachain_id, captchaResponse };

const url = network.endpoint;
if (!url) {
Expand Down
Loading
Loading