Web2認証にイーサリアムを使用する
はじめに
SAML (opens in a new tab)は、アイデンティティプロバイダー(IdP) (opens in a new tab)が[サービスプロバイダー(SP)](https://en.wikipedia.org/wiki/Service_provider_(SAML)にユーザー情報を提供できるようにするためにWeb2で使用される標準です。 (opens in a new tab)
このチュートリアルでは、イーサリアムの署名をSAMLと統合し、ユーザーがイーサリアムウォレットを使用して、まだイーサリアムをネイティブにサポートしていないWeb2サービスで自分自身を認証できるようにする方法を学びます。
なお、このチュートリアルは2つの異なる読者層に向けて書かれています。
- イーサリアムを理解しており、SAMLを学ぶ必要があるイーサリアム関係者
- SAMLとWeb2認証を理解しており、イーサリアムを学ぶ必要があるWeb2関係者
その結果、すでに知っている入門的な内容が多く含まれることになります。必要に応じて読み飛ばしてください。
イーサリアム関係者向けのSAML
SAMLは中央集権型のプロトコルです。サービスプロバイダー(SP)は、アイデンティティプロバイダー(IdP)との間、またはそのIdPの証明書に署名した認証局 (opens in a new tab)との間に既存の信頼関係がある場合にのみ、IdPからのアサーション(「これは私のユーザーであるJohnであり、A、B、Cを行う権限を持つべきである」など)を受け入れます。
例えば、SPは企業に旅行サービスを提供する旅行代理店であり、IdPは企業の社内ウェブサイトである場合があります。従業員が出張を予約する必要がある場合、旅行代理店は実際に旅行を予約させる前に、企業による認証のために従業員を送信します。
これは、ブラウザ、SP、IdPの3つのエンティティがアクセスを交渉する方法です。SPは、ブラウザを使用しているユーザーについて事前に何も知る必要はなく、IdPを信頼するだけで済みます。
SAML関係者向けのイーサリアム
イーサリアムは分散型システムです。
ユーザーは秘密鍵(通常はブラウザ拡張機能に保持されます)を持っています。秘密鍵から公開鍵を導出でき、そこから20バイトのアドレスを導出できます。ユーザーがシステムにログインする必要がある場合、ナンス(1回限りの値)を含むメッセージに署名するよう求められます。サーバーは、その署名がそのアドレスによって作成されたことを検証できます。
署名はイーサリアムのアドレスのみを検証します。他のユーザー属性を取得するには、通常アテステーション (opens in a new tab)を使用します。アテステーションには通常、以下のフィールドがあります。
- Attestor(アテスター): アテステーションを行ったアドレス
- Recipient(受信者): アテステーションが適用されるアドレス
- Data(データ): 名前、権限など、証明されるデータ
- Schema(スキーマ): データを解釈するために使用されるスキーマのID
イーサリアムの分散型の性質により、どのユーザーでもアテステーションを行うことができます。どのアテステーションを信頼できると見なすかを特定するためには、アテスターの身元が重要です。
セットアップ
最初のステップは、SAML SPとSAML IdPが相互に通信できるようにすることです。
- ソフトウェアをダウンロードします。この記事のサンプルソフトウェアはGitHub上 (opens in a new tab)にあります。異なるステージは異なるブランチに保存されており、このステージでは
saml-onlyが必要です。
git clone https://github.com/qbzzt/saml-eth-idp.git
cd saml-eth-idp
git checkout stage-1
yarn
- 自己署名証明書を使用して鍵を作成します。これは、鍵がそれ自身の認証局であることを意味し、サービスプロバイダーに手動でインポートする必要があります。詳細については、OpenSSLのドキュメント (opens in a new tab)を参照してください。
openssl req -x509 -newkey rsa:4096 -keyout src/idp-private-key.pem -out src/idp-public-cert.pem -days 365 -nodes
openssl req -x509 -newkey rsa:4096 -keyout src/sp-private-key.pem -out src/sp-public-cert.pem -days 365 -nodes
- サーバー(SPとIdPの両方)を起動します。
yarn start
-
URL http://localhost:3000/ (opens in a new tab) でSPにアクセスし、ボタンをクリックしてIdP(ポート3001)にリダイレクトされます。
-
IdPにメールアドレスを提供し、Login to the service providerをクリックします。サービスプロバイダー(ポート3000)にリダイレクトされ、メールアドレスによってあなたが認識されていることを確認します。
詳細な説明
ステップバイステップで何が起こるかを説明します。
src/config.mts
このファイルには、アイデンティティプロバイダーとサービスプロバイダーの両方の設定が含まれています。通常、これら2つは異なるエンティティですが、ここでは簡略化のためにコードを共有できます。
// 簡略化のために、SPとIdPの両方の設定をここに置きます。
// 実際のシステムでは、これらは別々のエンティティになります。
import fs from "fs"
// テスト中なので、HTTPを使用しても問題ありません。
const spUrl = "http://localhost:3000"
const idpUrl = "http://localhost:3001"
今のところテスト中なので、HTTPを使用しても問題ありません。
// 公開鍵を読み込みます。これらは通常、両方のコンポーネントで利用可能です
// (直接信頼されるか、信頼できる認証局によって署名されています)。
const idpPubKey = fs.readFileSync("src/idp-public-cert.pem", "utf8")
const spPubKey = fs.readFileSync("src/sp-public-cert.pem", "utf8")
公開鍵を読み込みます。これは通常、両方のコンポーネントで利用可能です(直接信頼されるか、信頼できる認証局によって署名されています)。
// サービスプロバイダーの公開データ
export const spPublicData = {
entityID: `${spUrl}/sp/metadata`,
assertionConsumerService: [
{
Binding: "urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST",
Location: `${spUrl}/sp/acs`,
},
],
signingCertificates: [spPubKey],
}
両方のコンポーネントのURLです。
// サービスプロバイダーの公開データ
export const spPublicData = {
entityID: `${spUrl}/sp/metadata`,
assertionConsumerService: [
{
Binding: "urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST",
Location: `${spUrl}/sp/acs`,
},
],
signingCertificates: [spPubKey],
}
サービスプロバイダーの公開データです。
entityID: `${spUrl}/sp/metadata`,
慣例として、SAMLではentityIDはエンティティのメタデータが利用可能なURLです。このメタデータは、XML形式であることを除いて、ここでの公開データに対応します。
assertionConsumerService: [
{
Binding: "urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST",
Location: `${spUrl}/sp/acs`,
},
],
私たちの目的において最も重要な定義はassertionConsumerServerです。これは、サービスプロバイダーに何かをアサートする(例えば、「この情報を送信するユーザーはsomebody@example.com (opens email client)である」)には、URL http://localhost:3000/sp/assertionに対してHTTP POST (opens in a new tab)を使用する必要があることを意味します。
// アイデンティティプロバイダーの公開データ
export const idpPublicData = {
entityID: `${idpUrl}/idp/metadata`,
singleSignOnService: [
{
Binding: "urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST",
Location: `${idpUrl}/idp/sso`,
},
],
singleLogoutService: [
{
Binding: "urn:oasis:names:tc:SAML:2.0:bindings:HTTP-POST",
Location: `${idpUrl}/idp/slo`,
},
],
signingCertificates: [idpPubKey],
}
アイデンティティプロバイダーの公開データも同様です。ユーザーをログインさせるにはhttp://localhost:3001/idp/loginにPOSTし、ユーザーをログアウトさせるにはhttp://localhost:3001/idp/logoutにPOSTすることを指定しています。
src/sp.mts
これはサービスプロバイダーを実装するコードです。
import * as samlify from "samlify"
SAMLを実装するためにsamlify (opens in a new tab)ライブラリを使用します。
import * as validator from "@authenio/samlify-node-xmllint"
samlify.setSchemaValidator(validator)
samlifyライブラリは、XMLが正しいこと、期待される公開鍵で署名されていることなどを検証するパッケージがあることを想定しています。この目的のために@authenio/samlify-node-xmllint (opens in a new tab)を使用します。
import express from "express"
const spRouter = express.Router()
express (opens in a new tab)のRouter (opens in a new tab)は、ウェブサイト内にマウントできる「ミニウェブサイト」です。この場合、すべてのサービスプロバイダーの定義をグループ化するために使用します。
import fs from "fs"
import { spPublicData, idpPublicData } from "./config.mjs"
const sp = samlify.ServiceProvider({
...spPublicData,
privateKey: fs.readFileSync("src/sp-private-key.pem", "utf8"),
})
サービスプロバイダー自身の表現は、すべての公開データと、情報に署名するために使用する秘密鍵です。
const idp = samlify.IdentityProvider(idpPublicData)
公開データには、サービスプロバイダーがアイデンティティプロバイダーについて知る必要があるすべてが含まれています。
spRouter.get("/metadata", (req, res) => {
res.header("Content-Type", "text/xml").send(sp.getMetadata())
})
他のSAMLコンポーネントとのインターオペラビリティを可能にするために、サービスプロバイダーとアイデンティティプロバイダーは、公開データ(メタデータと呼ばれます)をXML形式で/metadataで利用できるようにする必要があります。
spRouter.post("/acs", async (req, res) => {
try {
// console.log(Buffer.from(req.body.SAMLResponse, 'base64').toString('utf8'))
これは、ブラウザが自身を識別するためにアクセスするページです。アサーションにはユーザー識別子(ここではメールアドレスを使用)が含まれ、追加の属性を含めることができます。これは、上記のシーケンス図のステップ7のハンドラーです。
// console.log(Buffer.from(req.body.SAMLResponse, 'base64').toString('utf8'))
コメントアウトされたコマンドを使用して、アサーションで提供されたXMLデータを確認できます。これはBase64エンコード (opens in a new tab)されています。
const { extract } = await sp.parseLoginResponse(idp, "post", req)
アイデンティティサーバーからのログインリクエストを解析します。
res.send(`
<html>
<body>
<h1>Login successful</h1>
<p>Welcome, ${extract.nameID}</p>
</body>
</html>
`)
ログインを受け取ったことをユーザーに示すために、HTMLレスポンスを送信します。
} catch (e) {
console.error(e)
res.status(500).send("Login failed")
}
})
失敗した場合はユーザーに通知します。
spRouter.get("/login", (req, res) => {
ブラウザがこのページを取得しようとしたときにログインリクエストを作成します。これは、上記のシーケンス図のステップ1のハンドラーです。
const { id, context } = sp.createLoginRequest(idp, "redirect")
ログインリクエストをPOSTするための情報を取得します。
res.send(`
<html>
<body onload="document.forms[0].submit()">
このページはフォーム(下記参照)を自動的に送信します。これにより、ユーザーはリダイレクトされるために何もする必要がありません。これは、上記のシーケンス図のステップ2です。
<form method="post" action="${context.split("?")[0]}">
loginRequest.entityEndpoint(アイデンティティプロバイダーのエンドポイントのURL)にPOSTします。
<input type="hidden" name="SAMLRequest" value="${
context.split("=")[1]
}" />
<input type="submit" value="Login" />
</form>
</body>
</html>
`)
})
入力名はloginRequest.type(SAMLRequest)です。そのフィールドのコンテンツはloginRequest.contextであり、これもBase64エンコードされたXMLです。
const app = express()
app.use(express.urlencoded({ extended: true }))
このミドルウェア (opens in a new tab)は、HTTPリクエスト (opens in a new tab)のボディを読み取ります。ほとんどのリクエストでは必要ないため、デフォルトではExpressはこれを無視します。POSTはボディを使用するため、これが必要です。
app.use("/sp", spRouter)
サービスプロバイダーのディレクトリ(/sp)にルーターをマウントします。
app.get("/", (req, res) => {
res.send(`
<html>
<body>
<h1>Service Provider</h1>
<a href="/sp/login">Login</a>
</body>
</html>
`)
})
ブラウザがルートディレクトリを取得しようとした場合、ログインページへのリンクを提供します。
app.listen(3000, () => {
console.log("SP listening on port 3000")
})
このExpressアプリケーションでspPortをリッスンします。
src/idp.mts
これはアイデンティティプロバイダーです。サービスプロバイダーと非常に似ており、以下の説明は異なる部分に関するものです。
import { XMLParser } from "fast-xml-parser"
const parser = new XMLParser({ ignoreAttributes: false })
サービスプロバイダーから受け取るXMLリクエストを読み取り、理解する必要があります。
const loginPage = (reqId: string) => `
<html>
<body>
<h1>Identity Provider</h1>
<form method="post" action="/idp/login">
この関数は、上記のシーケンス図のステップ4で返される、自動送信フォームを含むページを作成します。
<input type="hidden" name="reqId" value="${reqId}" />
<input type="text" name="email" placeholder="Email" />
サービスプロバイダーに送信するフィールドは2つあります。
- 応答している
requestId。 - ユーザー識別子(今のところ、ユーザーが提供するメールアドレスを使用します)。
idpRouter.post("/login", async (req, res) => {
これは、上記のシーケンス図のステップ5のハンドラーです。idp.createLoginResponse (opens in a new tab)はログインレスポンスを作成します。
const { id, context } = await idp.createLoginResponse(
sp,
オーディエンスはサービスプロバイダーです。
{
extract: {
request: {
id: req.body.reqId,
},
},
},
リクエストから抽出された情報です。リクエストで私たちが気にする唯一のパラメーターはrequestIdであり、これによりサービスプロバイダーはリクエストとそのレスポンスを一致させることができます。
"post",
レスポンスに署名するためのデータを持つためにsigningKeyが必要です。サービスプロバイダーは署名されていないリクエストを信頼しません。
{
nameID: req.body.email,
}
)
これは、サービスプロバイダーに送り返すユーザー情報を含むフィールドです。
res.send(`
<html>
<body onload="document.forms[0].submit()">
<form method="post" action="${context.split("?")[0]}">
<input type="hidden" name="SAMLResponse" value="${
context.split("=")[1]
}" />
<input type="submit" value="Login" />
</form>
</body>
</html>
`)
})
ここでも、自動送信フォームを使用します。これは、上記のシーケンス図のステップ6です。
idpRouter.post("/sso", async (req, res) => {
これは、サービスプロバイダーからログインリクエストを受け取るエンドポイントです。これは、上記のシーケンス図のステップ3のハンドラーです。
const xml = Buffer.from(req.body.SAMLRequest, "base64").toString("utf8")
const parsed = parser.parse(xml)
const reqId = parsed["samlp:AuthnRequest"]["@_ID"]
res.send(loginPage(reqId))
})
認証リクエストのIDを読み取るためにidp.parseLoginRequest (opens in a new tab)を使用できるはずです。しかし、私はそれを機能させることができず、多くの時間を費やす価値がなかったため、単に汎用のXMLパーサー (opens in a new tab)を使用しています。必要な情報は、XMLのトップレベルにある<samlp:AuthnRequest>タグ内のID属性です。
イーサリアムの署名を使用する
ユーザーのアイデンティティをサービスプロバイダーに送信できるようになったので、次のステップは信頼できる方法でユーザーのアイデンティティを取得することです。Viemを使用すると、ウォレットにユーザーのアドレスを尋ねるだけで済みますが、これはブラウザに情報を尋ねることを意味します。私たちはブラウザを制御していないため、ブラウザから得られるレスポンスを自動的に信頼することはできません。
代わりに、IdPはブラウザに署名するための文字列を送信します。ブラウザ内のウォレットがこの文字列に署名した場合、それは本当にそのアドレスである(つまり、そのアドレスに対応する秘密鍵を知っている)ことを意味します。
これを実際に確認するには、既存のIdPとSPを停止し、以下のコマンドを実行します。
git checkout stage-2
yarn
yarn start
次に、SPにアクセス (opens in a new tab)し、指示に従います。
現時点では、イーサリアムのアドレスからメールアドレスを取得する方法がわからないため、代わりに<ethereum address>@bad.email.addressをSPに報告することに注意してください。
詳細な説明
変更点は、前の図のステップ4〜5にあります。
変更したファイルはidp.mtsのみです。変更された部分は以下の通りです。
import { v4 as uuidv4 } from "uuid"
import { verifyMessage } from "viem"
これら2つの追加ライブラリが必要です。ナンス (opens in a new tab)値を作成するためにuuid (opens in a new tab)を使用します。値自体は重要ではなく、1回だけ使用されるという事実が重要です。
viem (opens in a new tab)ライブラリを使用すると、イーサリアムの定義を使用できます。ここでは、署名が実際に有効であることを検証するために必要です。
const loginPage = (nonce: string) => `
<html>
<body>
<h1>Identity Provider</h1>
<p>Please sign the message in your wallet to log in.</p>
ウォレットは、メッセージに署名する許可をユーザーに求めます。単なるナンスであるメッセージはユーザーを混乱させる可能性があるため、このプロンプトを含めます。
const reqIds: Record<string, string> = {}
リクエストに応答できるようにするために、リクエスト情報が必要です。リクエストと一緒に送信し(ステップ4)、送り返してもらう(ステップ5)こともできます。しかし、潜在的に悪意のあるユーザーの制御下にあるブラウザから得られる情報を信頼することはできません。そのため、ナンスをキーとしてここに保存する方が良いでしょう。
簡略化のために、ここでは変数として行っていることに注意してください。ただし、これにはいくつかの欠点があります。
- サービス拒否攻撃に対して脆弱です。悪意のあるユーザーが複数回ログオンを試み、メモリをいっぱいに埋め尽くす可能性があります。
- IdPプロセスを再起動する必要がある場合、既存の値を失います。
- それぞれが独自の変数を持つことになるため、複数のプロセス間で負荷分散を行うことはできません。
本番システムでは、データベースを使用し、何らかの有効期限メカニズムを実装します。
idpRouter.post("/sso", async (req, res) => {
const xml = Buffer.from(req.body.SAMLRequest, "base64").toString("utf8")
const parsed = parser.parse(xml)
const reqId = parsed["samlp:AuthnRequest"]["@_ID"]
const nonce = uuidv4()
reqIds[nonce] = reqId
res.send(loginPage(nonce))
})
ナンスを作成し、将来の使用のためにrequestIdを保存します。
<script type="module">
このJavaScriptは、ページが読み込まれたときに自動的に実行されます。
import { createWalletClient, custom } from 'https://esm.sh/viem'
viemからいくつかの関数が必要です。
if (!window.ethereum) {
alert('Please install a wallet to log in.')
window.location.reload()
}
ブラウザにウォレットがある場合にのみ機能します。
const [account] = await window.ethereum.request({ method: 'eth_requestAccounts' })
ウォレットからアカウントのリストをリクエストします(window.ethereum)。少なくとも1つあると仮定し、最初のものだけを保存します。
const client = createWalletClient({
account,
transport: custom(window.ethereum)
})
ブラウザのウォレットと対話するためのウォレットクライアント (opens in a new tab)を作成します。
const signature = await client.signMessage({ message: "${nonce}" })
ユーザーにメッセージへの署名を求めます。このHTML全体がテンプレート文字列 (opens in a new tab)内にあるため、idpプロセスで定義された変数を使用できます。これはシーケンス図のステップ4.5です。
window.location.href = \`/idp/login?nonce=${nonce}&signature=\${signature}&account=\${account}\`
/idp/signature/<nonce>/<address>/<signature>にリダイレクトします。これはシーケンス図のステップ5です。
</script>
署名はブラウザによって送り返されますが、ブラウザは潜在的に悪意がある可能性があります(ブラウザで単にhttp://localhost:3001/idp/signature/bad-nonce/bad-address/bad-signatureを開くことを止めるものは何もありません)。したがって、IdPプロセスが不正な署名を正しく処理することを検証することが重要です。
</body>
</html>
`
残りは単なる標準的なHTMLです。
idpRouter.get("/login", async (req, res) => {
これはシーケンス図のステップ5のハンドラーです。
const nonce = req.query.nonce as string
const reqId = reqIds[nonce]
delete reqIds[nonce]
リクエストIDを取得し、再利用できないようにnoncesからナンスを削除します。
try {
署名が無効になる可能性のある方法は非常に多いため、スローされたエラーをキャッチするためにこれをtry ... catchブロックでラップします。
const valid = await verifyMessage({
address: req.query.account as `0x${string}`,
message: nonce,
signature: req.query.signature as `0x${string}`,
})
if (!valid) {
throw new Error("Invalid signature")
}
シーケンス図のステップ5.5を実装するためにverifyMessage (opens in a new tab)を使用します。
const { id, context } = await idp.createLoginResponse(
sp,
{
extract: {
request: {
id: reqId,
},
},
},
"post",
{
nameID: `${req.query.account} (not an email)`,
}
)
ハンドラーの残りの部分は、1つの小さな変更を除いて、以前に/loginSubmittedハンドラーで行ったことと同等です。
nameID: `${req.query.account} (not an email)`,
実際のメールアドレスは持っていないため(次のセクションで取得します)、今のところはイーサリアムのアドレスを返し、それがメールアドレスではないことを明確にマークします。
idpRouter.post("/sso", async (req, res) => {
ステップ3のハンドラーで、getLoginPageの代わりにgetSignaturePageを使用するようになりました。
メールアドレスの取得
次のステップは、サービスプロバイダーから要求された識別子であるメールアドレスを取得することです。そのために、Ethereum Attestation Service(EAS) (opens in a new tab)を使用します。
アテステーションを取得する最も簡単な方法は、GraphQL API (opens in a new tab)を使用することです。以下のクエリを使用します。
query GetAttestationsByRecipient {
attestations(
where: {
recipient: { equals: "${getAddress(ethAddr)}" }
schemaId: { equals: "0xfa2eff59a916e3cc3246f9aec5e0ca00874ae9d09e4678e5016006f07622f977" }
}
take: 1
) {
data
id
attester
}
}
このschemaId (opens in a new tab)にはメールアドレスのみが含まれています。このクエリは、このスキーマのアテステーションを要求します。アテステーションの対象はrecipientと呼ばれます。これは常にイーサリアムのアドレスです。
警告:ここでアテステーションを取得する方法には、2つのセキュリティ上の問題があります。
-
APIエンドポイントである
https://optimism.easscan.org/graphqlにアクセスしていますが、これは中央集権型のコンポーネントです。id属性を取得し、オンチェーンでルックアップを行ってアテステーションが本物であることを検証することはできますが、APIエンドポイントはアテステーションについて私たちに伝えないことで、依然としてアテステーションを検閲することができます。この問題は解決不可能ではありません。独自のGraphQLエンドポイントを実行し、チェーンのログからアテステーションを取得することもできますが、私たちの目的には過剰です。
-
アテスターの身元を確認していません。誰でも私たちに偽の情報を与えることができます。現実世界の実装では、信頼できるアテスターのセットを用意し、彼らのアテステーションのみを確認します。
これを実際に確認するには、既存のIdPとSPを停止し、以下のコマンドを実行します。
git checkout stage-3
yarn
yarn start
次に、メールアドレスを提供します。これを行うには2つの方法があります。
-
秘密鍵を使用してウォレットをインポートし、テスト用の秘密鍵
0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80を使用します。 -
自分のメールアドレスのアテステーションを追加します。
-
Attest with Schemaをクリックします。
-
受信者としてイーサリアムのアドレス、メールアドレスとしてメールアドレスを入力し、Onchainを選択します。次に、Make Attestationをクリックします。
-
ウォレットでトランザクションを承認する。ガスの支払いには、オプティミズムブロックチェーン (opens in a new tab)上のETHがいくらか必要になります。
いずれにせよ、これを行った後、http://localhost:3000 (opens in a new tab) にアクセスし、指示に従います。テスト用の秘密鍵をインポートした場合、受け取るメールアドレスはtest_addr_0@example.comです。自分のアドレスを使用した場合、それはあなたがアテステーションした内容になります。
詳細な説明
新しいステップは、GraphQL通信であるステップ5.6と5.7です。
繰り返しになりますが、idp.mtsの変更された部分は以下の通りです。
import { GraphQLClient, gql } from "graphql-request"
import { SchemaEncoder } from "@ethereum-attestation-service/eas-sdk"
import { getAddress } from "viem"
必要なライブラリをインポートします。
const EAS_GRAPHQL_URL = "https://optimism.easscan.org/graphql"
各ブロックチェーンに個別のエンドポイント (opens in a new tab)があります。
const graphQLClient = new GraphQLClient(EAS_GRAPHQL_URL)
エンドポイントのクエリに使用できる新しいGraphQLClientクライアントを作成します。
const schemaEncoder = new SchemaEncoder("string email")
GraphQLは、バイトを含む不透明なデータオブジェクトのみを提供します。それを理解するにはスキーマが必要です。
const getEmail = async (ethAddr: string) => {
イーサリアムのアドレスからメールアドレスを取得する関数です。
const query = gql`
これはGraphQLクエリです。
query GetAttestationsByRecipient {
attestations(
アテステーションを探しています。
where: {
recipient: { equals: "${getAddress(ethAddr)}" }
schemaId: { equals: "0xfa2eff59a916e3cc3246f9aec5e0ca00874ae9d09e4678e5016006f07622f977" }
}
必要なアテステーションは、受信者がgetAddress(ethAddr)であるスキーマ内のものです。getAddress (opens in a new tab)関数は、アドレスが正しいチェックサム (opens in a new tab)を持っていることを確認します。GraphQLは大文字と小文字を区別するため、これが必要です。「0xBAD060A7」、「0xBad060A7」、および「0xbad060a7」は異なる値です。
take: 1
見つかったアテステーションの数に関係なく、最初のものだけが必要です。
) {
data
id
attester
}
}
`
受け取りたいフィールドです。
attester: アテステーションを送信したアドレス。通常、これはアテステーションを信頼するかどうかを決定するために使用されます。id: アテステーションID。この値を使用してオンチェーンでアテステーションを読み取り (opens in a new tab)、GraphQLクエリからの情報が正しいことを検証できます。data: スキーマデータ(この場合はメールアドレス)。
const response: any = await graphQLClient.request(query)
if (response.attestations.length === 0) {
return `${ethAddr} (no email attestation)`
}
アテステーションがない場合は、明らかに間違っているが、サービスプロバイダーには有効に見える値を返します。
const decodedData = schemaEncoder.decodeData(response.attestations[0].data)
return decodedData[0].value.value
}
値がある場合は、decodeDataを使用してデータをデコードします。それが提供するメタデータは必要なく、値自体だけが必要です。
{
nameID: await getEmail(req.query.account as string),
}
新しい関数を使用してメールアドレスを取得します。
分散化についてはどうですか?
この構成では、イーサリアムからメールアドレスへのマッピングについて信頼できるアテスターに依存している限り、ユーザーは自分以外の誰かになりすますことはできません。しかし、私たちのアイデンティティプロバイダーは依然として中央集権型のコンポーネントです。アイデンティティプロバイダーの秘密鍵を持っている人は誰でも、サービスプロバイダーに偽の情報を送信できます。
マルチパーティ計算(MPC) (opens in a new tab)を使用した解決策があるかもしれません。将来のチュートリアルでそれについて書きたいと思います。
まとめ
イーサリアムの署名などのログオン標準の採用は、鶏と卵の問題に直面しています。サービスプロバイダーは、可能な限り幅広い市場にアピールしたいと考えています。ユーザーは、自分のログオン標準のサポートを心配することなくサービスにアクセスできることを望んでいます。 イーサリアムIdPなどのアダプターを作成することで、このハードルを乗り越えることができます。
私の他の作品についてはこちらをご覧ください (opens in a new tab)。
ページの最終更新: 2026年4月3日





