如果你的目的是要使用 Gmail Api 取代舊的 Gmail SMTP 來發送通知信,建議你先跳到最下方看一下結論。
如果你是想要看一下 Gmail API 和 Google API 憑証的使用方法,可以看一下這篇文章。
1. 在 google cloud platform 建立新的專案.
https://console.cloud.google.com/
data:image/s3,"s3://crabby-images/3c4bd/3c4bd91939e377227d02bd0dd0eaf5e775e88d9e" alt=""
data:image/s3,"s3://crabby-images/86d4c/86d4cf0c51b4910c9ca65a937bc6118d82e44874" alt=""
data:image/s3,"s3://crabby-images/252f1/252f1805c75004b7e89511a2c2777caeddff2855" alt=""
啟用 Gmail API
data:image/s3,"s3://crabby-images/01730/01730f7c64060b428b2e89ba6cea8c7b89b438e7" alt=""
data:image/s3,"s3://crabby-images/41fc6/41fc61ec7eb29b40371bbe20e64c8ea3659eaca3" alt=""
data:image/s3,"s3://crabby-images/8a91d/8a91d311e619601aa50ad261a530653e23d34220" alt=""
data:image/s3,"s3://crabby-images/fc65d/fc65db9d1705ed6ea64467ef5340950e2e419a47" alt=""
data:image/s3,"s3://crabby-images/ff859/ff859110f15b331b2671b34fdb9aef752142a63f" alt=""
data:image/s3,"s3://crabby-images/c9ef2/c9ef2482e924f1b5934a911b3e8a82c819d04c4d" alt=""
data:image/s3,"s3://crabby-images/24804/24804cd16a24524f521628764ba7595f042baab9" alt=""
data:image/s3,"s3://crabby-images/09886/098868f6c6407fdc31578e5f8a215ae13ecc576f" alt=""
因為我們要透過 OAuth 取得使用者授權,所以要設定使用 OAuth 的同意畫面。
data:image/s3,"s3://crabby-images/f97e4/f97e4eabfebabcb535a077aa103e4a577a8194f4" alt=""
指定授權的範圍
data:image/s3,"s3://crabby-images/8c520/8c520846650cb1ad42334211e09adc80b0025306" alt=""
data:image/s3,"s3://crabby-images/c91e6/c91e6439bb6873a3f39412fd9fc17bb9486323df" alt=""
因為剛建立的專案,不會被公開,所以要指定測試使用者
data:image/s3,"s3://crabby-images/101cb/101cbd1f51aab025a14b7642010168d78c966169" alt=""
如果要給任意使用者,必需經過發布的流程,但準備工作有點麻煩,所以這次就不發布了。
data:image/s3,"s3://crabby-images/438e5/438e5b2a7922fafb54a620dee08bab0286e85a99" alt=""
建立 OAuth 2.0 用戶端 ID 憑証
data:image/s3,"s3://crabby-images/42497/4249750fe88e09656e0776f3fdad34529fadc7fa" alt=""
這裡除了名稱外,還有一個設定重導 Uri 的項目。現在不填寫,但稍後要回來補這個資料。
data:image/s3,"s3://crabby-images/0f2f4/0f2f45f904bbf3fa6cb7015fb08493eaa7b72ac9" alt=""
data:image/s3,"s3://crabby-images/81f77/81f7780507cc27255ab92ff7d980fd03f6105754" alt=""
下載 json 之後,命名為 client_secret.json 保留後續使用。
再來就要建立專案了. 用 VS2022 建立一個新專案
data:image/s3,"s3://crabby-images/9eb35/9eb353061607425ee1ecb8ce49d01e999e46a11e" alt=""
data:image/s3,"s3://crabby-images/d9226/d9226a3b1cac28ba62f081e976ffdc3823ddf217" alt=""
data:image/s3,"s3://crabby-images/1504a/1504a09c44a6535c5597d38a85732561bd8dfb4e" alt=""
data:image/s3,"s3://crabby-images/8e739/8e739692d53e8a14a7daebfb6b6c9cd19d68aa7b" alt=""
記錄網址, 本測試專案是 https://localhost:44340/ ,請依實際網址為準。
data:image/s3,"s3://crabby-images/4ff03/4ff03d441b1fd996ec69b3feb3dc1fedf1621d4f" alt=""
回到 OAuth 2.0 用戶端 ID 的設定頁. 在已授權的重新導向 URI 中填入 https://localhost:44340/Home/AuthReturn (填入的網址依實際專案的狀況,可能會有變化)
data:image/s3,"s3://crabby-images/2c156/2c156a799cb98e8be8494d57c208e25540f33ac6" alt=""
data:image/s3,"s3://crabby-images/d71ea/d71ea64a72529f8b351b374e2832ddd136e8394a" alt=""
在 VS2022 中,使用 Nuget 安裝套件: (有漏的再麻煩和我說)
Google.Apis.Gmail.v1
Google.Apis.Auth
MimeKit (發送 gmail 時使用)
建立認証用的網址:
建立一個 Action, 用來取得認証用的網址:
/// <summary>
/// 取得授權的項目
/// </summary>
static string[] Scopes = { GmailService.Scope.GmailSend };
// 和登入 google 的帳號無關
// 任意值,若未來有使用者認証,可使用使用者編號或登入帳號。
string Username = "ABC";
/// <summary>
/// 存放 client_secret 和 credential 的地方
/// </summary>
string SecretPath = @"D:\project\GmailTest\Data\Secrets";
/// <summary>
/// 認証完成後回傳的網址, 必需和 OAuth 2.0 Client Id 中填寫的 "已授權的重新導向 URI" 相同。
/// </summary>
string RedirectUri = $"https://localhost:44340/Home/AuthReturn";
/// <summary>
/// 取得認証用的網址
/// </summary>
/// <returns></returns>
public async Task<string> GetAuthUrl()
{
using (var stream = new FileStream(Path.Combine(SecretPath, "client_secret.json"), FileMode.Open, FileAccess.Read))
{
FileDataStore dataStore = null;
var credentialRoot = Path.Combine(SecretPath, "Credentials");
if (!Directory.Exists(credentialRoot))
{
Directory.CreateDirectory(credentialRoot);
}
//存放 credential 的地方,每個 username 會建立一個目錄。
string filePath = Path.Combine(credentialRoot, Username);
dataStore = new FileDataStore(filePath);
IAuthorizationCodeFlow flow = new GoogleAuthorizationCodeFlow(new GoogleAuthorizationCodeFlow.Initializer
{
ClientSecrets = GoogleClientSecrets.Load(stream).Secrets,
Scopes = Scopes,
DataStore = dataStore
});
var authResult = await new AuthorizationCodeWebApp(flow, RedirectUri, Username)
.AuthorizeAsync(Username, CancellationToken.None);
return authResult.RedirectUri;
}
}
執行結果:
data:image/s3,"s3://crabby-images/7a308/7a3086cdb65b730b8ca119e0ee4275bc7d2d6b95" alt=""
用 chrome 開啟產生的網址:
data:image/s3,"s3://crabby-images/ddbc5/ddbc5a7922d239edf09e20bcb21bd0af0dc4bacd" alt=""
選取任一帳號,如果出現以下錯誤,請回到 "OAuth 同意畫面" 去新增測試使用者
data:image/s3,"s3://crabby-images/a2302/a2302945cdc0a43b9d30541d25560119a9ef8eea" alt=""
data:image/s3,"s3://crabby-images/033c3/033c3b9a217124e1c275fc015c29f981933be8bc" alt=""
因為應用程式尚未發布,所以會看到警告,勇敢的繼續下去
data:image/s3,"s3://crabby-images/515b3/515b3f3efa60f0f8b3738ab50d05b2a0fac22582" alt=""
這裡會要求授權使用你的名義發送信件。(這是在程式中取得授權的項目 Scopes 中所指定的)
data:image/s3,"s3://crabby-images/61233/61233f88ff154cd85184cfa2d24b66c4e9218d02" alt=""
再繼續之後,會被重導至我們在 redirectUri 指定的網址。因為我們尚未完成,所以會看到錯誤,順便也可以看一下,會帶回哪一些參數。有 state, code, scope,共三個。
data:image/s3,"s3://crabby-images/142a2/142a22e1dda29b766499c2c2ab699548822558fc" alt=""
順便看一下,google 的套件會在 Credentials 的目錄下幫使用者建立一個目錄,在完成驗証前,會先放一個 System.String-oauth_XXX 的檔案,裡面的值和回傳的 state 是一樣的,這個應該是用來驗証回傳資料的。
data:image/s3,"s3://crabby-images/ae9e8/ae9e811ed8df477467d0c09c2376fc4dea80295e" alt=""
接下來我們要新增 Action "AuthReturn" 如下:
public async Task<string> AuthReturn(AuthorizationCodeResponseUrl authorizationCode)
{
string[] scopes = new[] { GmailService.Scope.GmailSend };
using (var stream = new FileStream(Path.Combine(SecretPath, "client_secret.json"), FileMode.Open, FileAccess.Read))
{
//確認 credential 的目錄已建立.
var credentialRoot = Path.Combine(SecretPath, "Credentials");
if (!Directory.Exists(credentialRoot))
{
Directory.CreateDirectory(credentialRoot);
}
//暫存憑証用目錄
string tempPath = Path.Combine(credentialRoot, authorizationCode.State);
IAuthorizationCodeFlow flow = new GoogleAuthorizationCodeFlow(
new GoogleAuthorizationCodeFlow.Initializer
{
ClientSecrets = GoogleClientSecrets.Load(stream).Secrets,
Scopes = scopes,
DataStore = new FileDataStore(tempPath)
});
//這個動作應該是要把 code 換成 token
await flow.ExchangeCodeForTokenAsync(Username, authorizationCode.Code, RedirectUri, CancellationToken.None).ConfigureAwait(false);
if (!string.IsNullOrWhiteSpace(authorizationCode.State))
{
string newPath = Path.Combine(credentialRoot, Username);
if (tempPath.ToLower() != newPath.ToLower())
{
if (Directory.Exists(newPath))
Directory.Delete(newPath, true);
Directory.Move(tempPath, newPath);
}
}
return "OK";
}
}
再跑一次上面的流程,最後回到 AuthReturn
data:image/s3,"s3://crabby-images/2d211/2d211adca50880dea781efadd5f3f7b199cab2c4" alt=""
在 D:\project\GmailTest\Data\Secrets\Credentials\ABC 裡面會產生一個檔案: 這個就是我們的 token 了。
data:image/s3,"s3://crabby-images/60d1a/60d1a81e200adaa972043c55c0d807937c2b6529" alt=""
看一下裡面的內容, 有 access_token, refresh_token, scope 等等, 用途應該很好猜了.. 不知道各項目的目途也沒有關係。只要有這個 token 就可以了。
data:image/s3,"s3://crabby-images/80efa/80efa663ba22e22f026310a3f459176ec7da33cb" alt=""
refresh_token 的效期請參考以下文件:
https://developers.google.com/identity/protocols/oauth2 。也可以參考下圖, 若是要用 gmail api 來發送通知信(例如連絡我們),紅色的地方是比較令人困擾的,例如 6 個月以上,沒有人留言,原來留下的 refresh_token 就失效了。使用者必需重新建立一個 refresh_token 。
data:image/s3,"s3://crabby-images/acb91/acb91e566fdf1fe11992ad6cdadbc5eb9ee62d9a" alt=""
最後來使用 gmail api 發送通知信, 直接看程式碼如下: 在這個過程中遇到最大的問題除了憑証問題之外,另一個問題是編碼。直到最後找到可以用 MimeKit 把 System.Net.Mail.MailMessage 編碼成 Gmail API 的格式才解決。程式碼如下:
public async Task<bool> SendTestMail()
{
var service = await GetGmailService();
GmailMessage message = new GmailMessage();
message.Subject = "標題";
message.Body = $"<h1>內容</h1>";
message.FromAddress = "bikehsu@gmail.com";
message.IsHtml = true;
message.ToRecipients = "bikehsu@gmail.com";
message.Attachments = new List<Attachment>();
string filePath = @"C:\Users\bike\Pictures\Vegetable_pumpkin.jpg"; //要附加的檔案
Attachment attachment1 = new Attachment(filePath);
message.Attachments.Add(attachment1);
SendEmail(message, service);
Console.WriteLine("OK");
return true;
}
async Task<GmailService> GetGmailService()
{
UserCredential credential = null;
var credentialRoot = Path.Combine(SecretPath, "Credentials");
if (!Directory.Exists(credentialRoot))
{
Directory.CreateDirectory(credentialRoot);
}
string filePath = Path.Combine(credentialRoot, Username);
using (var stream = new FileStream(Path.Combine(SecretPath, "client_secret.json"), FileMode.Open, FileAccess.Read))
{
credential = await GoogleWebAuthorizationBroker.AuthorizeAsync(
GoogleClientSecrets.Load(stream).Secrets,
Scopes,
Username,
CancellationToken.None,
new FileDataStore(filePath));
}
var service = new GmailService(new BaseClientService.Initializer()
{
HttpClientInitializer = credential,
ApplicationName = "Send Mail",
});
return service;
}
public class GmailMessage
{
public string FromAddress { get; set; }
public string ToRecipients { get; set; }
public string Subject { get; set; }
public string Body { get; set; }
public bool IsHtml { get; set; }
public List<System.Net.Mail.Attachment> Attachments { get; set; }
}
public static void SendEmail(GmailMessage email, GmailService service)
{
var mailMessage = new System.Net.Mail.MailMessage();
mailMessage.From = new System.Net.Mail.MailAddress(email.FromAddress);
mailMessage.To.Add(email.ToRecipients);
mailMessage.ReplyToList.Add(email.FromAddress);
mailMessage.Subject = email.Subject;
mailMessage.Body = email.Body;
mailMessage.IsBodyHtml = email.IsHtml;
if (email.Attachments != null)
{
foreach (System.Net.Mail.Attachment attachment in email.Attachments)
{
mailMessage.Attachments.Add(attachment);
}
}
var mimeMessage = MimeKit.MimeMessage.CreateFromMailMessage(mailMessage);
var gmailMessage = new Google.Apis.Gmail.v1.Data.Message
{
Raw = Encode(mimeMessage)
};
Google.Apis.Gmail.v1.UsersResource.MessagesResource.SendRequest request = service.Users.Messages.Send(gmailMessage, "me");
request.Execute();
}
public static string Encode(MimeMessage mimeMessage)
{
using (MemoryStream ms = new MemoryStream())
{
mimeMessage.WriteTo(ms);
return Convert.ToBase64String(ms.GetBuffer())
.TrimEnd('=')
.Replace('+', '-')
.Replace('/', '_');
}
}
收到的信件:
data:image/s3,"s3://crabby-images/019c5/019c5f0cc86bf764cf25d1e6c3ad544e0e9765e8" alt=""
結論:
使用 Gmail API 最大的原因是要增加安全性,和舊的 smtp 不同的地方是,使用 gmail api 之後,客戶不需要提供 gmail 的帳號和密碼就可以讓系統使用 gmail 發送信件,不過由於 refresh_token 的效期問題,可能會造成無法發送通知信而沒有任何人發現的情況,整個實用性會變的很低。
另一個還沒有測試的部份是應用程的啟用。這個審核不知道會不很麻煩,不過可以而知的時,整個流程會花更多的時間。
取代的做法: 可能要改用 Amazon 的 SES 來寄信,而且為了避免每個小網站都要跑 SES 的建立流程,準備來寫一個 API 給各網站使用,可以發送簡單的通知信。
以上的程式碼可以在這裡下載: https://github.com/bikehsu/GmailTest