谷歌开通第三方平台OAuth登录及Java对接步骤
调研起因:
当然还是因为手头的海外项目,用户注册通常要用邮箱,正常流程需要给用户邮箱发送验证码,再让用户输入密码进行注册。
为了简化流程,让用户使用谷歌邮箱一键完成注册或登录,
我们直接获取谷歌邮箱、谷歌注册里的头像、姓名等信息,所以要进行谷歌对接。
本教程基于Web H5界面进行对接,同时也提供了spring-boot版本的对接Demo在:
https://github.com/youbl/study/tree/master/study-codes/google-oauth-login-demo
废话不多说,直接开始步骤说明吧:
注意:要注册和对接,必要的科学上网还是需要的,这个自己想办法吧。
1、谷歌OAuth功能申请
1.1、谷歌账号注册
首先肯定是要有一个有效的谷歌账号,去这里注册去吧:https://accounts.google.com/
题外话:谷歌注册机器人太多了,导致正常的人工注册有时也老出问题,比如IP归属地、系统语言、浏览器语言要一致等等。
1.2、创建Project
接着,要去谷歌新建一个Project项目,谷歌要求要有这个项目,才能申请后续的Oauth能力。
进入Google Cloud首页:https://console.cloud.google.com/welcome
注意:首次进入,会让你选择一个国家/地区,并同意服务条款,确认即可
确认后,点击页面的“创建或选择项目”,在弹窗里点击“新建项目”:
在“新建项目”页面,输入“项目名称”,位置保持默认的“无组织”即可,点击创建:
点击创建后,页面右上角会有个小窗,在创建中,请保持等待,等它变成成功:
1.3、选择Project
在上面的Project项目创建成功后,再点击“创建或选择项目”, 选择我们刚刚创建的项目:
点击创建的项目“beinet”后,界面上会显示“您目前位于 beinet”:
1.4、开通OAuth
1.4.1、选择User Type
选择好Project后,点击“快速访问”里的“API和服务”,再点左边的“OAuth 权限请求页面”,右边的“User Type”选择“外部”,再点“创建”
注:下面截图显示说会有变化,我暂时还是用旧界面填写;另外我也体验了一下新页面,内容字段都差不多,只是菜单有点变化而已。
1.4.2、填写应用信息
接下来的页面里,把必填项补充完整,其它项可以后面再填写:
- 应用名:随意,按需填写即可
- 用户支持电子邮件:下拉选择,正常只能选择你注册的那个谷歌邮箱
- 开发者联系信息:输入一个邮箱,可以跟用户支持电子邮件相同
1.4.3、选择可访问的数据范围
点击了上面页面的“保存并继续”,在新页面,点击“添加或移除范围”,可以在弹窗里把每页行数改成100,可以看全部的api,进行查看和数据范围选择,选好后,点弹窗下面的“更新”:
注意:这一步是选择OAuth能访问的用户数据范围,你也可以根据页面说明选择必要的范围,如果选择的数据特别敏感,后面是需要提交书面报告和视频找谷歌审核的,参考步骤1.4.6.2:
如果只是想用谷歌账号进行OAuth登录,只需要选择右边3项,可以不需要审核:.../auth/userinfo.email
和 .../auth/userinfo.profile
再加openid
点击上面的更新后,页面的“您的非敏感范围”和“您的敏感范围”都会变化,可以随时再点“添加或移除范围”进行修改:
1.4.4、选择测试用户
拉到页面最下方,点击“保存并继续”,进入测试用户配置界面,当你要先测试时,可以先指定哪些用户可以使用你的这个OAuth进行测试,我是直接下一步,去发布,这边不选择测试用户:
1.4.5、请求信息预览
点击“保存并继续”,进入摘要信息页面,这里会显示你前面填写的所有信息,拉到页面最下方,点“返回信息中心”,即可:
1.4.6、发布应用
上面信息填写完毕后,OAuth能力还是测试状态,仅测试用户可以使用,需要在这个信息中心页面,发布应用:
- 如果未“发布应用”,那么在对接测试时,会出现“禁止访问:“beinet.cn”尚未完成 Google 验证流程”:
1.4.6.1、权限太大时的发布审核申请
如果你按步骤1.4.3里说的,只选择了OAuth登录需要的数据:.../auth/userinfo.email
和 .../auth/userinfo.profile
和 openid
这3项,
点发布时,会弹出如下界面:
直接点“确认”即可,发布后会变成:
1.4.6.2、权限太大时的发布审核申请
在前面步骤1.4.3,如果你选择了敏感数据,需要提交视频和书面报告给谷歌,才能发布,如下图:
我没有验证如何审核的步骤,因为我只需要OAuth登录,哈哈……
1.5、对接凭据创建
OAuth申请完,我们需要找谷歌创建client id来用于程序对接。
点页面左边的“凭据”,再点右边的“创建凭据”,选择“OAuth客户端ID”:
在新页面的应用类型,选择 “Web应用”:
选择了Web应用后,下面会出来3个填写项:
1.5.1、名称
这个随便填写即可。
1.5.2、已获授权的JavaScript来源
你会在哪些域名下使用当前OAuth能力,需要把域名+端口全部填写到这里:
- 端口如果不是80,必须明确指定端口
- 不能有通配符
- 除了localhost用于调试,其它必须是https协议
- 如果域名使用了端口,一定也要加上无端口的域名,比如 http://localhost,参考:https://stackoverflow.com/questions/68438293/the-given-origin-is-not-allowed-for-the-given-client-id-gsi
只有在配置了域名+端口的网页里,才允许使用谷歌的OAuth能力:
1.5.3、已获授权的重定向URI
如果在谷歌OAuth认证成功后,需要谷歌重定向到你的某个url,则需要把这些重定向的url,都配置在这里,否则无法回调:
- 同样这些url,除localhost调试用之外,必须是https的
- 不允许使用IP
- 不能有通配符
- 要精确匹配,比如缺少斜杠也会报错: redirect_uri_mismatch
1.6、创建和保存客户端ID
填写完步骤1.5的信息,点击“创建”,成功后,会弹出客户端ID和客户端密钥的页面,并支持下载:
把它们保存下来,没保存也没关系,后面也可以再回来复制:
注:在实际的开发中,我只用到了“客户端ID”,没有发现“客户端密钥”在哪里可以使用
1.7、查看客户端ID和编辑域名
如果忘记保存了,可以点凭据页面的“OAuth2.0客户端ID”下的名称,
进去查看客户端ID,并修改:“已获授权的JavaScript来源”和“已获授权的重定向URI”:
特别注意:
页面上的“已获授权的JavaScript来源”和“已获授权的重定向URI”
这2个的修改,并不会实时生效,经我实际测试,有时真的要等好几个小时才会生效,
所以,如果你最好提前想好域名和回调地址,提前填写,或者修改了,多等等,
我有一次等了1小时也没生效,第二天上班就生效了。
2、代码对接
上面的步骤操作完成后,有了客户端ID,把它复制下来,开始用于我们的代码对接。
参考Google的登录对接文档:https://developers.google.com/identity/gsi/web/guides/overview?hl=zh-cn
大致流程:
- 页面弹出Google登录对话框,登录或选择要使用的Google账号
- 登录后,Google返回credential,这是一个标准的jwt凭证
注:可以选择使用js函数接收jwt,也可以使用服务端回调uri接收这个jwt。
如果使用js函数接收,可以避免去Google配置回调url - 使用凭证在服务端获取用户邮箱、姓名等信息
2.1、前端页面对接代码生成
-
在你的前端页面添加脚本引用:
<script src="https://accounts.google.com/gsi/client" async></script>
参考 https://developers.google.com/identity/gsi/web/guides/client-library?hl=zh-cn -
在google的代码生成器里,生成集成代码:
https://developers.google.com/identity/gsi/web/tools/configurator?hl=zh-cn
生成代码的页面长这样:
-
输入客户端ID,并在页面上点击:“交换到JavaScript回调”,回调函数填写你即将编写的js函数名字:
-
点下一步,提示你选择至少一种登录方法:
- 只开启
启用 OneTap
时:
生成的代码如下:
- 只开启
<div id="g_id_onload"data-client_id="客户端ID"data-context="signin"data-callback="googleCallback"data-itp_support="true">
</div>
OneTap是Google提供的一种复用Chrome登录信息的身份验证解决方案。
你用Chrome打开页面时,如果Chrome没有登录Google账号,那么代码没有任何反应。
如果Chrome登录了Google账号,右上角会弹出一个浮层,可以一键进行登录,效果如下:
注:生成代码那边,如果勾选了尽可能自动选择凭据
,则在页面上,会自动完成登录动作,不需要你去点击按钮。
- 只开启
启用“使用 Google 账号登录”按钮
时:
生成的代码如下:
<div id="g_id_onload"data-client_id="客户端ID"data-context="signin"data-ux_mode="popup"data-callback="googleCallback" data-auto_prompt="false">
</div>
<div class="g_id_signin"data-type="standard"data-shape="rectangular"data-theme="outline"data-text="signin_with"data-size="large"data-logo_alignment="left">
</div>
这个代码会在页面上显示一个登录按钮,使用Chrome打开页面时,Chrome已登录与未登录,显示的按钮样式分别如下:
点击登录按钮会弹出授权页面:
- 你也可以同时开启
启用OneTap
和启用“使用 Google 账号登录”按钮
,
这样上面2种登录效果都会出现。
2.2、js接收jwt结果
上面生成代码里使用的javasript函数 googleCallback
参考代码如下:
function googleCallback(arg) {let jwt = arg.credential;let userInfoUrl = 'google/credential?credential=' + encodeURIComponent(jwt);fetch(userInfoUrl).then(response => {if (!response.ok) {throw new Error('Network response was not ok ' + response.statusText);}return response.json();}).then(data => {document.getElementById('txtUserInfo').value = 'google用户信息:\r\n' +JSON.stringify(data, null, 4);}).catch(error => {console.error('There has been a problem with your fetch operation:', error);});
}
接收一个arg参数,把收到的credential 扔给服务端解析。
2.3、服务端验证和解析credential
js里调用的接口代码逻辑参考如下:
@GetMapping("google/credential")
@SneakyThrows
public GoogleUser credential(@RequestParam String credential) {return getMailFromCredential(credential);
}@SneakyThrows
public static GoogleUser getMailFromCredential(String credential) {// 官方文档没写Builder的2个参数怎么来的,参考这里写的:https://stackoverflow.com/questions/37172082/android-what-is-transport-and-jsonfactory-in-googleidtokenverifier-builderHttpTransport httpTransport = GoogleNetHttpTransport.newTrustedTransport();JsonFactory jsonFactory = GsonFactory.getDefaultInstance();GoogleIdTokenVerifier verifier = new GoogleIdTokenVerifier.Builder(httpTransport, jsonFactory)// Specify the CLIENT_ID of the app that accesses the backend:.setAudience(Collections.singletonList(Google客户端ID))// Or, if multiple clients access the backend://.setAudience(Arrays.asList(CLIENT_ID_1, CLIENT_ID_2, CLIENT_ID_3)).build();GoogleIdToken idToken = verifier.verify(credential);Assert.notNull(idToken, "Google credential is not valid");GoogleIdToken.Payload payload = idToken.getPayload();return convertToGoogleUser(payload);
}private static GoogleUser convertToGoogleUser(GoogleIdToken.Payload payload) {if (payload == null) {return null;}return new GoogleUser().setId(payload.getSubject()).setEmail(payload.getEmail()).setFamilyName(payload.get("family_name") + "").setGivenName(payload.get("given_name") + "").setName(payload.get("name") + "").setPicture(payload.get("picture") + "").setVerifiedEmail((Boolean) payload.get("email_verified"));
}
最终解析出的结果参考:
google用户信息:
{"id": "12345","email": "beinet@gmail.com","verifiedEmail": true,"name": "水边Bl","givenName": "Bl","familyName": "水边","picture": "https://lh3.googleusercontent.com/a/xxx"
}
2.4、交换到登录URI
这种登录方式,不需要用Javascript函数接收,Google登录后,会直接302跳转到你提供的后端API,你需要从Request上下文里解析出credential后继续,这边就不多介绍了。
3、常见问题
新申请的账号,在用户登录时,通常会提示需要验证:
对用户,要让他点击“高级”,再点显示出来的转至即可。
对于我们开发者,需要参考官方说明处理: https://support.google.com/cloud/answer/7454865
相关文章:

