2022-12-28 22:22:36 -05:00
|
|
|
package discourse
|
|
|
|
|
|
|
|
import (
|
2022-12-29 14:35:01 -05:00
|
|
|
"bufio"
|
|
|
|
"crypto/rand"
|
|
|
|
"crypto/rsa"
|
|
|
|
"crypto/x509"
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
|
|
|
"encoding/pem"
|
|
|
|
"fmt"
|
|
|
|
"net/url"
|
|
|
|
"os"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/google/uuid"
|
2022-12-28 22:22:36 -05:00
|
|
|
"github.com/mrusme/gobbs/models/post"
|
|
|
|
"github.com/mrusme/gobbs/system/adapter"
|
2022-12-29 14:35:01 -05:00
|
|
|
"github.com/pkg/browser"
|
2022-12-28 22:22:36 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
type System struct {
|
2022-12-29 14:35:01 -05:00
|
|
|
config map[string]interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
type UserAPIKey struct {
|
|
|
|
Key string `json:"key"`
|
|
|
|
Nonce string `json:"nonce"`
|
|
|
|
Push bool `json:"push"`
|
|
|
|
API int `json:"api"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sys *System) GetConfig() map[string]interface{} {
|
|
|
|
return sys.config
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sys *System) SetConfig(cfg *map[string]interface{}) {
|
|
|
|
sys.config = *cfg
|
2022-12-28 22:22:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (sys *System) Load() error {
|
2022-12-29 14:35:01 -05:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
func (sys *System) Connect(sysURL string) error {
|
2022-12-29 14:35:01 -05:00
|
|
|
var err error
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// Request input from user
|
|
|
|
scanner := bufio.NewScanner(os.Stdin)
|
|
|
|
var username string = ""
|
|
|
|
for username == "" {
|
|
|
|
fmt.Printf(
|
|
|
|
"Please enter your username: ",
|
|
|
|
)
|
|
|
|
scanner.Scan()
|
|
|
|
username = strings.ReplaceAll(scanner.Text(), " ", "")
|
|
|
|
if username == "" {
|
|
|
|
fmt.Println("Invalid input")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Private key
|
|
|
|
privateKey, err := rsa.GenerateKey(rand.Reader, 2048)
|
2022-12-29 14:35:01 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-12-29 17:46:27 -05:00
|
|
|
privateKeyBytes := x509.MarshalPKCS1PrivateKey(privateKey)
|
|
|
|
privateKeyPEM := string(pem.EncodeToMemory(&pem.Block{Type: "PRIVATE KEY", Bytes: privateKeyBytes}))
|
2022-12-29 14:35:01 -05:00
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// Public key
|
|
|
|
publicKeyBytes, err := x509.MarshalPKIXPublicKey(&privateKey.PublicKey)
|
2022-12-29 14:35:01 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-12-29 17:46:27 -05:00
|
|
|
publicKeyPEM := string(pem.EncodeToMemory(&pem.Block{Type: "PUBLIC KEY", Bytes: publicKeyBytes}))
|
2022-12-29 14:35:01 -05:00
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// Client ID
|
2022-12-29 14:35:01 -05:00
|
|
|
uuidV4, err := uuid.NewRandom()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
clientID := uuidV4.String()
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// Nonce
|
2022-12-29 14:35:01 -05:00
|
|
|
randomBytes := make([]byte, 20)
|
|
|
|
_, err = rand.Read(randomBytes)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
nonce := base64.URLEncoding.EncodeToString(randomBytes)
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// URL
|
|
|
|
baseURL := fmt.Sprintf("%s/user-api-key/new", sysURL)
|
2022-12-29 14:35:01 -05:00
|
|
|
values := url.Values{}
|
|
|
|
values.Set("application_name", "gobbs")
|
|
|
|
values.Set("client_id", clientID)
|
|
|
|
values.Set("scopes", "read,write,notifications")
|
2022-12-29 17:46:27 -05:00
|
|
|
values.Set("public_key", publicKeyPEM)
|
2022-12-29 14:35:01 -05:00
|
|
|
values.Set("nonce", nonce)
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// Open in browser
|
2022-12-29 14:35:01 -05:00
|
|
|
openURL := baseURL + "?" + values.Encode()
|
|
|
|
if err := browser.OpenURL(openURL); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// Request input from user
|
|
|
|
scanner = bufio.NewScanner(os.Stdin)
|
2022-12-29 14:35:01 -05:00
|
|
|
var encodedUserAPIKey string = ""
|
|
|
|
for encodedUserAPIKey == "" {
|
2022-12-29 17:46:27 -05:00
|
|
|
fmt.Printf(
|
|
|
|
"\nPlease copy the user API key after authorizing and paste it here: ",
|
|
|
|
)
|
2022-12-29 14:35:01 -05:00
|
|
|
scanner.Scan()
|
|
|
|
encodedUserAPIKey = strings.ReplaceAll(scanner.Text(), " ", "")
|
|
|
|
if encodedUserAPIKey == "" {
|
|
|
|
fmt.Println("Invalid input")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// API key
|
|
|
|
decodedUserAPIKey, err := base64.StdEncoding.DecodeString(encodedUserAPIKey)
|
2022-12-29 14:35:01 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
decryptedUserAPIKey, err := privateKey.Decrypt(
|
2022-12-29 14:35:01 -05:00
|
|
|
rand.Reader,
|
2022-12-29 17:46:27 -05:00
|
|
|
decodedUserAPIKey,
|
2022-12-29 14:35:01 -05:00
|
|
|
nil,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var userAPIKey UserAPIKey
|
|
|
|
err = json.Unmarshal(decryptedUserAPIKey, &userAPIKey)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-12-29 17:46:27 -05:00
|
|
|
// Credentials
|
|
|
|
credentials := make(map[string]string)
|
|
|
|
credentials["pk"] = privateKeyPEM
|
|
|
|
credentials["username"] = username
|
|
|
|
credentials["key"] = userAPIKey.Key
|
|
|
|
|
|
|
|
if sys.config == nil {
|
|
|
|
sys.config = make(map[string]interface{})
|
|
|
|
}
|
|
|
|
sys.config["credentials"] = credentials
|
2022-12-29 14:35:01 -05:00
|
|
|
|
2022-12-28 22:22:36 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sys *System) ListPosts() ([]post.Post, error) {
|
|
|
|
return []post.Post{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sys *System) GetCapabilities() []adapter.Capability {
|
|
|
|
var caps []adapter.Capability
|
|
|
|
|
|
|
|
caps = append(caps, adapter.Capability{
|
|
|
|
ID: "posts",
|
|
|
|
Name: "Posts",
|
|
|
|
})
|
|
|
|
caps = append(caps, adapter.Capability{
|
|
|
|
ID: "groups",
|
|
|
|
Name: "Groups",
|
|
|
|
})
|
|
|
|
caps = append(caps, adapter.Capability{
|
|
|
|
ID: "search",
|
|
|
|
Name: "Search",
|
|
|
|
})
|
|
|
|
|
|
|
|
return caps
|
|
|
|
}
|