青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
金慶的專欄
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)
相冊
公告照片
搜索
積分與排名
積分 - 662300
排名 - 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 © 金慶
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
国产欧美日韩免费
|
久久久久久久久久久一区
|
久久视频免费观看
|
巨乳诱惑日韩免费av
|
99成人在线
|
在线亚洲一区二区
|
国产精品久久夜
|
裸体丰满少妇做受久久99精品
|
免费日韩av
|
先锋影院在线亚洲
|
免费不卡在线观看av
|
亚洲欧美日韩成人高清在线一区
|
久久电影一区
|
宅男噜噜噜66一区二区66
|
久久国产精品一区二区三区
|
亚洲精品一区中文
|
香蕉成人伊视频在线观看
|
亚洲精品美女免费
|
欧美专区福利在线
|
亚洲天堂免费在线观看视频
|
久久九九久精品国产免费直播
|
99视频一区二区
|
久久精品国产亚洲aⅴ
|
99日韩精品
|
久久久综合精品
|
欧美亚洲午夜视频在线观看
|
欧美第一黄网免费网站
|
久久婷婷国产综合尤物精品
|
欧美日韩一区二区在线观看
|
欧美chengren
|
国产伦精品一区二区三区高清
|
亚洲人成毛片在线播放
|
国产欧美成人
|
中文在线不卡
|
亚洲第一在线视频
|
国产欧美va欧美不卡在线
|
欧美理论电影在线播放
|
亚洲国产网站
|
久久久夜夜夜
|
欧美一区二区高清
|
中文日韩欧美
|
9人人澡人人爽人人精品
|
在线观看91精品国产入口
|
国产精品嫩草99av在线
|
国产精品国产三级国产专区53
|
欧美黑人在线观看
|
欧美大胆人体视频
|
久久综合九色综合欧美狠狠
|
性视频1819p久久
|
小黄鸭视频精品导航
|
午夜精品免费视频
|
午夜欧美不卡精品aaaaa
|
亚洲综合色视频
|
性伦欧美刺激片在线观看
|
欧美在线观看视频一区二区三区
|
久久精品女人的天堂av
|
欧美成年网站
|
在线亚洲自拍
|
亚洲一品av免费观看
|
在线亚洲+欧美+日本专区
|
亚洲一区中文
|
久久久免费精品视频
|
欧美人与禽猛交乱配
|
国产日韩在线亚洲字幕中文
|
亚洲精品1区2区
|
亚洲欧美日韩国产中文在线
|
免费久久99精品国产
|
日韩视频中文字幕
|
亚洲一区三区视频在线观看
|
亚洲女人天堂av
|
老司机免费视频一区二区
|
亚洲第一在线综合网站
|
老巨人导航500精品
|
亚洲免费高清视频
|
久久免费视频网
|
国产视频欧美视频
|
亚洲视频1区2区
|
欧美激情按摩在线
|
久久成人国产
|
国产尤物精品
|
久久精品中文
|
午夜精品福利一区二区蜜股av
|
欧美成人一区二区三区在线观看
|
欧美大片免费
|
韩日成人在线
|
久久国产福利
|
性视频1819p久久
|
国产综合久久久久久
|
亚洲日本电影
|
欧美国产日韩一二三区
|
久久深夜福利免费观看
|
136国产福利精品导航网址应用
|
欧美在线免费
|
久久99伊人
|
亚洲激情在线播放
|
99成人在线
|
国产亚洲a∨片在线观看
|
欧美一区二区视频免费观看
|
亚洲女爱视频在线
|
伊人久久成人
|
国产精品久久九九
|
国产精品videosex极品
|
亚洲少妇在线
|
亚洲欧美久久久久一区二区三区
|
亚洲丶国产丶欧美一区二区三区
|
久久精品视频在线观看
|
另类图片国产
|
免费一级欧美片在线观看
|
亚洲国产小视频
|
99国产一区
|
亚洲欧美一区二区激情
|
一区二区三区我不卡
|
亚洲调教视频在线观看
|
欧美午夜片在线免费观看
|
久久久久久久综合日本
|
久久一区国产
|
午夜激情综合网
|
久热爱精品视频线路一
|
亚洲一区二区视频在线观看
|
欧美成人性网
|
亚洲视频axxx
|
欧美成人午夜影院
|
久久一区中文字幕
|
欧美日韩精品在线
|
欧美成人免费va影院高清
|
国产女人aaa级久久久级
|
亚洲人成7777
|
亚洲乱码国产乱码精品精天堂
|
欧美日韩精品免费观看视一区二区
|
亚洲一区二区精品
|
欧美手机在线视频
|
久久久精品五月天
|
国产精品一区二区男女羞羞无遮挡
|
亚洲福利视频一区
|
亚洲国产日韩欧美综合久久
|
久久久久久亚洲精品杨幂换脸
|
欧美中文字幕在线视频
|
久久精品国产亚洲a
|
国产亚洲一级
|
久久亚洲不卡
|
最新国产成人av网站网址麻豆
|
亚洲国产精品一区二区第四页av
|
国内成人在线
|
久久国产精品一区二区三区
|
国产精品日韩久久久久
|
在线亚洲免费
|
欧美午夜精品电影
|
亚洲一区二区欧美
|
久久女同互慰一区二区三区
|
在线观看三级视频欧美
|
欧美77777
|
亚洲理伦在线
|
国内精品久久久久久影视8
|
欧美aⅴ99久久黑人专区
|
一本色道久久88精品综合
|
欧美专区福利在线
|
亚洲精品一区二区三区av
|
欧美天堂亚洲电影院在线播放
|
亚洲欧美欧美一区二区三区
|
欧美freesex交免费视频
|
亚洲一二三级电影
|
黑人中文字幕一区二区三区
|
欧美精品一区二区蜜臀亚洲
|
亚洲欧美日韩精品久久久久
|
亚洲国产精品嫩草影院
|
久久精品国产综合精品
|
亚洲砖区区免费
|
日韩视频在线播放
|
黄网站色欧美视频
|
国产三级欧美三级日产三级99
|
欧美高清视频免费观看
|
欧美在线亚洲
|
欧美一区在线看
|
西西人体一区二区
|
亚洲免费在线观看
|
这里只有精品丝袜
|
中日韩高清电影网
|
av成人免费在线
|
99精品国产热久久91蜜凸
|
91久久精品国产91久久
|
欧美国产日韩精品
|
亚洲国产精品一区二区www在线
|
欧美成va人片在线观看
|
男女视频一区二区
|
亚洲第一二三四五区
|
亚洲欧洲三级电影
|
妖精成人www高清在线观看
|
99精品视频一区
|
亚洲图片欧美日产
|
久久riav二区三区
|
女人色偷偷aa久久天堂
|
欧美日韩视频在线一区二区
|
欧美日韩一区在线
|
国产专区欧美专区
|
99成人精品
|
麻豆久久婷婷
|
av不卡在线
|
另类欧美日韩国产在线
|
国产精品久久久久久久一区探花
|
一区二区三区导航
|