basic認証用のプロバイダとベアラトークンを取得するためのエンドポイントアクセスを有効に

This commit is contained in:
keita
2022-04-23 18:00:41 +09:00
parent 560fe707a6
commit d99ba765aa
17 changed files with 300 additions and 67 deletions
+8 -2
View File
@@ -1,5 +1,7 @@
#include <cocoatweet/api/api.h>
#include <cocoatweet/authentication/authenticate.h>
#include <cocoatweet/authentication/bearer.h>
#include <cocoatweet/authentication/plain.h>
#include <iostream>
@@ -12,8 +14,8 @@ API::API(CocoaTweet::Authentication::Key _key) {
void API::swapKey(const CocoaTweet::Authentication::Key _key){
if(_key.authType() == CocoaTweet::Authentication::Key::OAUTH10A){
oauth_ = std::make_shared<CocoaTweet::Authentication::OAuth1>(_key);
}else{
}else if(_key.authType() == CocoaTweet::Authentication::Key::OAUTH2){
oauth_ = std::make_shared<CocoaTweet::Authentication::Bearer>(_key);
}
user_ = Users::User(oauth_);
status_ = Statuses::Status(oauth_);
@@ -21,6 +23,7 @@ void API::swapKey(const CocoaTweet::Authentication::Key _key){
media_ = Medias::Media(oauth_);
directMessage_ = DirectMessages::DirectMessage(oauth_);
oauth1_ = OAuth1::OAuth(oauth_);
oauth2_ = OAuth2::OAuth2(oauth_);
}
// const std::string& API::generateBearerToken() const {
@@ -49,4 +52,7 @@ DirectMessages::DirectMessage API::directMessage() const {
OAuth1::OAuth API::oauth1() const {
return oauth1_;
}
OAuth2::OAuth2 API::oauth2() const {
return oauth2_;
}
} // namespace CocoaTweet::API
+3
View File
@@ -8,6 +8,7 @@
#include <cocoatweet/api/directMessage/directMessage.h>
#include <cocoatweet/authentication/authenticator.h>
#include <cocoatweet/api/oauth1/oauth.h>
#include <cocoatweet/api/oauth2/oauth2.h>
namespace CocoaTweet::API {
/// @brief Twitter API Entry Point
@@ -32,6 +33,7 @@ public:
DirectMessages::DirectMessage directMessage() const;
OAuth1::OAuth oauth1() const;
OAuth2::OAuth2 oauth2() const;
const std::string& generateBearerToken() const;
void swapKey(const CocoaTweet::Authentication::Key _key);
@@ -43,6 +45,7 @@ private:
Medias::Media media_;
DirectMessages::DirectMessage directMessage_;
OAuth1::OAuth oauth1_;
OAuth2::OAuth2 oauth2_;
std::shared_ptr<CocoaTweet::Authentication::AuthenticatorBase> oauth_;
};
} // namespace CocoaTweet::API
+6
View File
@@ -6,6 +6,8 @@
#include <cocoatweet/exception/tweetTooLongException.h>
#include <cocoatweet/exception/rateLimitException.h>
#include <cocoatweet/exception/tokenInvalidException.h>
#include <cocoatweet/exception/missingRequiredParamException.h>
#include <cocoatweet/exception/credentialNotAllowedException.h>
#include "nlohmann/json.hpp"
#include <iterator>
#include <memory>
@@ -128,6 +130,10 @@ void HttpGet::process(std::weak_ptr<CocoaTweet::Authentication::AuthenticatorBas
throw CocoaTweet::Exception::RateLimitException(message.get<std::string>().c_str());
} else if (error.get<int>() == 186) {
throw CocoaTweet::Exception::TweetTooLongException(message.get<std::string>().c_str());
}else if(error.get<int>() == 170){
throw CocoaTweet::Exception::MissingRequiredParamException(message.get<std::string>().c_str());
}else if(error.get<int>() == 220){
throw CocoaTweet::Exception::CredentialNotAllowedException(message.get<std::string>().c_str());
} else {
}
}
+6
View File
@@ -6,6 +6,8 @@
#include <cocoatweet/exception/tweetTooLongException.h>
#include <cocoatweet/exception/rateLimitException.h>
#include <cocoatweet/exception/tokenInvalidException.h>
#include <cocoatweet/exception/missingRequiredParamException.h>
#include <cocoatweet/exception/credentialNotAllowedException.h>
#include "nlohmann/json.hpp"
#include <iterator>
#include <memory>
@@ -146,6 +148,10 @@ void HttpPost::process(std::weak_ptr<CocoaTweet::Authentication::AuthenticatorBa
throw CocoaTweet::Exception::RateLimitException(message.get<std::string>().c_str());
} else if (error.get<int>() == 186) {
throw CocoaTweet::Exception::TweetTooLongException(message.get<std::string>().c_str());
}else if(error.get<int>() == 170){
throw CocoaTweet::Exception::MissingRequiredParamException(message.get<std::string>().c_str());
}else if(error.get<int>() == 220){
throw CocoaTweet::Exception::CredentialNotAllowedException(message.get<std::string>().c_str());
}
}
+16
View File
@@ -0,0 +1,16 @@
#include <cocoatweet/api/oauth2/oauth2.h>
#include <cocoatweet/authentication/basic.h>
namespace CocoaTweet::API::OAuth2 {
OAuth2::OAuth2(std::shared_ptr<CocoaTweet::Authentication::AuthenticatorBase> _oauth){
oauth_ = _oauth;
}
const std::string OAuth2::token() const{
auto key = oauth_.lock()->key();
auto oauth = std::make_shared<CocoaTweet::Authentication::Basic>(key);
CocoaTweet::API::OAuth2::Token token;
return token.process(oauth);
}
} // namespace CocoaTweet::API::Statuses
+26
View File
@@ -0,0 +1,26 @@
#ifndef COCOATWEET_API_OAUTH2_OAUTH2_H_
#define COCOATWEET_API_OAUTH2_OAUTH2_H_
#include "cocoatweet/api/interface/groupInterface.h"
#include <cocoatweet/api/oauth2/token.h>
#include <cocoatweet/api/model/oauthToken.h>
#include <vector>
#include <utility>
namespace CocoaTweet::API::OAuth2 {
/// @brief class for using users/show endpoint
class OAuth2 : public groupInterface {
public:
/// @brief primary constructor
OAuth2() = default;
/// @brief constructor which finally should to be called.
/// @param[in] std::shared_ptr<CocoaTweet::Authentication::AuthenticatorBase> : pointer to OAuth object
OAuth2(std::shared_ptr<CocoaTweet::Authentication::AuthenticatorBase> _oauth);
const std::string token() const;
};
} // namespace CocoaTweet::API::Statuses
#endif
+25
View File
@@ -0,0 +1,25 @@
#include <cocoatweet/api/oauth2/token.h>
#include "nlohmann/json.hpp"
namespace CocoaTweet::API::OAuth2{
Token::Token(){
contentType_ = "application/x-www-form-urlencoded";
url_ = "https://api.twitter.com/oauth2/token";
bodyParam_.insert_or_assign("grant_type", "client_credentials");
}
const std::string Token::process(std::weak_ptr<CocoaTweet::Authentication::AuthenticatorBase> _oauth) {
auto basic = std::make_shared<CocoaTweet::Authentication::Basic>(_oauth.lock()->key());
std::string bearer = "";
HttpPost::process(basic, [&bearer](const std::string& _rcv) {
auto j = nlohmann::json::parse(_rcv);
bearer = j["access_token"];
});
return bearer;
}
}
+15
View File
@@ -0,0 +1,15 @@
#ifndef COCOATWEET_API_OAUTH2_TOKEN_H
#define COCOATWEET_API_OAUTH2_TOKEN_H
#include <cocoatweet/api/interface/httpPost.h>
#include <cocoatweet/authentication/basic.h>
namespace CocoaTweet::API::OAuth2{
class Token: public CocoaTweet::API::Interface::HttpPost{
public:
Token();
const std::string process(std::weak_ptr<CocoaTweet::Authentication::AuthenticatorBase> _oauth);
};
}
#endif
@@ -78,68 +78,6 @@ const std::string OAuth1::calculateAuthHeader(std::map<std::string, std::string>
return oauthHeader;
}
// const std::string& OAuth1::generateBearerToken() {
// auto signature = key_.consumerKey() + ":" + key_.consumerSecret();
// auto k64Signature = base64(signature);
// auto authHeader = std::string("Authorization: Basic ") + k64Signature;
// auto contentType =
// std::string("Content-Type: application/x-www-form-urlencoded;charset=UTF-8");
// auto url = std::string("https://api.twitter.com/oauth2/token");
// auto requestBody = std::string("grant_type=client_credentials");
// // do post
// CURL* curl;
// CURLcode res;
// std::string rcv;
// long responseCode;
// curl = curl_easy_init();
// if (curl) {
// curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
// curl_easy_setopt(curl, CURLOPT_POST, 1);
// curl_easy_setopt(curl, CURLOPT_POSTFIELDS, requestBody.c_str());
// curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, requestBody.length());
// curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
// curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, curlCallback_);
// curl_easy_setopt(curl, CURLOPT_WRITEDATA, (std::string*)&rcv);
// #ifndef NDEBUG
// std::cout << "requestBody : " << requestBody << std::endl;
// curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L);
// #endif
// // Headerを保持するcurl_slist*を初期化
// struct curl_slist* headers = NULL;
// // Authorizationをヘッダに追加
// headers = curl_slist_append(headers, authHeader.c_str());
// headers = curl_slist_append(headers, contentType.c_str());
// curl_easy_setopt(curl, CURLOPT_HTTPHEADER, headers);
// res = curl_easy_perform(curl);
// curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &responseCode);
// curl_easy_cleanup(curl);
// }
// if (res != CURLE_OK) {
// throw std::runtime_error(std::string("INTERNAL ERROR : curl(") + std::to_string(res) + ")");
// exit(1);
// }
// auto j = nlohmann::json::parse(rcv);
// if ((responseCode / 100) == 4) {
// auto error = j["errors"][0]["code"];
// auto message = j["errors"][0]["message"];
// if (j.count("error") != 0) {
// // この形式はエラーコードを持たないのでエラー種別が特定できない
// throw new CocoaTweet::Exception::Exception(j["error"]);
// }
// if (error.get<int>() == 44) {
// throw CocoaTweet::Exception::InvalidParameterException(
// message.get<std::string>().c_str());
// }
// }
// key_.bearerToken(j["access_token"]);
// authType_ = AuthType::Bearer;
// return key_.bearerToken();
// }
const std::string OAuth1::nonce() const {
std::random_device engine;
std::string nonceTable = "abcdefghijklmnopqrstuvwxyz0123456789";
@@ -8,6 +8,7 @@ public:
enum class AuthenticationMethod{
OAUTH10A,
OAUTH2,
BASIC,
PLAIN,
NONE
};
+77
View File
@@ -0,0 +1,77 @@
#include "basic.h"
#include "cocoatweet/util/util.h"
#include <random>
#include <ctime>
#include <bitset>
#include <sstream>
#include <string>
#include <cstring>
#include <iterator>
#include <nlohmann/json.hpp>
#include <cocoatweet/exception/invalidParameterException.h>
extern "C" {
#include <openssl/hmac.h>
#include <openssl/sha.h>
#include <openssl/buffer.h>
#include <curl/curl.h>
}
#ifndef NDEBUG
#include <iostream>
#endif
namespace CocoaTweet::Authentication {
Basic::Basic() {
method_ = AuthenticationMethod::BASIC;
}
Basic::Basic(const Key _key){
key_ = _key;
method_ = AuthenticationMethod::BASIC;
}
const std::string Basic::calculateAuthHeader(std::map<std::string, std::string> _bodyParam,
const std::string& _method,
const std::string& _url) {
auto signature = key_.consumerKey() + ":" + key_.consumerSecret();
auto k64Signature = base64(signature);
auto authHeader = std::string("Authorization: Basic ") + k64Signature;
return authHeader;
}
const std::string Basic::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;
}
} // namespace CocoaTweet::Authentication
+25
View File
@@ -0,0 +1,25 @@
#ifndef COCOATWEET_AUTHENTICATION_BASIC_H_
#define COCOATWEET_AUTHENTICATION_BASIC_H_
#include <string>
#include <map>
#include <memory>
#include "key.h"
#include <cocoatweet/authentication/authenticator.h>
namespace CocoaTweet::Authentication {
class Basic: public AuthenticatorBase {
public:
Basic();
Basic(const Key _key);
// const std::string& generateBearerToken();
const std::string calculateAuthHeader(std::map<std::string, std::string> _bodyParam,
const std::string& _method, const std::string& _url);
const std::string base64(const std::string& _raw);
};
} // namespace CocoaTweet::Authentication
#endif
+44
View File
@@ -0,0 +1,44 @@
#include "bearer.h"
#include "cocoatweet/util/util.h"
#include <random>
#include <ctime>
#include <bitset>
#include <sstream>
#include <string>
#include <cstring>
#include <iterator>
#include <nlohmann/json.hpp>
#include <cocoatweet/exception/invalidParameterException.h>
extern "C" {
#include <openssl/hmac.h>
#include <openssl/sha.h>
#include <openssl/buffer.h>
#include <curl/curl.h>
}
#ifndef NDEBUG
#include <iostream>
#endif
namespace CocoaTweet::Authentication {
Bearer::Bearer() {
method_ = AuthenticationMethod::OAUTH2;
}
Bearer::Bearer(const Key _key){
key_ = _key;
method_ = AuthenticationMethod::OAUTH2;
}
const std::string Bearer::calculateAuthHeader(std::map<std::string, std::string> _bodyParam,
const std::string& _method,
const std::string& _url) {
auto authHeader = std::string("Authorization: Bearer ") + key_.bearerToken();
return authHeader;
}
} // namespace CocoaTweet::Authentication
+22
View File
@@ -0,0 +1,22 @@
#ifndef COCOATWEET_AUTHENTICATION_BEARER_H_
#define COCOATWEET_AUTHENTICATION_BEARER_H_
#include <string>
#include <map>
#include <memory>
#include "key.h"
#include <cocoatweet/authentication/authenticator.h>
namespace CocoaTweet::Authentication {
class Bearer: public AuthenticatorBase {
public:
Bearer();
Bearer(const Key _key);
const std::string calculateAuthHeader(std::map<std::string, std::string> _bodyParam,
const std::string& _method, const std::string& _url);
};
} // namespace CocoaTweet::Authentication
#endif
+1 -2
View File
@@ -9,8 +9,7 @@ class Key {
public:
enum AUTH_TYPE{
OAUTH10A,
OAUTH2,
PLAIN
OAUTH2
};
private:
@@ -0,0 +1,12 @@
#ifndef COCOATWEET_EXCEPTION_CREDETIALNOTALLOWEDEXCEPTION_H_
#define COCOATWEET_EXCEPTION_CREDETIALNOTALLOWEDEXCEPTION_H_
#include <cocoatweet/exception/exception.h>
namespace CocoaTweet::Exception {
class CredentialNotAllowedException final : public Exception {
using Exception::Exception;
};
} // namespace CocoaTweet::Exception
#endif
@@ -0,0 +1,12 @@
#ifndef COCOATWEET_EXCEPTION_MISSINGREQUIREDPARAMEXCEPTION_H_
#define COCOATWEET_EXCEPTION_MISSINGREQUIREDPARAMEXCEPTION_H_
#include <cocoatweet/exception/exception.h>
namespace CocoaTweet::Exception {
class MissingRequiredParamException final : public Exception {
using Exception::Exception;
};
} // namespace CocoaTweet::Exception
#endif