feat: using generics for getters.

This commit is contained in:
Andrey Parhomenko 2024-06-05 02:08:54 +05:00
parent baf3d06c3e
commit c8cab4def3
8 changed files with 232 additions and 198 deletions

View file

@ -32,4 +32,3 @@ func NewClient(secretPath string) (*Client, error) {
}, nil }, nil
} }

View file

@ -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))

View file

@ -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)
}) })

View file

@ -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
View 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)
}

View file

@ -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
) )

View file

@ -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
View file

@ -0,0 +1,7 @@
package amo
import "errors"
var (
ErrTooManyEntities = errors.New("too many entities")
)