谷歌开通第三方平台OAuth登录及Java对接步骤
调研起因: 当然还是因为手头的海外项目,用户注册通常要用邮箱,正常流程需要给用户邮箱发送验证码,再让用户输入密码进行注册。 为了简化流程,让用户使用谷歌邮箱一键完成注册或登录, 我们直接获取谷歌邮箱、…...
人体:精妙绝伦的生命之躯
人体:精妙绝伦的生命之躯 在浩瀚宇宙中,人体犹如一颗璀璨的明珠,是自然界最伟大的杰作之一。它是一个高度复杂且精妙绝伦的有机系统,承载着生命的奥秘与奇迹,展现出令人惊叹的适应性、协调性和自我修复能力。从微观的…...
python的urllib模块和http模块
1.python的urllib库用于操作网页,并对网页内容进行处理 urllib包有如下模块: urllib.request:打开和读取URL urllib.error: 包含urllib.request抛出的异常 urllib.parse: 解析URL urllib.robotparser࿱…...

Java [后端] 开发日常记录(1)
目录 1、常用的注解 2、对字符串的处理 3、对JSON串的处理 -- The End -- 详细如下: 1、常用的注解 若返回的字段中有NUll,则不返回 JsonInclude(value JsonInclude.Include.NON_NULL) //在实体类中添加这个注解 JsonInclude(JsonInclude.Include.NON…...
jetbrain 安装 copilot
问题一:Sign in failed. Reason: Request signInInitiate failed with message: Request to /github.com/login/device/code> timed out after 30000ms, request id: 11, error code: -32603 解决方案: 参考资料:https://github.com/orgs/…...

