青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
金慶的專欄
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)
相冊
公告照片
搜索
積分與排名
積分 - 663333
排名 - 24
最新評論
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" 錯誤(11078)
2.?SVN中邪惡的replace(11003)
3.?VS2005編譯libevent(10462)
4.?混音算法的學習與研究(10265)
5.?C調用lua腳本的效率測試(9027)
評論排行榜
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
金慶
閱讀(2346)
評論(0)
編輯
收藏
引用
只有注冊用戶
登錄
后才能發表評論。
網站導航:
博客園
IT新聞
BlogJava
博問
Chat2DB
管理
Powered by:
C++博客
Copyright © 金慶
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
欧美日韩一区二区在线观看
|
另类图片综合电影
|
中国亚洲黄色
|
亚洲免费视频一区二区
|
午夜精品久久久久久久99热浪潮
|
欧美在线播放高清精品
|
久久精品日韩一区二区三区
|
久久亚洲精选
|
欧美日韩国产综合视频在线观看中文
|
国产精品美女久久久久久久
|
国产精品成人aaaaa网站
|
国产精品毛片
|
在线日韩中文字幕
|
亚洲视频在线观看免费
|
久久久精品免费视频
|
欧美高清视频免费观看
|
一区二区三区四区五区视频
|
久久国产精品第一页
|
欧美不卡在线
|
美女网站在线免费欧美精品
|
亚洲精品一区二区三区四区高清
|
亚洲欧美日韩国产综合
|
欧美电影资源
|
91久久精品国产91性色
|
久久伊伊香蕉
|
性做久久久久久久免费看
|
欧美日韩免费精品
|
亚洲人线精品午夜
|
欧美国产免费
|
免费亚洲电影在线观看
|
在线看一区二区
|
开心色5月久久精品
|
久久精品亚洲一区二区
|
国产婷婷成人久久av免费高清
|
亚洲综合精品
|
亚洲欧美日本另类
|
国产一区激情
|
猫咪成人在线观看
|
欧美岛国激情
|
亚洲午夜伦理
|
午夜精品在线看
|
国产一区二区高清
|
久久综合中文字幕
|
国产精品久久久久久久久久三级
|
亚洲大片一区二区三区
|
亚洲国产天堂网精品网站
|
午夜精品亚洲
|
一区二区高清
|
久久亚洲国产成人
|
永久555www成人免费
|
欧美一区二区三区在线看
|
亚洲精品国产拍免费91在线
|
欧美国产日本
|
香港久久久电影
|
久久香蕉精品
|
性欧美大战久久久久久久免费观看
|
亚洲欧美日本日韩
|
亚洲电影在线看
|
亚洲激情小视频
|
国产精品久久久一区麻豆最新章节
|
欧美喷水视频
|
久久中文欧美
|
欧美视频在线不卡
|
欧美电影在线观看完整版
|
欧美日韩一区高清
|
久久视频在线免费观看
|
欧美二区在线播放
|
亚洲欧美在线另类
|
欧美福利视频在线
|
欧美在线免费观看
|
另类av导航
|
亚洲综合成人在线
|
欧美久久久久
|
能在线观看的日韩av
|
国产精品久久久久毛片大屁完整版
|
久久手机精品视频
|
久久精品视频在线播放
|
欧美午夜免费
|
亚洲伦理中文字幕
|
99精品99久久久久久宅男
|
久久亚洲免费
|
免费观看亚洲视频大全
|
红桃视频一区
|
欧美永久精品
|
欧美在线二区
|
国产精自产拍久久久久久
|
这里只有精品丝袜
|
亚洲一区三区在线观看
|
欧美日韩精品二区第二页
|
亚洲国产欧美在线
|
亚洲精品午夜精品
|
欧美人与性禽动交情品
|
91久久在线观看
|
一本色道88久久加勒比精品
|
国产精品爱久久久久久久
|
亚洲免费视频在线观看
|
欧美专区一区二区三区
|
激情综合亚洲
|
欧美日韩三级视频
|
亚洲在线免费视频
|
久久婷婷麻豆
|
亚洲精品国精品久久99热一
|
欧美日韩福利在线观看
|
午夜精品久久久久久久99樱桃
|
久久成人18免费观看
|
亚洲高清不卡
|
国产精品久久一区二区三区
|
久久精品国产清自在天天线
|
亚洲国产精品女人久久久
|
欧美一区二区女人
|
亚洲一区二区三区精品视频
|
国产一区91精品张津瑜
|
欧美午夜寂寞影院
|
麻豆成人91精品二区三区
|
99视频精品全国免费
|
久久伊人精品天天
|
亚洲综合国产
|
亚洲激情专区
|
国模吧视频一区
|
国产精品影院在线观看
|
欧美国产日韩一区二区在线观看
|
欧美一级视频免费在线观看
|
9久re热视频在线精品
|
亚洲国产精品免费
|
亚洲精品1区
|
日韩天天综合
|
亚洲人成欧美中文字幕
|
欧美激情亚洲
|
欧美ab在线视频
|
欧美电影免费观看高清
|
免费人成网站在线观看欧美高清
|
久久久久高清
|
欧美大片免费观看在线观看网站推荐
|
亚洲自拍16p
|
一个色综合av
|
亚洲一区免费网站
|
亚洲免费视频一区二区
|
欧美一区在线视频
|
美女精品自拍一二三四
|
欧美影院在线
|
欧美精品aa
|
欧美天天影院
|
狠狠色2019综合网
|
亚洲肉体裸体xxxx137
|
夜夜嗨av一区二区三区四区
|
亚洲视频电影图片偷拍一区
|
久久久www成人免费毛片麻豆
|
久久亚洲午夜电影
|
久久在线免费
|
亚洲精品乱码久久久久久黑人
|
国产日本亚洲高清
|
9久草视频在线视频精品
|
亚洲综合不卡
|
欧美大片va欧美在线播放
|
亚洲三级网站
|
久久国产精品毛片
|
欧美三级在线播放
|
亚洲靠逼com
|
免费成人性网站
|
欧美亚洲一区二区在线观看
|
欧美二区在线
|
精品动漫3d一区二区三区免费版
|
亚洲一区二区三区高清
|
亚洲精品一区二区三
|
久久在线免费观看视频
|
国产日韩精品在线播放
|
亚洲一区二区三区四区中文
|
欧美成人一区二免费视频软件
|
亚洲夜间福利
|
欧美日韩精品一区
|
亚洲激情偷拍
|
亚洲人午夜精品免费
|
欧美一区二区三区在线观看
|
欧美午夜剧场
|
久久国产精品色婷婷
|
一区二区三区高清在线
|
欧美精品在线免费
|
亚洲精品美女91
|
亚洲日本国产
|
老司机凹凸av亚洲导航
|
91久久在线视频
|
99精品欧美
|
国产在线欧美
|
欧美69wwwcom
|
久久蜜桃资源一区二区老牛
|
**网站欧美大片在线观看
|
欧美粗暴jizz性欧美20
|
欧美另类videos死尸
|
亚洲在线中文字幕
|
欧美一区二区黄
|
亚洲日本中文字幕区
|
99成人精品
|
樱桃国产成人精品视频
|
欧美激情精品久久久久久黑人
|
欧美伊久线香蕉线新在线
|
亚洲黄色在线
|
亚洲视频你懂的
|
在线播放不卡
|
99亚洲一区二区
|
亚洲高清久久
|