とりあえずコミット。認証とおらん(#47)(48)(#49)

This commit is contained in:
keita
2021-02-16 07:31:34 +09:00
parent 3a7976763a
commit 6fa7c457c0
16 changed files with 520 additions and 0 deletions
+34
View File
@@ -0,0 +1,34 @@
#ifndef COCOATWEET_OAUTH_KEY_H_
#define COCOATWEET_OAUTH_KEY_H_
#include <map>
namespace CocoaTweet::OAuth{
class Key{
const std::string consumerKey_;
const std::string consumerSecret_;
const std::string accessToken_;
const std::string accessTokenSecret_;
public:
Key() = default;
Key(const std::string& _consumerKey, const std::string& _consumerSecret, const std::string& _accessToken, const std::string& _accessTokenSecret):consumerKey_(_consumerKey), consumerSecret_(_consumerSecret), accessToken_(_accessToken), accessTokenSecret_(_accessTokenSecret){}
const std::string& consumerKey() const{return consumerKey_;}
const std::string& consumerSecret() const{return consumerSecret_;}
const std::string& accessToken() const{return accessToken_;}
const std::string& accessTokenSecret() const{return accessTokenSecret_;}
std::map<std::string, std::string> noSecret()const{
return std::map<std::string, std::string>{
{"oauth_consumer_key", consumerKey_},
{"oauth_token", accessToken_}
};
}
const std::map<std::string, std::string> secret()const{
return std::map<std::string, std::string>{
{"oauth_consumer_key", consumerSecret_},
{"oauth_token", accessTokenSecret_}
};
}
};
}
#endif
+131
View File
@@ -0,0 +1,131 @@
#include "oauth.h"
#include <random>
#include <ctime>
#include <bitset>
#include <sstream>
#include <string>
#include <iterator>
#include <iostream>
extern "C"{
#include <openssl/hmac.h>
#include <openssl/sha.h>
}
namespace CocoaTweet::OAuth{
OAuth1::OAuth1(){
}
OAuth1::OAuth1(const Key _key):key_(_key){
}
std::map<std::string, std::string> OAuth1::signature(const std::map<std::string, std::string>& _param, const std::string& _method, const std::string& _url){
std::vector<std::string> tmp;
for(const auto& [key, value] : _param){
tmp.push_back(key + "=" + value);
}
std::ostringstream os;
std::copy(tmp.begin(), tmp.end(), std::ostream_iterator<std::string>(os, "&"));
std::string query = os.str();
query.erase(query.size() - std::char_traits<char>::length("&"));
auto significateKey = key().consumerSecret() + "&" + key().accessTokenSecret();
auto significateBase = _method + "&" + _url + "&" + query;
auto result = hmacSha1(significateKey, significateBase);
std::cout << "significate key : " << significateKey << std::endl;
std::cout << "significate base : " << significateBase << std::endl;
std::cout << "hmac-sha1 : " << base64(result) << std::endl;
auto ret = std::map<std::string, std::string>{
{"oauth_signature", base64(result)}
};
return ret;
}
const std::string OAuth1::nonce()const{
std::random_device engine;
std::string nonceTable = "abcdefghijklmnopqrstuvwxyz0123456789";
std::uniform_int_distribution<std::size_t> dist(0, nonceTable.length() - 1);
std::string nonce;
for (auto i = 0; i < 32; ++i) {
nonce += nonceTable[dist(engine)];
}
return nonce;
}
const std::string OAuth1::timestamp() const{
return std::to_string(time(nullptr));
}
const std::string OAuth1::method() const{
return SIGNATURE_METHOD_;
}
const std::string OAuth1::version() const{
return OAUTH_VERSION_;
}
const Key OAuth1::key() const{
return key_;
}
std::map<std::string, std::string> OAuth1::oauthParam() const{
auto tmp = std::map<std::string, std::string>{
{"oauth_nonce", nonce()},
{"oauth_signature_method", method()},
{"oauth_timestamp", timestamp()},
{"oauth_version", version()}
};
tmp.merge(key().noSecret());
return tmp;
}
const std::string OAuth1::base64(const std::string& _raw){
auto base64Table = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
std::stringstream ss;
for(auto r : _raw){
ss << std::bitset<8>(r);
}
if(_raw.length() % 3 == 1){
ss << "0000";
}else if(_raw.length() % 3 == 2){
ss << "00";
}
auto bin = ss.str();
std::string base64 = "";
for(auto i = 0;i < bin.length() / 6;i++){
base64 += base64Table[std::stoi(bin.substr(i * 6, 6), nullptr, 2)];
}
if(base64.length() % 4 == 3){
base64 += "=";
}else if(base64.length() % 4 == 2){
base64 += "==";
}else if(base64.length() % 4 == 1){
base64 += "===";
}
return base64;
}
std::string OAuth1::hmacSha1(std::string _key, std::string _data){
unsigned char result[255];
unsigned int length = 255;
HMAC(EVP_sha1(), reinterpret_cast<const unsigned char*>(_key.c_str()), _key.length(), reinterpret_cast<const unsigned char*>(_data.c_str()), _data.length(), result, &length);
return std::string(reinterpret_cast<char*>(result));
}
}
+30
View File
@@ -0,0 +1,30 @@
#ifndef COCOATWEET_OAUTH_OAUTH_H_
#define COCOATWEET_OAUTH_OAUTH_H_
#include <map>
#include <memory>
#include "key.h"
namespace CocoaTweet::OAuth{
class OAuth1{
public:
OAuth1();
OAuth1(const Key _key);
std::map<std::string, std::string> signature(const std::map<std::string, std::string>& _param, const std::string& _method, const std::string& _url);
const std::string nonce()const;
const std::string timestamp()const;
const std::string method()const;
const std::string version()const;
const Key key()const;
std::map<std::string, std::string> oauthParam()const;
std::string hmacSha1(std::string _key, std::string _data);
const std::string base64(const std::string& _raw);
private:
Key key_;
const std::string SIGNATURE_METHOD_ = "HMAC-SHA1";
const std::string OAUTH_VERSION_ = "1.0";
};
}
#endif