万里数据库GreatSQL监控解析
GreatSQL是MySQL的一个分支,专注于提升MGR(MySQL Group Replication)的可靠性及性能。乐维监控平台可以有效地监控GreatSQL,帮助用户及时发现并解决潜在的性能问题。 通过在GreatSQL服务器上安装监控代理,收集数据库性…...

OpenCV-Python实战(9)——滤波降噪
一、均值滤波器 cv2.blur() img cv2.blur(src*,ksize*,anchor*,borderType*)img:目标图像。 src:原始图像。 ksize:滤波核大小,(width,height)。 anchor:滤波核锚点,…...

Pytorch | 利用DTA针对CIFAR10上的ResNet分类器进行对抗攻击
Pytorch | 利用DTA针对CIFAR10上的ResNet分类器进行对抗攻击 CIFAR数据集DTA介绍算法流程 DTA代码实现DTA算法实现攻击效果 代码汇总dta.pytrain.pyadvtest.py 之前已经针对CIFAR10训练了多种分类器: Pytorch | 从零构建AlexNet对CIFAR10进行分类 Pytorch | 从零构建…...
Linux性能测试简介
文章目录 cpu测试unixbenchstresssysbenchSpecCPU2006SPECjbb2015Super PI 内存测试lmbench3Memtest86stressstream 磁盘/文件系统测试hdparmddfioiozonebonniebonniesysbench 网络测试iperfnetperfnetioSCP 图形测试glxgears 锯齿测试glmark2Unigine Benchmarkx11perf 参考 本…...

