-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsheets.go
99 lines (89 loc) · 2.89 KB
/
sheets.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
package importer
import (
"context"
"fmt"
"google.golang.org/api/option"
"google.golang.org/api/sheets/v4"
"gopkg.in/errgo.v2/errors"
"log"
"regexp"
)
var googleSpreadsheetMatcher = regexp.MustCompile("^https:\\/\\/docs\\.google\\.com\\/spreadsheets\\/d\\/(?P<spreadsheetId>.*)\\/edit.*$")
func (i *Importer) putCsvsToSheet(ctx context.Context, spreadsheetUrl string, sheetName string, header []interface{}, data [][]interface{}) error {
srv, err := sheets.NewService(ctx, option.WithCredentialsJSON([]byte(i.GoogleCredentialsJson)), option.WithScopes(sheets.SpreadsheetsScope))
if err != nil {
return errors.Because(err, nil, "Unable to retrieve Sheets client")
}
match := googleSpreadsheetMatcher.FindStringSubmatch(spreadsheetUrl)
spreadsheetId := match[googleSpreadsheetMatcher.SubexpIndex("spreadsheetId")]
updateRange := fmt.Sprintf("%s!1:%d", sheetName, len(data)+1)
values := make([][]interface{}, 0)
values = append(values, header)
values = append(values, data...)
updateValueRange := sheets.ValueRange{
Values: values,
}
sheetId := int64(-1)
existingColCount := int64(-1)
existingRowCount := int64(-1)
spreadsheet, err := srv.Spreadsheets.Get(spreadsheetId).Do()
if err != nil {
return errors.Wrap(err)
}
for _, s := range spreadsheet.Sheets {
if s.Properties.Title == sheetName {
sheetId = s.Properties.SheetId
existingColCount = s.Properties.GridProperties.ColumnCount
existingRowCount = s.Properties.GridProperties.RowCount
}
}
spreadsheetRequest := sheets.BatchUpdateSpreadsheetRequest{
Requests: make([]*sheets.Request, 0),
}
if existingRowCount > int64(len(data)+1) {
spreadsheetRequest.Requests = append(spreadsheetRequest.Requests, &sheets.Request{
DeleteDimension: &sheets.DeleteDimensionRequest{
Range: &sheets.DimensionRange{
Dimension: "ROWS",
SheetId: sheetId,
StartIndex: int64(len(data)+1),
},
},
})
}
if existingColCount > int64(len(header)) {
spreadsheetRequest.Requests = append(spreadsheetRequest.Requests, &sheets.Request{
DeleteDimension: &sheets.DeleteDimensionRequest{
Range: &sheets.DimensionRange{
Dimension: "COLUMNS",
SheetId: sheetId,
StartIndex: int64(len(header)),
},
},
})
}
if i.Verbose {
log.Printf("clearing ranges %+v", spreadsheetRequest.Requests)
}
_, err = srv.Spreadsheets.Values.Update(spreadsheetId, updateRange, &updateValueRange).ValueInputOption("RAW").Do()
if err != nil {
return errors.Wrap(err)
}
if len(spreadsheetRequest.Requests) >0 {
_, err = srv.Spreadsheets.BatchUpdate(spreadsheetId, &spreadsheetRequest).Do()
if err != nil {
return errors.Wrap(err)
}
}
return nil
}
func intToLetters(number int64) (letters string) {
number--
if firstLetter := number / 26; firstLetter > 0 {
letters += intToLetters(firstLetter)
letters += string('A' + number%26)
} else {
letters += string('A' + number)
}
return
}