refactor etl go: load from byte slice instead of interface
- add benchmark - add profiling and trace
This commit is contained in:
parent
89864b3daa
commit
4c8da4777d
26
Makefile
26
Makefile
|
@ -1,16 +1,4 @@
|
||||||
COVERAGE_DIR=coverage
|
COVERAGE_DIR=coverage
|
||||||
BINARY_DIR=bin
|
|
||||||
BINARY_NAME=ess-etl-go
|
|
||||||
|
|
||||||
UNAME := $(shell uname -s)
|
|
||||||
ifeq ($(UNAME),Darwin)
|
|
||||||
OS = macos
|
|
||||||
else ifeq ($(UNAME),Linux)
|
|
||||||
OS = linux
|
|
||||||
else
|
|
||||||
$(error OS not supported by this Makefile)
|
|
||||||
endif
|
|
||||||
PACKAGE = $(shell head -1 go.mod | awk '{print $$2}')
|
|
||||||
|
|
||||||
lint:
|
lint:
|
||||||
golangci-lint run ./...
|
golangci-lint run ./...
|
||||||
|
@ -25,7 +13,21 @@ test-coverage:
|
||||||
go tool cover -html ${COVERAGE_DIR}/cover.out -o ${COVERAGE_DIR}/cover.html
|
go tool cover -html ${COVERAGE_DIR}/cover.out -o ${COVERAGE_DIR}/cover.html
|
||||||
benchmark:
|
benchmark:
|
||||||
go test -run none -bench . -benchtime 3s -benchmem
|
go test -run none -bench . -benchtime 3s -benchmem
|
||||||
|
benchmark_escape_analisys:
|
||||||
|
go test -gcflags "-m -m" -run none -bench . -benchtime 3s -benchmem -memprofile profile.out ./internal/etl
|
||||||
|
pprof:
|
||||||
|
go tool pprof -alloc_space profile.out
|
||||||
|
pprof_url:# top 40 -cum
|
||||||
|
go tool pprof -alloc_space http://localhost:5000/debug/pprof/allocs
|
||||||
|
benchmark_etl:
|
||||||
|
go test -gcflags "-m -m" -run none -bench . -benchtime 5s -benchmem -memprofile profile.out ./internal/etl/
|
||||||
|
trace_etl: build_etl# go tool trace t.out
|
||||||
|
$(build_etl)
|
||||||
|
time ./etl -trace > t.out
|
||||||
|
build_etl:
|
||||||
|
env GOOS=linux CGO_ENABLED=0 GOARCH=amd64 go build -v -o etl ./cmd/etl/main.go
|
||||||
build_server:
|
build_server:
|
||||||
env GOOS=linux CGO_ENABLED=0 GOARCH=amd64 go build -o ${BINARY_DIR}/${BINARY_NAME} ./main.go
|
env GOOS=linux CGO_ENABLED=0 GOARCH=amd64 go build -o ${BINARY_DIR}/${BINARY_NAME} ./main.go
|
||||||
run_server: build_server
|
run_server: build_server
|
||||||
./${BINARY_DIR}/${BINARY_NAME}
|
./${BINARY_DIR}/${BINARY_NAME}
|
||||||
|
|
||||||
|
|
|
@ -6,3 +6,6 @@ Start your server
|
||||||
|
|
||||||
Run your tests
|
Run your tests
|
||||||
`go test -race -v --bench ./. --benchmem ./benchmark `
|
`go test -race -v --bench ./. --benchmem ./benchmark `
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -26,7 +26,7 @@ func doRequest(wg *sync.WaitGroup, host string, employeeID int, errChan chan err
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
var err error
|
var err error
|
||||||
query := GraphqlQuery{
|
query := GraphqlQuery{
|
||||||
Query: `query ByEmployeeNumber($employeeNumber: Float!) { byEmployeeNumber(employeeNumber: $employeeNumber) { EmployeeNumber Date WorkInformation { EmployeeNumber Date Shifts { Start End ActualStart ActualEnd RoleCode ShiftCategoryCode _ReferenceId } DaysOff { DayOff { DayOffTypeCode Note _ReferenceId } } } Baselines { Shifts { DayOff { DayOffTypeCode } } BaselineType DaysOff { DayOff { DayOffTypeCode } } FullDayAbsences { FullDayAbsence { AbsenceTypeCode } } } CustomFields { CustomField { FullDayAbsence { AbsenceTypeCode } } } DataVersion FullDayAbsences { FullDayAbsence { AbsenceTypeCode Note _ReferenceId } } } }`,
|
Query: `query ByEmployeeNumber($employeeNumber: Float!) { byEmployeeNumber(employeeNumber: $employeeNumber) { EmployeeNumber Date WorkInformation { EmployeeNumber Date Shifts { Start End RoleCode } DaysOff { DayOff { DayOffTypeCode Note _ReferenceId } } } Baselines { Shifts { Start End RoleCode } BaselineType DaysOff { DayOff { DayOffTypeCode } } FullDayAbsences { FullDayAbsence { AbsenceTypeCode } } } CustomFields { CustomField { Name } } DataVersion FullDayAbsences { FullDayAbsence { _ReferenceId } } } } `,
|
||||||
OperationName: "ByEmployeeNumber",
|
OperationName: "ByEmployeeNumber",
|
||||||
Variables: ByEmployeeNumberQueryVariables{
|
Variables: ByEmployeeNumberQueryVariables{
|
||||||
EmployeeNumber: float64(employeeID),
|
EmployeeNumber: float64(employeeID),
|
||||||
|
|
|
@ -2,11 +2,15 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
|
"net/http"
|
||||||
_ "net/http/pprof"
|
_ "net/http/pprof"
|
||||||
"os"
|
"os"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
|
"runtime/pprof"
|
||||||
|
"runtime/trace"
|
||||||
"sync"
|
"sync"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
||||||
|
@ -21,28 +25,45 @@ import (
|
||||||
"go.mongodb.org/mongo-driver/mongo/options"
|
"go.mongodb.org/mongo-driver/mongo/options"
|
||||||
)
|
)
|
||||||
|
|
||||||
// var cpuprofile = flag.String("cpuprofile", "", "write cpu profile to file")
|
var cpuprofile = flag.String("cpuprofile", "", "write cpu profile to file")
|
||||||
|
var traceflag = flag.Bool("trace", false, "write trace to file")
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
// flag.Parse()
|
flag.Parse()
|
||||||
// if *cpuprofile != "" {
|
if *cpuprofile != "" {
|
||||||
// f, err := os.Create(*cpuprofile)
|
f, err := os.Create(*cpuprofile)
|
||||||
// if err != nil {
|
if err != nil {
|
||||||
// log.Fatal(err)
|
log.Fatal(err)
|
||||||
// }
|
}
|
||||||
// pprof.StartCPUProfile(f)
|
pprof.StartCPUProfile(f)
|
||||||
// defer pprof.StopCPUProfile()
|
defer pprof.StopCPUProfile()
|
||||||
// }
|
|
||||||
|
|
||||||
// // Add pprof endpoints
|
}
|
||||||
// go func() {
|
|
||||||
// log.Println(http.ListenAndServe("localhost:6060", nil))
|
// Add pprof endpoints
|
||||||
// }()
|
go func() {
|
||||||
|
if *cpuprofile != "" {
|
||||||
|
log.Println(http.ListenAndServe("localhost:6060", nil))
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
if *traceflag {
|
||||||
|
log.Println("trace on")
|
||||||
|
trace.Start(os.Stdout)
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if *traceflag {
|
||||||
|
trace.Stop()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
cr := crono.New()
|
cr := crono.New()
|
||||||
defer cr.Table()
|
if !*traceflag {
|
||||||
cfg := config.NewConfig(".env")
|
defer cr.Table()
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := config.NewConfig(".env")
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
|
|
||||||
dbOpts := options.Client()
|
dbOpts := options.Client()
|
||||||
|
@ -105,7 +126,7 @@ func main() {
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
for v := range ewiChan {
|
for v := range ewiChan {
|
||||||
log.Println("len v", len(v))
|
// log.Println("len v", len(v))
|
||||||
err := professionalRepo.InsertMany(ctx, v)
|
err := professionalRepo.InsertMany(ctx, v)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errChan <- err
|
errChan <- err
|
||||||
|
|
|
@ -2,14 +2,18 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"flag"
|
||||||
"log"
|
"log"
|
||||||
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
|
"runtime/pprof"
|
||||||
|
"runtime/trace"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
||||||
"gitea.urkob.com/urko/crono"
|
"gitea.urkob.com/urko/crono"
|
||||||
"gitea.urkob.com/urko/ess-etl-go/config"
|
"gitea.urkob.com/urko/ess-etl-go/config"
|
||||||
"gitea.urkob.com/urko/ess-etl-go/internal/api/http"
|
apihttp "gitea.urkob.com/urko/ess-etl-go/internal/api/http"
|
||||||
"gitea.urkob.com/urko/ess-etl-go/internal/services"
|
"gitea.urkob.com/urko/ess-etl-go/internal/services"
|
||||||
"gitea.urkob.com/urko/ess-etl-go/pkg/adapter/repository/mongodb/employee_wi"
|
"gitea.urkob.com/urko/ess-etl-go/pkg/adapter/repository/mongodb/employee_wi"
|
||||||
|
|
||||||
|
@ -17,7 +21,36 @@ import (
|
||||||
"go.mongodb.org/mongo-driver/mongo/options"
|
"go.mongodb.org/mongo-driver/mongo/options"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var cpuprofile = flag.String("cpuprofile", "", "write cpu profile to file")
|
||||||
|
var traceflag = flag.String("trace", "", "write trace to file")
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
flag.Parse()
|
||||||
|
if *cpuprofile != "" {
|
||||||
|
f, err := os.Create(*cpuprofile)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
pprof.StartCPUProfile(f)
|
||||||
|
defer pprof.StopCPUProfile()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add pprof endpoints
|
||||||
|
go func() {
|
||||||
|
if *cpuprofile != "" {
|
||||||
|
log.Println(http.ListenAndServe("localhost:6060", nil))
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
if *traceflag != "" {
|
||||||
|
log.Println("trace on")
|
||||||
|
trace.Start(os.Stdout)
|
||||||
|
defer func() {
|
||||||
|
log.Println("on stop")
|
||||||
|
trace.Stop()
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
cr := crono.New()
|
cr := crono.New()
|
||||||
defer cr.Table()
|
defer cr.Table()
|
||||||
cfg := config.NewConfig(".env")
|
cfg := config.NewConfig(".env")
|
||||||
|
@ -41,13 +74,12 @@ func main() {
|
||||||
employeeWICollection := client.Database(cfg.DbName).Collection(cfg.EmployeeWorkInformationCollection)
|
employeeWICollection := client.Database(cfg.DbName).Collection(cfg.EmployeeWorkInformationCollection)
|
||||||
professionalRepo := employee_wi.NewRepo(employeeWICollection)
|
professionalRepo := employee_wi.NewRepo(employeeWICollection)
|
||||||
|
|
||||||
restServer := http.
|
restServer := apihttp.
|
||||||
NewRestServer(cfg, cr).
|
NewRestServer(cfg, cr).
|
||||||
WithEmployeeWIHandler(services.NewEmployeeWIService(ctx, professionalRepo)).
|
WithEmployeeWIHandler(services.NewEmployeeWIService(ctx, professionalRepo)).
|
||||||
WithAMSHander()
|
WithAMSHander()
|
||||||
|
|
||||||
cr.MarkAndRestart("dependencies loaded")
|
cr.MarkAndRestart("dependencies loaded")
|
||||||
log.Println(cfg)
|
|
||||||
go func() {
|
go func() {
|
||||||
if err = restServer.Start(cfg.ApiPort, ""); err != nil {
|
if err = restServer.Start(cfg.ApiPort, ""); err != nil {
|
||||||
log.Fatalln("restServer.Start", err)
|
log.Fatalln("restServer.Start", err)
|
||||||
|
|
1
go.mod
1
go.mod
|
@ -5,6 +5,7 @@ go 1.19
|
||||||
require (
|
require (
|
||||||
gitea.urkob.com/urko/crono v0.0.0-20230405153202-0554f3e53a4c
|
gitea.urkob.com/urko/crono v0.0.0-20230405153202-0554f3e53a4c
|
||||||
gitea.urkob.com/urko/go-root-dir v0.0.0-20230311113851-2f6d4355888a
|
gitea.urkob.com/urko/go-root-dir v0.0.0-20230311113851-2f6d4355888a
|
||||||
|
github.com/docker/go-units v0.5.0
|
||||||
github.com/gofiber/fiber/v2 v2.43.0
|
github.com/gofiber/fiber/v2 v2.43.0
|
||||||
github.com/joho/godotenv v1.5.1
|
github.com/joho/godotenv v1.5.1
|
||||||
github.com/kelseyhightower/envconfig v1.4.0
|
github.com/kelseyhightower/envconfig v1.4.0
|
||||||
|
|
2
go.sum
2
go.sum
|
@ -7,6 +7,8 @@ github.com/andybalholm/brotli v1.0.5/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHG
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4=
|
||||||
|
github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
||||||
github.com/gofiber/fiber/v2 v2.43.0 h1:yit3E4kHf178B60p5CQBa/3v+WVuziWMa/G2ZNyLJB0=
|
github.com/gofiber/fiber/v2 v2.43.0 h1:yit3E4kHf178B60p5CQBa/3v+WVuziWMa/G2ZNyLJB0=
|
||||||
github.com/gofiber/fiber/v2 v2.43.0/go.mod h1:mpS1ZNE5jU+u+BA4FbM+KKnUzJ4wzTK+FT2tG3tU+6I=
|
github.com/gofiber/fiber/v2 v2.43.0/go.mod h1:mpS1ZNE5jU+u+BA4FbM+KKnUzJ4wzTK+FT2tG3tU+6I=
|
||||||
github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4=
|
github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4=
|
||||||
|
|
|
@ -0,0 +1,72 @@
|
||||||
|
package etl
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log"
|
||||||
|
"runtime"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/internal/xml_loader"
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/pkg/adapter/repository/mongodb/employee_wi"
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/pkg/domain"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Etl struct {
|
||||||
|
ewiLoader xml_loader.EmployeeWILoader
|
||||||
|
repo employee_wi.Repo
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(ewiLoader xml_loader.EmployeeWILoader, repo employee_wi.Repo) *Etl {
|
||||||
|
return &Etl{
|
||||||
|
ewiLoader: ewiLoader,
|
||||||
|
repo: repo,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (etl *Etl) FanOut(ctx context.Context, employeeNumber []string, from, to string) error {
|
||||||
|
g := runtime.GOMAXPROCS(0)
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
wg.Add(g)
|
||||||
|
|
||||||
|
employeeWIChan := make(chan []domain.EmployeeWorkInformation, g)
|
||||||
|
errChan := make(chan error, 1)
|
||||||
|
|
||||||
|
for i := 0; i < g; i++ {
|
||||||
|
go func() {
|
||||||
|
defer func() {
|
||||||
|
wg.Done()
|
||||||
|
}()
|
||||||
|
for v := range employeeWIChan {
|
||||||
|
func() {
|
||||||
|
err := etl.repo.InsertMany(ctx, v)
|
||||||
|
if err != nil {
|
||||||
|
errChan <- err
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
go func() {
|
||||||
|
var wg2 sync.WaitGroup
|
||||||
|
wg2.Add(len(employeeNumber))
|
||||||
|
for i := range employeeNumber {
|
||||||
|
go func(v string) {
|
||||||
|
defer wg2.Done()
|
||||||
|
wi, err := etl.ewiLoader.LoadEmployee(v, from, to)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("err", err)
|
||||||
|
errChan <- err
|
||||||
|
return
|
||||||
|
}
|
||||||
|
employeeWIChan <- wi
|
||||||
|
}(employeeNumber[i])
|
||||||
|
}
|
||||||
|
wg2.Wait()
|
||||||
|
close(employeeWIChan)
|
||||||
|
}()
|
||||||
|
|
||||||
|
wg.Wait()
|
||||||
|
errChan <- nil
|
||||||
|
return <-errChan
|
||||||
|
}
|
|
@ -0,0 +1,73 @@
|
||||||
|
package etl_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/config"
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/internal/etl"
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/internal/request"
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/internal/xml_loader"
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/pkg/adapter/repository/mongodb/employee_wi"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
"go.mongodb.org/mongo-driver/mongo"
|
||||||
|
"go.mongodb.org/mongo-driver/mongo/options"
|
||||||
|
)
|
||||||
|
|
||||||
|
func BenchmarkLoad(b *testing.B) {
|
||||||
|
cfg := config.NewConfig(".env")
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
dbOpts := options.Client()
|
||||||
|
dbOpts.ApplyURI(cfg.DbAddress)
|
||||||
|
|
||||||
|
client, err := mongo.NewClient(dbOpts)
|
||||||
|
require.NoError(b, err, "mongo.NewClient")
|
||||||
|
|
||||||
|
require.NoError(b, client.Connect(ctx), "client.Connect")
|
||||||
|
|
||||||
|
employeeWICollection := client.Database(cfg.DbName).Collection(cfg.EmployeeWorkInformationCollection)
|
||||||
|
if err = employeeWICollection.Drop(ctx); err != nil {
|
||||||
|
log.Fatalln("employeeWICollection.Drop", err)
|
||||||
|
}
|
||||||
|
repo := employee_wi.NewRepo(employeeWICollection)
|
||||||
|
|
||||||
|
r := request.NewRequestService(cfg.AmsApi, cfg.AmsApiKey)
|
||||||
|
|
||||||
|
ewiLoader := xml_loader.NewEmployeeWILoader(r)
|
||||||
|
from, to := "2023-01-01", "2023-01-31"
|
||||||
|
e := etl.New(ewiLoader, *repo)
|
||||||
|
|
||||||
|
b.ResetTimer()
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
require.NoError(b, e.FanOut(ctx, cfg.EmployeeIdList, from, to))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLoad(t *testing.T) {
|
||||||
|
cfg := config.NewConfig(".env")
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
dbOpts := options.Client()
|
||||||
|
dbOpts.ApplyURI(cfg.DbAddress)
|
||||||
|
|
||||||
|
client, err := mongo.NewClient(dbOpts)
|
||||||
|
require.NoError(t, err, "mongo.NewClient")
|
||||||
|
|
||||||
|
require.NoError(t, client.Connect(ctx), "client.Connect")
|
||||||
|
|
||||||
|
employeeWICollection := client.Database(cfg.DbName).Collection(cfg.EmployeeWorkInformationCollection)
|
||||||
|
if err = employeeWICollection.Drop(ctx); err != nil {
|
||||||
|
log.Fatalln("employeeWICollection.Drop", err)
|
||||||
|
}
|
||||||
|
repo := employee_wi.NewRepo(employeeWICollection)
|
||||||
|
|
||||||
|
r := request.NewRequestService(cfg.AmsApi, cfg.AmsApiKey)
|
||||||
|
|
||||||
|
ewiLoader := xml_loader.NewEmployeeWILoader(r)
|
||||||
|
from, to := "2023-01-01", "2023-01-31"
|
||||||
|
e := etl.New(ewiLoader, *repo)
|
||||||
|
|
||||||
|
require.NoError(t, e.FanOut(ctx, cfg.EmployeeIdList, from, to))
|
||||||
|
}
|
|
@ -1,7 +1,6 @@
|
||||||
package request
|
package request
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
@ -38,12 +37,12 @@ func getPayload(employeeIDList []string) (string, error) {
|
||||||
return employees.String(), nil
|
return employees.String(), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r RequestService) EmployeeWorkInformation(employeeIDList []string, from, to string) (io.Reader, error) {
|
func (r RequestService) EmployeeWorkInformation(data *[]byte, employeeIDList []string, from, to string) error {
|
||||||
url := r.api + "/EmployeeWorkInformation/Search/" + from + "/" + to + "/"
|
url := r.api + "/EmployeeWorkInformation/Search/" + from + "/" + to + "/"
|
||||||
|
|
||||||
stringPayload, err := getPayload(employeeIDList)
|
stringPayload, err := getPayload(employeeIDList)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("getPayload: %s", err)
|
return fmt.Errorf("getPayload: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
payload := strings.NewReader(stringPayload)
|
payload := strings.NewReader(stringPayload)
|
||||||
|
@ -51,7 +50,7 @@ func (r RequestService) EmployeeWorkInformation(employeeIDList []string, from, t
|
||||||
req, err := http.NewRequest("POST", url, payload)
|
req, err := http.NewRequest("POST", url, payload)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("http.NewRequest: %s", err)
|
return fmt.Errorf("http.NewRequest: %s", err)
|
||||||
}
|
}
|
||||||
req.Header.Add("Cache-Control", "no-cache")
|
req.Header.Add("Cache-Control", "no-cache")
|
||||||
req.Header.Add("Authorization", r.apiKey)
|
req.Header.Add("Authorization", r.apiKey)
|
||||||
|
@ -59,16 +58,30 @@ func (r RequestService) EmployeeWorkInformation(employeeIDList []string, from, t
|
||||||
|
|
||||||
res, err := client.Do(req)
|
res, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("client.Do: %s", err)
|
return fmt.Errorf("client.Do: %s", err)
|
||||||
}
|
}
|
||||||
defer res.Body.Close()
|
defer res.Body.Close()
|
||||||
|
|
||||||
body, err := io.ReadAll(res.Body)
|
for {
|
||||||
if err != nil {
|
if len(*data) == cap(*data) {
|
||||||
return nil, fmt.Errorf("ioutil.ReadAll: %s", err)
|
// Add more capacity (let append pick how much).
|
||||||
|
*data = append(*data, 0)[:len(*data)]
|
||||||
|
}
|
||||||
|
n, err := res.Body.Read((*data)[len(*data):cap(*data)])
|
||||||
|
*data = (*data)[:len(*data)+n]
|
||||||
|
if err != nil {
|
||||||
|
if err == io.EOF {
|
||||||
|
err = nil
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//log.Println("readed", string(body))
|
// data, err = io.ReadAll(res.Body)
|
||||||
|
// if err != nil {
|
||||||
|
// return fmt.Errorf("ioutil.ReadAll: %s", err)
|
||||||
|
// }
|
||||||
|
|
||||||
return bytes.NewReader(body), nil
|
//log.Println("readed", string(body))
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,10 +3,10 @@ package xml_loader
|
||||||
import (
|
import (
|
||||||
"encoding/xml"
|
"encoding/xml"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
|
||||||
|
|
||||||
"gitea.urkob.com/urko/ess-etl-go/internal/request"
|
"gitea.urkob.com/urko/ess-etl-go/internal/request"
|
||||||
"gitea.urkob.com/urko/ess-etl-go/pkg/domain"
|
"gitea.urkob.com/urko/ess-etl-go/pkg/domain"
|
||||||
|
"github.com/docker/go-units"
|
||||||
)
|
)
|
||||||
|
|
||||||
type EmployeeWILoader struct {
|
type EmployeeWILoader struct {
|
||||||
|
@ -18,31 +18,35 @@ func NewEmployeeWILoader(r request.RequestService) EmployeeWILoader {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e EmployeeWILoader) LoadEmployeeList(employeeIDList []string, from, to string) ([]domain.EmployeeWorkInformation, error) {
|
func (e EmployeeWILoader) LoadEmployeeList(employeeIDList []string, from, to string) ([]domain.EmployeeWorkInformation, error) {
|
||||||
reader, err := e.r.EmployeeWorkInformation(employeeIDList, from, to)
|
xmlBts := make([]byte, 0, units.MiB*5)
|
||||||
|
err := e.r.EmployeeWorkInformation(&xmlBts, employeeIDList, from, to)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("r.EmployeeWorkInformation: %s", err)
|
return nil, fmt.Errorf("r.EmployeeWorkInformation: %s", err)
|
||||||
}
|
}
|
||||||
|
if len(xmlBts) <= 0 {
|
||||||
|
return nil, fmt.Errorf("couldn't load xml ")
|
||||||
|
}
|
||||||
|
|
||||||
return loadFromXML(reader)
|
return loadFromXML(xmlBts)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e EmployeeWILoader) LoadEmployee(employeeID, from, to string) ([]domain.EmployeeWorkInformation, error) {
|
func (e EmployeeWILoader) LoadEmployee(employeeID, from, to string) ([]domain.EmployeeWorkInformation, error) {
|
||||||
employeeIDList := []string{employeeID}
|
employeeIDList := []string{employeeID}
|
||||||
|
xmlBts := make([]byte, 0, units.MiB*5)
|
||||||
reader, err := e.r.EmployeeWorkInformation(employeeIDList, from, to)
|
err := e.r.EmployeeWorkInformation(&xmlBts, employeeIDList, from, to)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("r.EmployeeWorkInformation: %s", err)
|
return nil, fmt.Errorf("r.EmployeeWorkInformation: %s", err)
|
||||||
}
|
}
|
||||||
|
if len(xmlBts) <= 0 {
|
||||||
return loadFromXML(reader)
|
return nil, fmt.Errorf("couldn't load xml ")
|
||||||
|
}
|
||||||
|
return loadFromXML(xmlBts)
|
||||||
}
|
}
|
||||||
|
|
||||||
func loadFromXML(xmlFile io.Reader) ([]domain.EmployeeWorkInformation, error) {
|
func loadFromXML(xmlFile []byte) ([]domain.EmployeeWorkInformation, error) {
|
||||||
var awi domain.ArrayOfEmployeeWorkInformation
|
var awi domain.ArrayOfEmployeeWorkInformation
|
||||||
|
if err := xml.Unmarshal(xmlFile, &awi); err != nil {
|
||||||
if err := xml.NewDecoder(xmlFile).Decode(&awi); err != nil {
|
return nil, fmt.Errorf("xml.Unmarshal: %s", err)
|
||||||
return nil, fmt.Errorf("xml.NewDecoder.Decode: %s", err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return awi.EmployeeWorkInfos, nil
|
return awi.EmployeeWorkInfos, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,22 @@
|
||||||
|
package xml_loader_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/config"
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/internal/request"
|
||||||
|
"gitea.urkob.com/urko/ess-etl-go/internal/xml_loader"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestEmployeeWILoader_LoadEmployee(t *testing.T) {
|
||||||
|
cfg := config.NewConfig(".env")
|
||||||
|
|
||||||
|
r := request.NewRequestService(cfg.AmsApi, cfg.AmsApiKey)
|
||||||
|
|
||||||
|
loader := xml_loader.NewEmployeeWILoader(r)
|
||||||
|
employeeID, from, to := cfg.EmployeeIdList[0], "2023-01-01", "2023-01-31"
|
||||||
|
got, err := loader.LoadEmployee(employeeID, from, to)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Greater(t, len(got), 0)
|
||||||
|
}
|
Loading…
Reference in New Issue