client.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. package gdrivefs
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "io/ioutil"
  7. "net/http"
  8. "net/url"
  9. "os"
  10. "os/user"
  11. "path/filepath"
  12. drive "google.golang.org/api/drive/v3"
  13. "golang.org/x/oauth2"
  14. "golang.org/x/oauth2/google"
  15. )
  16. func getClient(ctx context.Context, config *oauth2.Config) *http.Client {
  17. cacheFile, err := tokenCacheFile()
  18. if err != nil {
  19. log.Fatalf("Unable to get path to cached credential file. %v", err)
  20. }
  21. tok, err := tokenFromFile(cacheFile)
  22. if err != nil {
  23. tok = getTokenFromWeb(config)
  24. saveToken(cacheFile, tok)
  25. }
  26. return config.Client(ctx, tok)
  27. }
  28. func tokenCacheFile() (string, error) {
  29. tokenCacheDir, err := getConfigPath()
  30. if err != nil {
  31. return "", err
  32. }
  33. os.MkdirAll(tokenCacheDir, 0700)
  34. return filepath.Join(tokenCacheDir, url.QueryEscape("auth.json")), err
  35. }
  36. func getTokenFromWeb(config *oauth2.Config) *oauth2.Token {
  37. authURL := config.AuthCodeURL("state-token", oauth2.AccessTypeOffline)
  38. fmt.Printf(
  39. `Go to the following link in your browser:
  40. ----------------------------------------------------------------------------------------------
  41. %v
  42. ----------------------------------------------------------------------------------------------
  43. type the authorization code: `, authURL)
  44. var code string
  45. if _, err := fmt.Scan(&code); err != nil {
  46. log.Fatalf("Unable to read authorization code %v", err)
  47. }
  48. tok, err := config.Exchange(oauth2.NoContext, code)
  49. if err != nil {
  50. log.Fatalf("Unable to retrieve token from web: %v", err)
  51. }
  52. return tok
  53. }
  54. func tokenFromFile(file string) (*oauth2.Token, error) {
  55. f, err := os.Open(file)
  56. if err != nil {
  57. return nil, err
  58. }
  59. defer f.Close()
  60. t := &oauth2.Token{}
  61. err = json.NewDecoder(f).Decode(t)
  62. return t, err
  63. }
  64. func saveToken(file string, token *oauth2.Token) {
  65. log.Printf("Saving credential file to: %s\n", file)
  66. f, err := os.OpenFile(file, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0600)
  67. if err != nil {
  68. log.Fatalf("Unable to cache oauth token: %v\n", err)
  69. }
  70. defer f.Close()
  71. json.NewEncoder(f).Encode(token)
  72. }
  73. func getConfigPath() (string, error) {
  74. usr, err := user.Current()
  75. if err != nil {
  76. return "", err
  77. }
  78. configDir := filepath.Join(usr.HomeDir, ".gdrivemount")
  79. return configDir, nil
  80. }
  81. func GetDriveService() *drive.Service {
  82. configPath, err := getConfigPath()
  83. if err != nil {
  84. log.Fatal("Unable to fetch config path")
  85. }
  86. ctx := context.Background()
  87. b, err := ioutil.ReadFile(filepath.Join(configPath, "client_secret.json"))
  88. if err != nil {
  89. log.Fatalf("Unable to read client secret file: %v", err)
  90. }
  91. config, err := google.ConfigFromJSON(b, drive.DriveScope)
  92. if err != nil {
  93. log.Fatalf("Unable to parse client secret file: %v", err)
  94. }
  95. client := getClient(ctx, config)
  96. srv, err := drive.New(client)
  97. if err != nil {
  98. log.Fatalf("Unable to retrieve drive Client: %v", err)
  99. }
  100. return srv
  101. }