feat: using generics for getters.
This commit is contained in:
parent
baf3d06c3e
commit
c8cab4def3
8 changed files with 232 additions and 198 deletions
|
@ -32,4 +32,3 @@ func NewClient(secretPath string) (*Client, error) {
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -63,7 +63,7 @@ func MakeGetterFlags(
|
||||||
&opts.All,
|
&opts.All,
|
||||||
"all",
|
"all",
|
||||||
false,
|
false,
|
||||||
"get all leads",
|
"get all entities",
|
||||||
)
|
)
|
||||||
flags.IntVar(
|
flags.IntVar(
|
||||||
&opts.StartPage,
|
&opts.StartPage,
|
||||||
|
@ -160,7 +160,7 @@ func RunForSliceInThreads[V any](
|
||||||
return ret
|
return ret
|
||||||
}
|
}
|
||||||
|
|
||||||
func ReadIDs(idStrs []string, flags *mtool.Flags) []int {
|
func ReadIDs(idStrs []string) []int {
|
||||||
var ids []int
|
var ids []int
|
||||||
if len(idStrs) > 0 {
|
if len(idStrs) > 0 {
|
||||||
ids = make([]int, 0, len(idStrs))
|
ids = make([]int, 0, len(idStrs))
|
||||||
|
|
|
@ -3,53 +3,32 @@ package main
|
||||||
import "surdeus.su/core/amo"
|
import "surdeus.su/core/amo"
|
||||||
import "surdeus.su/core/ss/urlenc"
|
import "surdeus.su/core/ss/urlenc"
|
||||||
import "surdeus.su/core/cli/mtool"
|
import "surdeus.su/core/cli/mtool"
|
||||||
import "encoding/json"
|
//import "encoding/json"
|
||||||
import "strconv"
|
//import "strconv"
|
||||||
import "log"
|
//import "log"
|
||||||
import "fmt"
|
//import "fmt"
|
||||||
//import "os"
|
//import "os"
|
||||||
|
|
||||||
var getComs = mtool.T("get-coms").Func(func(flags *mtool.Flags){
|
|
||||||
var (
|
|
||||||
secretPath string
|
|
||||||
)
|
|
||||||
|
|
||||||
flags.StringVar(
|
type CompanyGetter struct{}
|
||||||
&secretPath,
|
|
||||||
"secret",
|
func (l CompanyGetter) GetValues(
|
||||||
"",
|
c *amo.Client,
|
||||||
"path to JSON file with AMO CRM secrets",
|
opts ...urlenc.Builder,
|
||||||
"AMO_SECRET",
|
) ([]amo.Company, amo.NextFunc[[]amo.Company], error) {
|
||||||
)
|
return c.GetCompanies(opts...)
|
||||||
idStrs := flags.Parse()
|
|
||||||
ids := make([]int, len(idStrs))
|
|
||||||
for i, idStr := range idStrs {
|
|
||||||
var err error
|
|
||||||
ids[i], err = strconv.Atoi(idStr)
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("Error: Atoi(%q): %s\n", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
c, err := amo.NewClient(secretPath)
|
func (l CompanyGetter) GetNameMul() string {
|
||||||
if err != nil {
|
return "companies"
|
||||||
log.Fatalf("NewAmoClient(...): %s\n", err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
coms, err := c.GetCompanies(
|
func (l CompanyGetter) GetFuncName() string {
|
||||||
urlenc.Array[int]{
|
return "amo.GetCompanies"
|
||||||
"id",
|
|
||||||
ids,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("GetCompanies(...): %s\n", err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bts, err := json.MarshalIndent(coms, "", " ")
|
var getComs = mtool.T("get-companies").Func(func(
|
||||||
if err != nil {
|
flags *mtool.Flags,
|
||||||
log.Fatalf("json.Marshal(...): %s\n", err)
|
){
|
||||||
}
|
RunGetter(CompanyGetter{}, flags)
|
||||||
fmt.Printf("%s\n", bts)
|
|
||||||
})
|
})
|
||||||
|
|
|
@ -1,140 +1,29 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import "surdeus.su/core/amo"
|
|
||||||
//import "surdeus.su/core/amo/api"
|
|
||||||
import "surdeus.su/core/ss/urlenc"
|
|
||||||
import "surdeus.su/core/cli/mtool"
|
import "surdeus.su/core/cli/mtool"
|
||||||
import "encoding/json"
|
import "surdeus.su/core/amo"
|
||||||
import "log"
|
import "surdeus.su/core/ss/urlenc"
|
||||||
import "fmt"
|
|
||||||
import "time"
|
|
||||||
import "os"
|
|
||||||
//import "sync"
|
|
||||||
|
|
||||||
var getLead = mtool.T("get-leads").Func(func(flags *mtool.Flags){
|
type LeadGetter struct {}
|
||||||
var (
|
|
||||||
opts DefaultFlags
|
|
||||||
)
|
|
||||||
|
|
||||||
now := time.Now()
|
func (l LeadGetter) GetValues(
|
||||||
MakeDefaultFlags(&opts, flags)
|
c *amo.Client,
|
||||||
MakeGetterFlags(&opts, flags)
|
opts ...urlenc.Builder,
|
||||||
|
) ([]amo.Lead, amo.NextFunc[[]amo.Lead], error) {
|
||||||
idStrs := flags.Parse()
|
return c.GetLeads(opts...)
|
||||||
c, err := amo.NewClient(opts.SecretPath)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("NewAmoClient(...): %s\n", err)
|
|
||||||
}
|
|
||||||
c.API.SetMRPS(opts.MRPS)
|
|
||||||
finalLeads := []amo.Lead{}
|
|
||||||
|
|
||||||
if opts.All {
|
|
||||||
page := opts.StartPage
|
|
||||||
leads, next, err := c.GetLeads(
|
|
||||||
urlenc.Value[int]{"page", page},
|
|
||||||
urlenc.Value[int]{"limit", opts.MEPR},
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("amo.GetLeads(...): %s\n", err)
|
|
||||||
}
|
|
||||||
finalLeads = append(finalLeads, leads...)
|
|
||||||
if opts.Verbose {
|
|
||||||
log.Printf("Got %d leads (%d, page %d)\n",
|
|
||||||
len(leads), len(finalLeads), page)
|
|
||||||
}
|
|
||||||
page++
|
|
||||||
for page <= opts.EndPage && next != nil {
|
|
||||||
leads, next, err = next()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("amo.GetLeads(...): %s\n", err)
|
|
||||||
}
|
|
||||||
finalLeads = append(finalLeads, leads...)
|
|
||||||
if opts.Verbose {
|
|
||||||
log.Printf("Got %d leads (%d, page %d)\n",
|
|
||||||
len(leads), len(finalLeads), page)
|
|
||||||
}
|
|
||||||
page++
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bts, err := json.MarshalIndent(finalLeads, "", " ")
|
func (l LeadGetter) GetNameMul() string {
|
||||||
if err != nil {
|
return "leads"
|
||||||
log.Fatalf("json.MarshalIndent(...) %s\n", err)
|
|
||||||
}
|
|
||||||
os.Stdout.Write(bts)
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ids := ReadIDs(idStrs, flags)
|
func (l LeadGetter) GetFuncName() string {
|
||||||
if len(ids) == 0 {
|
return "amo.GetLeads"
|
||||||
log.Fatalf("Got no IDs to read leads")
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
var getLead = mtool.T("get-leads").Func(func(
|
||||||
leadChan := make(chan []amo.Lead)
|
flags *mtool.Flags,
|
||||||
finish := RunForSliceInThreads[int](
|
){
|
||||||
opts.Threads, opts.MEPR,
|
RunGetter(LeadGetter{}, flags)
|
||||||
ids, func(thread int, s []int){
|
|
||||||
leads, _, err := c.GetLeads(
|
|
||||||
urlenc.Array[int]{
|
|
||||||
"id",
|
|
||||||
s,
|
|
||||||
},
|
|
||||||
urlenc.Value[string]{
|
|
||||||
"with",
|
|
||||||
"contacts",
|
|
||||||
},
|
|
||||||
urlenc.Value[int]{
|
|
||||||
"limit",
|
|
||||||
opts.MEPR,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("GetLeadsByIDs(...): %s\n", err)
|
|
||||||
}
|
|
||||||
leadChan <- leads
|
|
||||||
if opts.Verbose {
|
|
||||||
log.Printf(
|
|
||||||
"%d: Got %d leads\n",
|
|
||||||
thread, len(leads),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
//var wg sync.WaitGroup
|
|
||||||
go func(){
|
|
||||||
// Waiting for appending so we do not lose data.
|
|
||||||
<-finish
|
|
||||||
for len(leadChan) > 0 {}
|
|
||||||
close(leadChan)
|
|
||||||
}()
|
|
||||||
|
|
||||||
for leads := range leadChan {
|
|
||||||
finalLeads = append(finalLeads, leads...)
|
|
||||||
}
|
|
||||||
|
|
||||||
if opts.Verbose {
|
|
||||||
rm := c.API.RequestsMade()
|
|
||||||
log.Printf(
|
|
||||||
"Summarized got %d leads\n",
|
|
||||||
len(finalLeads),
|
|
||||||
)
|
|
||||||
log.Printf(
|
|
||||||
"Made %d requests in process\n",
|
|
||||||
rm,
|
|
||||||
)
|
|
||||||
took := time.Since(now).Seconds()
|
|
||||||
log.Printf(
|
|
||||||
"Took %f seconds\n",
|
|
||||||
took,
|
|
||||||
)
|
|
||||||
log.Printf("RPS = %f\n", float64(rm)/took)
|
|
||||||
}
|
|
||||||
bts, err := json.MarshalIndent(finalLeads, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("json.Marshal(...): %s\n", err)
|
|
||||||
}
|
|
||||||
fmt.Printf("%s\n", bts)
|
|
||||||
}).Usage(
|
}).Usage(
|
||||||
"[id1 id2 ... idN]",
|
"[id1 id2 ... idN]",
|
||||||
)
|
)
|
||||||
|
|
150
cmd/amocli/getter.go
Normal file
150
cmd/amocli/getter.go
Normal file
|
@ -0,0 +1,150 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import "surdeus.su/core/amo"
|
||||||
|
import "surdeus.su/core/ss/urlenc"
|
||||||
|
|
||||||
|
import "surdeus.su/core/cli/mtool"
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
import "log"
|
||||||
|
import "time"
|
||||||
|
import "os"
|
||||||
|
import "encoding/json"
|
||||||
|
|
||||||
|
type Getter[V any] interface {
|
||||||
|
GetValues(*amo.Client, ...urlenc.Builder) ([]V, amo.NextFunc[[]V], error)
|
||||||
|
GetNameMul() string
|
||||||
|
GetFuncName() string
|
||||||
|
}
|
||||||
|
|
||||||
|
func RunGetter[V any, G Getter[V]](g G, flags *mtool.Flags) {
|
||||||
|
var (
|
||||||
|
opts DefaultFlags
|
||||||
|
)
|
||||||
|
|
||||||
|
now := time.Now()
|
||||||
|
MakeDefaultFlags(&opts, flags)
|
||||||
|
MakeGetterFlags(&opts, flags)
|
||||||
|
|
||||||
|
idStrs := flags.Parse()
|
||||||
|
c, err := amo.NewClient(opts.SecretPath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("NewAmoClient(...): %s\n", err)
|
||||||
|
}
|
||||||
|
c.API.SetMRPS(opts.MRPS)
|
||||||
|
finalValues := []V{}
|
||||||
|
|
||||||
|
if opts.All {
|
||||||
|
page := opts.StartPage
|
||||||
|
values, next, err := g.GetValues(
|
||||||
|
c,
|
||||||
|
urlenc.Value[int]{"page", page},
|
||||||
|
urlenc.Value[int]{"limit", opts.MEPR},
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf(
|
||||||
|
"%s(...): %s\n", g.GetFuncName(), err,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
finalValues = append(finalValues, values...)
|
||||||
|
if opts.Verbose {
|
||||||
|
log.Printf("Got %d %s (%d, page %d)\n",
|
||||||
|
len(values), g.GetNameMul(), len(finalValues), page)
|
||||||
|
}
|
||||||
|
page++
|
||||||
|
for page <= opts.EndPage && next != nil {
|
||||||
|
values, next, err = next()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("amo.GetLeads(...): %s\n", err)
|
||||||
|
}
|
||||||
|
finalValues = append(finalValues, values...)
|
||||||
|
if opts.Verbose {
|
||||||
|
log.Printf("Got %d %s (%d, page %d)\n",
|
||||||
|
len(values), g.GetNameMul(),
|
||||||
|
len(finalValues), page)
|
||||||
|
}
|
||||||
|
page++
|
||||||
|
}
|
||||||
|
|
||||||
|
bts, err := json.MarshalIndent(finalValues, "", " ")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("json.MarshalIndent(...) %s\n", err)
|
||||||
|
}
|
||||||
|
os.Stdout.Write(bts)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ids := ReadIDs(idStrs)
|
||||||
|
if len(ids) == 0 {
|
||||||
|
log.Fatalf("Got no IDs to read %s", g.GetNameMul())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
valueChan := make(chan []V)
|
||||||
|
finish := RunForSliceInThreads[int](
|
||||||
|
opts.Threads, opts.MEPR,
|
||||||
|
ids, func(thread int, s []int){
|
||||||
|
values, _, err := g.GetValues(
|
||||||
|
c,
|
||||||
|
urlenc.Array[int]{
|
||||||
|
"id",
|
||||||
|
s,
|
||||||
|
},
|
||||||
|
urlenc.Value[string]{
|
||||||
|
"with",
|
||||||
|
"contacts",
|
||||||
|
},
|
||||||
|
urlenc.Value[int]{
|
||||||
|
"limit",
|
||||||
|
opts.MEPR,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("GetLeadsByIDs(...): %s\n", err)
|
||||||
|
}
|
||||||
|
valueChan <- values
|
||||||
|
if opts.Verbose {
|
||||||
|
log.Printf(
|
||||||
|
"%d: Got %d %s\n",
|
||||||
|
thread, len(values),
|
||||||
|
g.GetNameMul(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
//var wg sync.WaitGroup
|
||||||
|
go func(){
|
||||||
|
// Waiting for appending so we do not lose data.
|
||||||
|
<-finish
|
||||||
|
for len(valueChan) > 0 {}
|
||||||
|
close(valueChan)
|
||||||
|
}()
|
||||||
|
|
||||||
|
for values := range valueChan {
|
||||||
|
finalValues = append(finalValues, values...)
|
||||||
|
}
|
||||||
|
|
||||||
|
if opts.Verbose {
|
||||||
|
rm := c.API.RequestsMade()
|
||||||
|
log.Printf(
|
||||||
|
"Summarized got %d %s\n",
|
||||||
|
len(finalValues), g.GetNameMul(),
|
||||||
|
)
|
||||||
|
log.Printf(
|
||||||
|
"Made %d requests in process\n",
|
||||||
|
rm,
|
||||||
|
)
|
||||||
|
took := time.Since(now).Seconds()
|
||||||
|
log.Printf(
|
||||||
|
"Took %f seconds\n",
|
||||||
|
took,
|
||||||
|
)
|
||||||
|
log.Printf("RPS = %f\n", float64(rm)/took)
|
||||||
|
}
|
||||||
|
bts, err := json.MarshalIndent(finalValues, "", " ")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("json.Marshal(...): %s\n", err)
|
||||||
|
}
|
||||||
|
fmt.Printf("%s\n", bts)
|
||||||
|
}
|
|
@ -11,7 +11,7 @@ import "os"
|
||||||
const MEPR = api.MaxEntitiesPerRequest
|
const MEPR = api.MaxEntitiesPerRequest
|
||||||
|
|
||||||
var updateComs =
|
var updateComs =
|
||||||
mtool.T("update-coms").Func(func(flags *mtool.Flags){
|
mtool.T("update-companies").Func(func(flags *mtool.Flags){
|
||||||
var (
|
var (
|
||||||
opts DefaultFlags
|
opts DefaultFlags
|
||||||
)
|
)
|
||||||
|
|
42
companies.go
42
companies.go
|
@ -9,7 +9,7 @@ import "fmt"
|
||||||
func (client *Client) GetCompany(
|
func (client *Client) GetCompany(
|
||||||
companyID int,
|
companyID int,
|
||||||
opts ...urlenc.Builder,
|
opts ...urlenc.Builder,
|
||||||
) (*companies.Company, error) {
|
) (*Company, error) {
|
||||||
deal := new(companies.Company)
|
deal := new(companies.Company)
|
||||||
resource := fmt.Sprintf(
|
resource := fmt.Sprintf(
|
||||||
"/api/v4/companies/%d?%s",
|
"/api/v4/companies/%d?%s",
|
||||||
|
@ -26,32 +26,39 @@ func (client *Client) GetCompany(
|
||||||
// Get list of leads.
|
// Get list of leads.
|
||||||
func (client *Client) GetCompanies(
|
func (client *Client) GetCompanies(
|
||||||
opts ...urlenc.Builder,
|
opts ...urlenc.Builder,
|
||||||
) ([]companies.Company, error) {
|
) ([]Company, NextFunc[[]Company], error) {
|
||||||
res := fmt.Sprintf(
|
res := fmt.Sprintf(
|
||||||
"/api/v4/companies?%s",
|
"/api/v4/companies?%s",
|
||||||
urlenc.Join(opts...).Encode(),
|
urlenc.Join(opts...).Encode(),
|
||||||
)
|
)
|
||||||
|
|
||||||
ret := []companies.Company{}
|
return client.GetCompaniesByURL(res)
|
||||||
for {
|
}
|
||||||
var coms companies.Companies
|
|
||||||
err := client.API.Get(res, &coms)
|
func (client *Client) GetCompaniesByURL(
|
||||||
|
u string,
|
||||||
|
) ([]Company, NextFunc[[]Company], error) {
|
||||||
|
var fn NextFunc[[]Company]
|
||||||
|
|
||||||
|
coms := companies.Companies{}
|
||||||
|
err := client.API.Get(u, &coms)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if errors.Is(err, api.ErrNoContent) {
|
if errors.Is(err, api.ErrNoContent) {
|
||||||
break
|
return nil, nil, nil
|
||||||
}
|
}
|
||||||
return nil, err
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = append(ret, coms.Embedded.Companies...)
|
//ret = append(ret, coms.Embedded.Companies...)
|
||||||
|
|
||||||
if coms.Links.Next.Href == "" {
|
nextHref := coms.Links.Next.Href
|
||||||
break
|
if nextHref != "" {
|
||||||
|
fn = MakeNextFunc(
|
||||||
|
nextHref,
|
||||||
|
client.GetCompaniesByURL,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
res = coms.Links.Next.Href
|
return coms.Embedded.Companies, fn, nil
|
||||||
|
|
||||||
}
|
|
||||||
return ret, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (client *Client) UpdateCompany(
|
func (client *Client) UpdateCompany(
|
||||||
|
@ -67,7 +74,10 @@ func (client *Client) UpdateCompany(
|
||||||
func (client *Client) UpdateCompanies(
|
func (client *Client) UpdateCompanies(
|
||||||
cs []companies.Company,
|
cs []companies.Company,
|
||||||
) error {
|
) error {
|
||||||
//ret := []companies.Company{}
|
if len(cs) > MEPR {
|
||||||
|
return ErrTooManyEntities
|
||||||
|
}
|
||||||
|
//ret := []Company{}
|
||||||
err := client.API.Patch(
|
err := client.API.Patch(
|
||||||
"/api/v4/companies",
|
"/api/v4/companies",
|
||||||
cs,
|
cs,
|
||||||
|
|
7
errors.go
Normal file
7
errors.go
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
package amo
|
||||||
|
|
||||||
|
import "errors"
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrTooManyEntities = errors.New("too many entities")
|
||||||
|
)
|
Loading…
Reference in a new issue