青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
金慶的專欄
C++博客
::
首頁
::
新隨筆
::
聯系
::
聚合
::
管理
::
423 隨筆 :: 0 文章 :: 454 評論 :: 0 Trackbacks
公告
我的隨筆
我的評論
我參與的隨筆
留言簿
(12)
給我留言
查看公開留言
查看私人留言
隨筆分類
(502)
1. C/C++(166)
(rss)
2. 網游開發(108)
(rss)
3. Golang(20)
(rss)
4. Linux/Unix(30)
(rss)
5. 軟工與管理(44)
(rss)
6. Python(23)
(rss)
7. Erlang(18)
(rss)
8. Rust(16)
(rss)
9. 其它(77)
(rss)
隨筆檔案
(423)
2023年1月 (1)
2022年11月 (1)
2022年10月 (2)
2022年9月 (1)
2022年4月 (6)
2022年1月 (2)
2021年12月 (4)
2021年11月 (6)
2021年10月 (2)
2021年9月 (2)
2021年8月 (7)
2021年7月 (2)
2021年5月 (2)
2021年3月 (1)
2021年2月 (2)
2021年1月 (1)
2020年12月 (1)
2020年10月 (1)
2020年9月 (5)
2020年8月 (1)
2020年7月 (1)
2020年6月 (1)
2020年4月 (2)
2020年3月 (3)
2020年2月 (3)
2020年1月 (1)
2019年12月 (1)
2019年9月 (2)
2019年4月 (2)
2019年1月 (1)
2018年12月 (1)
2018年11月 (3)
2018年10月 (1)
2018年9月 (3)
2018年8月 (3)
2018年7月 (2)
2018年6月 (4)
2018年5月 (4)
2018年4月 (4)
2018年3月 (1)
2018年1月 (2)
2017年12月 (2)
2017年11月 (3)
2017年10月 (3)
2017年8月 (7)
2017年7月 (1)
2017年6月 (1)
2017年5月 (3)
2017年4月 (3)
2017年3月 (3)
2017年2月 (2)
2017年1月 (2)
2016年12月 (5)
2016年11月 (2)
2016年10月 (2)
2016年9月 (1)
2016年8月 (6)
2016年7月 (3)
2016年6月 (2)
2016年5月 (4)
2016年4月 (2)
2016年3月 (2)
2016年1月 (3)
2015年12月 (2)
2015年11月 (2)
2015年10月 (1)
2015年8月 (2)
2015年7月 (1)
2015年6月 (1)
2015年5月 (4)
2015年4月 (3)
2015年3月 (4)
2015年2月 (5)
2015年1月 (4)
2014年12月 (3)
2014年11月 (3)
2014年10月 (2)
2014年9月 (3)
2014年8月 (1)
2014年4月 (4)
2014年3月 (1)
2014年2月 (4)
2014年1月 (5)
2013年12月 (5)
2013年11月 (5)
2013年9月 (2)
2013年8月 (2)
2013年7月 (2)
2013年6月 (2)
2013年5月 (1)
2013年1月 (2)
2012年12月 (1)
2012年11月 (1)
2012年9月 (1)
2012年8月 (3)
2012年7月 (2)
2012年6月 (1)
2012年4月 (3)
2012年3月 (2)
2012年2月 (3)
2012年1月 (2)
2011年11月 (2)
2011年10月 (3)
2011年9月 (2)
2011年8月 (2)
2011年7月 (3)
2011年6月 (2)
2011年5月 (3)
2011年1月 (2)
2010年12月 (1)
2010年11月 (2)
2010年10月 (2)
2010年9月 (3)
2010年8月 (2)
2010年7月 (3)
2010年6月 (1)
2010年5月 (3)
2010年4月 (3)
2010年3月 (5)
2010年2月 (4)
2010年1月 (4)
2009年12月 (2)
2009年11月 (3)
2009年10月 (4)
2009年9月 (3)
2009年8月 (2)
2009年7月 (4)
2009年6月 (1)
2009年5月 (3)
2009年4月 (4)
2009年3月 (2)
2009年2月 (5)
2009年1月 (1)
2008年12月 (7)
2008年11月 (4)
2008年10月 (1)
2008年9月 (3)
2008年8月 (4)
2008年7月 (3)
2008年6月 (4)
2008年5月 (6)
2008年4月 (7)
2008年3月 (6)
2008年1月 (5)
2007年12月 (7)
2007年11月 (4)
2007年10月 (5)
2007年9月 (6)
2007年8月 (8)
2007年7月 (5)
相冊
公告照片
搜索
積分與排名
積分 - 662258
排名 - 25
最新評論
1.?re: boost::asio::spawn 將一統C++網絡庫
asio 成為C++首選網絡庫
--linda
2.?re: log4cxx中文輸出錯誤補丁
評論內容較長,點擊標題查看
--金慶
3.?re: mingw編譯OrzNet
能發送一個mingw編譯好的OrzNet庫給我嗎? liuweiqcxy@163.com
謝謝!
--劉威
4.?re: log4cxx中文輸出錯誤補丁
評論內容較長,點擊標題查看
--bigbad
5.?re: log4cxx中文輸出錯誤補丁
評論內容較長,點擊標題查看
--bigbad
閱讀排行榜
1.?"multiple definition of" 錯誤(11075)
2.?SVN中邪惡的replace(11000)
3.?VS2005編譯libevent(10456)
4.?混音算法的學習與研究(10258)
5.?C調用lua腳本的效率測試(9025)
評論排行榜
1.?VC6正在被拋棄(35)
2.?VS2005編譯libevent(21)
3.?"multiple definition of" 錯誤(18)
4.?C++引用優于指針(17)
5.?ACE與ASIO之間關于Socket編程的比較(16)
grpc加TLS加密和令牌認證
grpc加TLS加密和令牌認證
(金慶的專欄 2018.11)
用 golang 創建 grpc 服務,開啟 TLS 加密,并采用令牌認證。
然后用 C++ 和 golang 分別創建客戶端連接服務器。
參考:
https://segmentfault.com/a/1190000007933303
服務器
import (
...
grpc_auth "github.com/grpc-ecosystem/go-grpc-middleware/auth"
"google.golang.org/grpc"
"google.golang.org/grpc/credentials"
)
func main() {
listen, err := net.Listen("tcp", ":12345")
if err != nil {
grpclog.Fatalf("failed to listen: %v", err)
}
// TLS認證
creds, err := credentials.NewServerTLSFromFile("keys/server.crt", "keys/server.key")
if err != nil {
grpclog.Fatalf("Failed to generate credentials %v", err)
}
// 實例化grpc Server, 并開啟TLS認證
s := grpc.NewServer(grpc.Creds(creds),
grpc_auth.UnaryServerInterceptor(auth.Authenticate),
grpc_auth.StreamServerInterceptor(auth.Authenticate))
// 注冊HelloService
pb.RegisterHelloServer(s, HelloService)
grpclog.Println("Listen on " + Address + " with TLS")
s.Serve(listen)
}
其中 server.key 是私鑰,server.crt 是自簽名證書,如下生成:
$ openssl genrsa -out server.key 2048
$ openssl req -new -x509 -sha256 -key server.key \
-out server.crt -days 36500 \
-subj /C=CN/ST=Shanghai/L=Songjiang/O=ztgame/OU=tech/CN=mydomain.ztgame.com/emailAddress=myname@ztgame.com
查看證書文件
$ openssl x509 -in server.crt -noout -text
`auth.Authenticate` 如下,作為 interceptor, 對每個請求進行令牌驗證。
package auth
import (
"context"
"sync"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/metadata"
"google.golang.org/grpc/status"
)
// from token.yaml file
var tokenToAppName = &sync.Map{}
func init() {
tokenToAppName.Store("test", "test")
}
// XXX load tokenToAppName from file
// Authenticate checks that a token exists and is valid.
// It removes the token from the context and
// stores the app name of the token in the returned context
func Authenticate(ctx context.Context) (context.Context, error) {
token, err := extractHeader(ctx, "authorization-token")
if err != nil {
return ctx, err
}
// Remove token from headers from here on
ctx = purgeHeader(ctx, "authorization-token")
valAppName, ok := tokenToAppName.Load(token)
if !ok {
return ctx, status.Errorf(codes.Unauthenticated, "no app for token '%s'", token)
}
appName := valAppName.(string)
return context.WithValue(ctx, keyAppName{}, appName), nil
}
func extractHeader(ctx context.Context, header string) (string, error) {
md, ok := metadata.FromIncomingContext(ctx)
if !ok {
return "", status.Error(codes.Unauthenticated, "no headers in request")
}
authHeaders, ok := md[header]
if !ok {
return "", status.Error(codes.Unauthenticated, "no header in request")
}
if len(authHeaders) != 1 {
return "", status.Error(codes.Unauthenticated, "more than 1 header in request")
}
return authHeaders[0], nil
}
func purgeHeader(ctx context.Context, header string) context.Context {
md, _ := metadata.FromIncomingContext(ctx)
mdCopy := md.Copy()
mdCopy[header] = nil
return metadata.NewIncomingContext(ctx, mdCopy)
}
type keyAppName struct{}
// GetAppName can be used to extract app name stored in a context.
func GetAppName(ctx context.Context) string {
// Authenticate()之后必然存在app name
return ctx.Value(keyAppName{}).(string)
}
`tokenToAppName` 是一個map, 將合法的令牌映射為應用名。
每個應用(即用戶)分配一個令牌,根據令牌可查到該用戶是否合法,以及用戶的其他信息。
這里只需要應用名。
每個請求將調用 `Authenticate()`, 該方法將從 http 頭獲取請求的令牌,查找對應的應用名,
ctx 中將刪除令牌,替換成應用名。
`GetAppName()`將從 ctx 中獲取應用名。
服務方法實現如下:
func (s MailServer) Get(ctx context.Context, r *pb.GetRequest) (*pb.GetResponse, error) {
app := auth.GetAppName(ctx)
body, err := db.NewGetter(app).GetMailBody(r.MailIndex)
return &pb.GetResponse{
Result: getResult(err),
Body: body,
}, nil
}
先獲取應用名,然后根據應用名獲取相應的數據返回。
客戶端
golang
// Create the client TLS credentials
creds, err := credentials.NewClientTLSFromFile("key/server.crt", "mydomain.ztgame.com")
if err != nil {
panic(fmt.Errorf("could not load tls cert: %s", err))
}
// We don't need to error here, as this creates a pool and connections
// will happen later
conn, _ := grpc.Dial(
serviceURL,
grpc.WithTransportCredentials(creds),
grpc.WithPerRPCCredentials(auth.TokenAuth{
Token: "test",
}))
cli := pb.NewMailClient(conn)
客戶端只需要 server.crt, 其中包含服務器的公鑰。
NewClientTLSFromFile() 的第2個參數是個域名,是 server.crt 中的域名。
目前測試階段還沒有正式域名設置,所以輸入一個指定域名用于驗證 server.crt 中的域名。
生產環境運行時,應該不需要這個域名,可以直接查詢 DNS 進行驗證。
Dial() 輸入一個 WithPerRPCCredentials 用于令牌驗證。
auth.TokenAuth 需要實現 PerRPCCredentials 接口:
package auth
import (
"context"
)
type TokenAuth struct {
Token string
}
func (t TokenAuth) GetRequestMetadata(ctx context.Context, in ...string) (map[string]string, error) {
return map[string]string{
"authorization-token": t.Token,
}, nil
}
func (TokenAuth) RequireTransportSecurity() bool {
return true
}
"authorization-token" 是客戶端和服務器約定好的http認證頭字符串。
C++
C++ 端的客戶端代碼比golang的稍復雜,因為 grpc C++ 庫沒有 grpc-go 成熟。
代碼參照 grpc 示例 greeter_async_client2.cc:
int main(int argc, char** argv) {
grpc::SslCredentialsOptions ssl_options;
ssl_options.pem_root_certs = SERVER_CRT;
// Create a default SSL ChannelCredentials object.
auto channel_creds = grpc::SslCredentials(ssl_options);
grpc::ChannelArguments cargs;
cargs.SetSslTargetNameOverride("gamemail.ztgame.com"); // 如果加了 DNS 就不用這個了
auto call_creds = grpc::MetadataCredentialsFromPlugin(
std::unique_ptr<grpc::MetadataCredentialsPlugin>(new TokenAuthenticator(TOKEN)));
auto compsited_creds = grpc::CompositeChannelCredentials(channel_creds, call_creds);
// Create a channel using the credentials created in the previous step.
auto channel = grpc::CreateCustomChannel("1.2.3.4:8000", compsited_creds, cargs);
// Instantiate the client.
MailClient tester(channel);
...
return 0;
}
因為 C++ 沒有提供從文件讀取 server.crt 的接口,所以在此直接用了一個常量字符串:
ssl_options.pem_root_certs = SERVER_CRT;
SERVER_CRT 定義如下:
// server.crt 的內容
const char SERVER_CRT[] = R"(
-----BEGIN CERTIFICATE-----
TjERMA8GA1UECAwIU2hhbmdoYWkxEjAQBgNVBAcMCVNvbmdqaWFuZzEPMA0GA1UE
...
E6v50RCQgtWGmna+oy1I2UTVABdjBFnyKPEuz106mBfOhT6cg80hBHVgrV7sLHq8
76QolJm8yzZPL1qpiO4dKHHsCP6R
-----END CERTIFICATE-----
)";
TokenAuthenticator 定義如下,是個自定義認證插件:
// TokenAuthenticator 用來支持令牌認證
// https://grpc.io/docs/guides/auth.html
class TokenAuthenticator : public grpc::MetadataCredentialsPlugin {
public:
TokenAuthenticator(const std::string& token) : token_(token) {}
grpc::Status GetMetadata(
grpc::string_ref service_url, grpc::string_ref method_name,
const grpc::AuthContext& channel_auth_context,
std::multimap<grpc::string, grpc::string>* metadata) override {
metadata->insert(std::make_pair("authorization-token", token_));
return grpc::Status::OK;
}
private:
std::string token_;
};
posted on 2018-11-26 10:39
金慶
閱讀(2343)
評論(0)
編輯
收藏
引用
只有注冊用戶
登錄
后才能發表評論。
網站導航:
博客園
IT新聞
BlogJava
博問
Chat2DB
管理
Powered by:
C++博客
Copyright © 金慶
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
日韩视频在线你懂得
|
在线视频欧美一区
|
亚洲深夜福利网站
|
99在线精品视频在线观看
|
91久久嫩草影院一区二区
|
亚洲国产精品一区二区第一页
|
影音先锋久久
|
亚洲精品免费在线
|
一卡二卡3卡四卡高清精品视频
|
欧美韩日一区二区三区
|
欧美黄色一级视频
|
国产精品第一页第二页第三页
|
国产精品视频网址
|
悠悠资源网久久精品
|
日韩小视频在线观看专区
|
亚洲一级高清
|
免费日韩成人
|
一级日韩一区在线观看
|
欧美在线一二三四区
|
免费不卡亚洲欧美
|
国产精品久久久久久妇女6080
|
国产精品拍天天在线
|
亚洲国产日韩一级
|
亚洲综合视频在线
|
欧美不卡视频一区
|
亚洲欧美www
|
欧美久久婷婷综合色
|
国产亚洲一级高清
|
在线中文字幕一区
|
欧美成人精品h版在线观看
|
一区二区三区四区国产
|
美女视频黄 久久
|
国产亚洲午夜高清国产拍精品
|
一区二区三区四区精品
|
久久亚洲一区二区
|
亚洲欧美日韩天堂一区二区
|
欧美大尺度在线
|
国产一区白浆
|
午夜视频一区在线观看
|
亚洲精品乱码久久久久
|
亚洲一区二区3
|
欧美另类视频在线
|
亚洲精品国产视频
|
欧美成人激情在线
|
久久精品国语
|
国产午夜精品理论片a级探花
|
欧美高清视频一区二区
|
久久影音先锋
|
欧美日韩一区二区三区视频
|
国内外成人在线视频
|
亚洲自拍偷拍网址
|
亚洲靠逼com
|
欧美激情视频给我
|
亚洲激情亚洲
|
欧美岛国激情
|
久久亚洲国产精品日日av夜夜
|
国产伦精品一区二区三区免费
|
亚洲免费成人av电影
|
欧美成人午夜免费视在线看片
|
欧美亚洲一区三区
|
亚洲图片你懂的
|
欧美视频在线免费看
|
av成人免费在线
|
91久久夜色精品国产九色
|
久久久五月婷婷
|
在线电影一区
|
亚洲国产日韩在线一区模特
|
久热精品视频在线观看
|
亚洲国产高清一区
|
亚洲国产经典视频
|
欧美伦理一区二区
|
亚洲特色特黄
|
午夜精品福利在线观看
|
国产一区二区三区久久久
|
久久噜噜噜精品国产亚洲综合
|
一区二区三区毛片
|
国产精品视频一二三
|
久久久www免费人成黑人精品
|
国产精品视屏
|
久久久亚洲国产美女国产盗摄
|
欧美在线视频一区二区三区
|
国产综合18久久久久久
|
欧美99久久
|
欧美区日韩区
|
久久精品欧美日韩
|
母乳一区在线观看
|
亚洲一级在线观看
|
性欧美在线看片a免费观看
|
极品少妇一区二区三区
|
亚洲国产日日夜夜
|
国产亚洲精品久久飘花
|
亚洲第一偷拍
|
国产精品免费一区二区三区观看
|
久久精品五月婷婷
|
欧美日韩成人
|
久久人人爽人人爽爽久久
|
欧美劲爆第一页
|
久久国产精品第一页
|
欧美成人在线免费观看
|
久久大逼视频
|
欧美日韩亚洲网
|
久久网站免费
|
国产精品二区二区三区
|
久久综合色天天久久综合图片
|
亚洲经典在线看
|
久久免费精品视频
|
欧美韩日一区二区
|
久久久久久久欧美精品
|
欧美日韩激情网
|
欧美xx视频
|
国产在线视频欧美
|
亚洲一区二区三区高清不卡
|
亚洲精品护士
|
久久天堂成人
|
久久乐国产精品
|
国产精品一卡二卡
|
亚洲伦理在线
|
夜夜嗨av色综合久久久综合网
|
久久av在线看
|
欧美影院视频
|
国产精品乱码一区二区三区
|
亚洲区一区二区三区
|
在线播放亚洲一区
|
久久成人18免费网站
|
欧美一区二区视频网站
|
国产精品hd
|
这里只有精品视频在线
|
制服丝袜激情欧洲亚洲
|
欧美激情2020午夜免费观看
|
在线视频亚洲欧美
|
久久国产一区二区
|
夜夜嗨av色综合久久久综合网
|
欧美日本一道本
|
99国产精品视频免费观看
|
国产亚洲aⅴaaaaaa毛片
|
一本一本久久
|
一区二区三区国产
|
欧美高清在线一区二区
|
欧美大尺度在线
|
亚洲高清资源
|
欧美~级网站不卡
|
欧美激情a∨在线视频播放
|
极品裸体白嫩激情啪啪国产精品
|
午夜日韩福利
|
久久免费精品视频
|
亚洲电影免费在线
|
欧美**字幕
|
亚洲精品国产精品乱码不99
|
亚洲国产精品久久久久婷婷老年
|
久久久国产视频91
|
久久国产精品免费一区
|
亚洲视频免费在线
|
久久久福利视频
|
免费国产自线拍一欧美视频
|
亚洲欧美国产高清va在线播
|
亚洲欧美日韩一区二区在线
|
久久精视频免费在线久久完整在线看
|
国产精品免费在线
|
午夜一级在线看亚洲
|
久久精品理论片
|
亚洲风情亚aⅴ在线发布
|
欧美成人午夜剧场免费观看
|
亚洲国产一区二区视频
|
一本色道久久99精品综合
|
久久精品视频99
|
欧美第十八页
|
久久久久国产一区二区
|
亚洲日本乱码在线观看
|
国产精品三上
|
亚洲精品自在久久
|
国产日韩欧美高清
|
亚洲狼人综合
|
午夜视频一区在线观看
|
欧美精品 日韩
|
亚洲天堂成人在线视频
|
久久成人精品一区二区三区
|
亚洲黄色在线
|
国产精品久久久久久久浪潮网站
|
亚洲一区二区av电影
|
久热国产精品
|
亚洲综合三区
|
亚洲人体一区
|
国内精品嫩模av私拍在线观看
|
欧美激情1区2区3区
|
亚洲在线成人精品
|
欧美激情欧美激情在线五月
|
亚洲欧美激情一区
|
亚洲国产人成综合网站
|
国产目拍亚洲精品99久久精品
|
欧美成人在线免费观看
|
欧美一级久久
|
亚洲视频网站在线观看
|
欧美激情亚洲一区
|
久久综合精品一区
|
欧美在线播放
|
亚洲自拍偷拍福利
|
亚洲伦伦在线
|
亚洲激情网址
|
136国产福利精品导航网址
|
国产精品资源在线观看
|