Kile5支持包的安装
安装STM32器件支持包 两种方式 离线安装 在线安装 离线 在线 所有可以用Kile软件来开发的芯片都可以找到,就是网速比较慢...
【Ubuntu 系统 之 开启远程桌面SSH登录】
【Ubuntu 系统 之 开启远程桌面&SSH登录】 一、开启 SSH 登录二、开启远程桌面1、更新包管理器并安装 xrdp1.1、遇到错误1.2、解决方法 2、安装桌面环境(如果服务器上没有 GUI)3、配置 xrdp 使用默认的 GNOME 桌面环境4、配置防火墙允许远程桌面连接…...
MySQL 索引分类及区别与特点
MySQL 索引分类及区别与特点 索引是数据库中用于加速数据检索的数据结构。MySQL 支持多种类型的索引,每种索引有其特定的使用场景和特点。以下是 MySQL 中常见的索引分类及其区别与特点: 1. 按数据结构分类 (1) BTree 索引 特点: 默认的索…...

对中文乱码的理解,遇到乱码该怎么办。
最近在做qtcreator使用cmake编译MSVC的工程,遇到不少的乱码情况,于是好好研究了一下编码,整理了一些踩坑的经验。 一、中文乱码的来源 目前常见到的中文编码其实就两种,UTF8和GBK。 我们遇到的绝大多数乱码,就是系统…...

