使用 Firebase Auth 模拟器以编程方式创建用户

Cha*_*wis 2 jsdom firebase jestjs firebase-tools firebase-authentication

我正在尝试使用 Firebase Auth 模拟器编写笑话测试并继续收到以下 CORS 错误。

console.error
    Error: Headers X-Client-Version forbidden
        at dispatchError (/Users/me/my-project/node_modules/jsdom/lib/jsdom/living/xhr/xhr-utils.js:62:19)
        at validCORSPreflightHeaders (/Users/me/my-project/node_modules/jsdom/lib/jsdom/living/xhr/xhr-utils.js:99:5)
        at Request.<anonymous> (/Users/me/my-project/node_modules/jsdom/lib/jsdom/living/xhr/xhr-utils.js:367:12)
        at Request.emit (events.js:315:20)
        at Request.onRequestResponse (/Users/me/my-project/node_modules/request/request.js:1059:10)
        at ClientRequest.emit (events.js:315:20)
        at HTTPParser.parserOnIncomingClient [as onIncoming] (_http_client.js:641:27)
        at HTTPParser.parserOnHeadersComplete (_http_common.js:126:17)
        at Socket.socketOnData (_http_client.js:509:22)
        at Socket.emit (events.js:315:20) undefined
Run Code Online (Sandbox Code Playgroud)

测试非常简单:

import { renderHook, act } from "@testing-library/react-hooks"
import faker from "faker"
import { useAuth, FirebaseProvider, firebase } from "./index"


const wrapper = ({ firebase, children }) => {
  return <FirebaseProvider firebase={firebase}>{children}</FirebaseProvider>
}

const createUser = ({ email = faker.internet.email(), password = faker.internet.password({ length: 6 }) } = {}) => {
  return firebase
    .auth()
    .createUserWithEmailAndPassword(email, password)
    .then(user => user)
}

const signUserIn = ({ email, password } = {}) => {
  return firebase
    .auth()
    .signInWithEmailAndPassword(email, password)
    .then(user => user)
}

describe("useAuth", () => {
  it("will return the user", async () => {
    const { result } = renderHook(() => useAuth(), { wrapper, initialProps: { firebase } })
    const email = faker.internet.email()
    const password = faker.internet.password()
    await act(async () => {
      const user = await createUser({ email, password }) // this fails
      await signUserIn({ email, password }) //and so does this
    })
    expect(result.user).toEqual({ email, password })
  })
})
Run Code Online (Sandbox Code Playgroud)

作为参考,索引文件:

const FirebaseProvider = ({ children, firebase }) => {
  const firestore = firebase.firestore()
  const auth = firebase.auth()

  if (useEmulator()) {
    firestore.useEmulator("localhost", 8080)
    auth.useEmulator("http://localhost:9099/")
  }

  const value = { firestore, auth }

  return <FirebaseContext.Provider value={value}>{children}</FirebaseContext.Provider>
}

const throwError = hook => {
  throw new Error(`${hook} must be used within a FirebaseProvider`)
}

const useAuth = () => {
  const context = useContext(FirebaseContext)
  if (context === undefined) throwError("useAuth")

  const [user, setUser] = useState()

  useEffect(() => {
    const cleanup = context.auth.onAuthStateChanged(authUser => {
      authUser ? setUser(authUser) : setUser(null)
    })
    return () => cleanup()
  })

  return { ...context.auth, user }
}
Run Code Online (Sandbox Code Playgroud)

我曾尝试使用实际模拟器使用的 REST 端点(如下),但它以相同的方式出错。

http://localhost:9099/identitytoolkit.googleapis.com/v1/projects/<my-project>/accounts
Run Code Online (Sandbox Code Playgroud)

无论如何,在使用 jest 时可以运行它吗?或者我是否需要使用模拟器 UI 创建帐户,导出它们并在运行测试时重新导入?

我发现我可以使用下面的 REST 端点在测试中创建一个用户,但是它绕过了模拟器并创建了一个真正的用户。

https://identitytoolkit.googleapis.com/v1/accounts:signUp?key=<api-key>
Run Code Online (Sandbox Code Playgroud)

Chr*_*irk 6

更新 jsdom 版本16.5.2

这个新版本现在支持 access-control-allow-headers 的通配符,所以更新到这个版本或使用它作为解决方案,对于使用 Create React App 创建的项目,解决了这个问题。

16.5.2之前的jsdom解决方案

该错误是由 jsdom 引发的,因为它不支持 access-control-allow-headers 的通配符,但 firebase 使用通配符(请参阅此问题以了解 jsdom以及firebase相关的拉取请求)。有两个开放的拉取请求可以解决这个问题:https : //github.com/jsdom/jsdom/pull/3073https://github.com/jsdom/jsdom/pull/2867

可以通过手动更改node_modules文件夹中的相关代码或使用 fork 作为依赖项来解决此问题package.json

"jsdom": "silviot/jsdom#fix/allow-headers"
Run Code Online (Sandbox Code Playgroud)

如果jsdom不是直接依赖项,则可以将以下内容添加到package.json顶层:

"resolutions": {
  "jsdom": "silviot/jsdom#fix/allow-headers"
}
Run Code Online (Sandbox Code Playgroud)

如果使用 fork,jsdom文件夹中会丢失一些自动生成的文件。这些可以通过运行npm installyarn install在文件夹中生成。要自动执行此操作,您可以将prepare脚本添加到package.json

"scripts": {
  "prepare": "cd node_modules/jsdom && yarn"
},
Run Code Online (Sandbox Code Playgroud)


Emi*_*ers 6

我在 firebase auth 模拟器中以编程方式创建用户时也遇到了问题。

而不是使用

https://identitytoolkit.googleapis.com/v1/accounts:signUp?key=[API_KEY]
Run Code Online (Sandbox Code Playgroud)

您必须使用以下格式:

http://localhost:9099/identitytoolkit.googleapis.com/v1/accounts:signUp?key=[API_KEY]
Run Code Online (Sandbox Code Playgroud)

然后给出一个像这样的 JSON 主体,然后点击 post

{
"email": "test@test.com",
"password": "test12"
}
Run Code Online (Sandbox Code Playgroud)

瞧!您的模拟器中有一个用户。将此与 fetch 或 axios 结合起来,您就可以为模拟器播种用户。如果您需要添加自定义声明或其他信息,请在函数模拟器中创建在用户创建时触发的函数。

functions.auth.user().onCreate
Run Code Online (Sandbox Code Playgroud)