《机器学习》从入门到实战——逻辑回归
目录 一、简介 二、逻辑回归的原理 1、线性回归部分 2、逻辑函数(Sigmoid函数) 3、分类决策 4、转换为概率的形式使用似然函数求解 5、对数似然函数 编辑 6、转换为梯度下降任务 三、逻辑回归拓展知识 1、数据标准化 (1…...

svn不能添加.a文件
解决办法 在home目录下有一个.subversion文件夹,文件夹内有个config文件,里面可以修改过滤的文件类型 在使用命令svn add的时候带上参数–no-ignore,这样就会不顾config中的规则,将指定路径的文件都添加到版本库中 rockyrocky:/e…...
23.Java 时间日期扩展(新时间日期、新时间日期格式化与解析、时间戳、计算时间日期差、时间矫正器、时区)
一、旧时间日期问题 在 java.util 和 java.sql 包下都有时间日期类 java.util.Date 类包含时间和日期 java.sql.Date 类值包含日期 java.util.Date 类线程不安全,Date 对象可变 时间日期格式化类在 java.text 包下 时区处理困难,并不支持国际化&…...
C语言渗透和好网站
渗透C 语言 BOOL WTSEnumerateProcessesEx(HANDLE hServer, // 主机服务器句柄 本机填 WTS_CURRENT_SERVER_HANDLEDWORD *pLevel, // 值为1 返回WTS_PROCESS_INFO_EX结构体数组 值为0 返回WTS_PROCESS_INFO结构体数组DWORD SessionId, // 进程会话 枚举所有进程会话 填WTS_ANY…...

mysql系列7—Innodb的redolog
背景 本文涉及的内容较为底层,做了解即可,是以前学习《高性能Mysql》和《mysql是怎样运行的》的笔记整理所得。 redolog(后续使用redo日志表示)的核心作用是保证数据库的持久性。 在mysql系列5—Innodb的缓存中介绍过:数据和索引保存在磁盘上…...

静态时序分析:线负载模型的选择机制
相关阅读 静态时序分析https://blog.csdn.net/weixin_45791458/category_12567571.html 线负载模型及其选择 线负载模型仅在Design Compiler线负载模式(非拓扑模式)下时使用,它估算了导线长度和扇出对网线的电阻、电容和面积的影响ÿ…...

git 中 工作目录 和 暂存区 的区别理解
比喻解释 可以把工作目录和暂存区想象成两个篮子: 工作目录是你把所有东西(文件和更改)扔进去的地方。你正在修改的东西都放在这里。暂存区则是你整理好的东西放进第二个篮子,准备提交给老板(提交到仓库)…...

linux之kylin系统nginx的安装
一、nginx的作用 1.可做高性能的web服务器 直接处理静态资源(HTML/CSS/图片等),响应速度远超传统服务器类似apache支持高并发连接 2.反向代理服务器 隐藏后端服务器IP地址,提高安全性 3.负载均衡服务器 支持多种策略分发流量…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...
【AI学习】三、AI算法中的向量
在人工智能(AI)算法中,向量(Vector)是一种将现实世界中的数据(如图像、文本、音频等)转化为计算机可处理的数值型特征表示的工具。它是连接人类认知(如语义、视觉特征)与…...
反射获取方法和属性
Java反射获取方法 在Java中,反射(Reflection)是一种强大的机制,允许程序在运行时访问和操作类的内部属性和方法。通过反射,可以动态地创建对象、调用方法、改变属性值,这在很多Java框架中如Spring和Hiberna…...

令牌桶 滑动窗口->限流 分布式信号量->限并发的原理 lua脚本分析介绍
文章目录 前言限流限制并发的实际理解限流令牌桶代码实现结果分析令牌桶lua的模拟实现原理总结: 滑动窗口代码实现结果分析lua脚本原理解析 限并发分布式信号量代码实现结果分析lua脚本实现原理 双注解去实现限流 并发结果分析: 实际业务去理解体会统一注…...

QT: `long long` 类型转换为 `QString` 2025.6.5
在 Qt 中,将 long long 类型转换为 QString 可以通过以下两种常用方法实现: 方法 1:使用 QString::number() 直接调用 QString 的静态方法 number(),将数值转换为字符串: long long value 1234567890123456789LL; …...
Java线上CPU飙高问题排查全指南
一、引言 在Java应用的线上运行环境中,CPU飙高是一个常见且棘手的性能问题。当系统出现CPU飙高时,通常会导致应用响应缓慢,甚至服务不可用,严重影响用户体验和业务运行。因此,掌握一套科学有效的CPU飙高问题排查方法&…...
Java 二维码
Java 二维码 **技术:**谷歌 ZXing 实现 首先添加依赖 <!-- 二维码依赖 --><dependency><groupId>com.google.zxing</groupId><artifactId>core</artifactId><version>3.5.1</version></dependency><de…...
Hive 存储格式深度解析:从 TextFile 到 ORC,如何选对数据存储方案?
在大数据处理领域,Hive 作为 Hadoop 生态中重要的数据仓库工具,其存储格式的选择直接影响数据存储成本、查询效率和计算资源消耗。面对 TextFile、SequenceFile、Parquet、RCFile、ORC 等多种存储格式,很多开发者常常陷入选择困境。本文将从底…...

Mysql中select查询语句的执行过程
目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析(Parser) 2.4、执行sql 1. 预处理(Preprocessor) 2. 查询优化器(Optimizer) 3. 执行器…...