586 lines
18 KiB
Go
Executable File
586 lines
18 KiB
Go
Executable File
package application
|
|
|
|
import (
|
|
"context"
|
|
"crypto/rand"
|
|
"encoding/base64"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"sort"
|
|
"sync"
|
|
"time"
|
|
|
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/identification"
|
|
"git.coopgo.io/coopgo-apps/parcoursmob/utils/sorting"
|
|
agenda "git.coopgo.io/coopgo-platform/agenda/grpcapi"
|
|
agendastorage "git.coopgo.io/coopgo-platform/agenda/storage"
|
|
fleets "git.coopgo.io/coopgo-platform/fleets/grpcapi"
|
|
fleetsstorage "git.coopgo.io/coopgo-platform/fleets/storage"
|
|
groupsmanagement "git.coopgo.io/coopgo-platform/groups-management/grpcapi"
|
|
"git.coopgo.io/coopgo-platform/groups-management/storage"
|
|
groupstorage "git.coopgo.io/coopgo-platform/groups-management/storage"
|
|
accounts "git.coopgo.io/coopgo-platform/mobility-accounts/grpcapi"
|
|
mobilityaccountsstorage "git.coopgo.io/coopgo-platform/mobility-accounts/storage"
|
|
"github.com/google/uuid"
|
|
"github.com/gorilla/mux"
|
|
"github.com/rs/zerolog/log"
|
|
"google.golang.org/protobuf/types/known/structpb"
|
|
)
|
|
|
|
func (h *ApplicationHandler) Administration(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
wg sync.WaitGroup
|
|
accounts, beneficiaries []mobilityaccountsstorage.Account
|
|
bookings []fleetsstorage.Booking
|
|
accountsErr, beneficiariesErr, bookingsErr, groupsResponseErr, eventsResponseErr, groupsBatchErr error
|
|
groups = []groupstorage.Group{}
|
|
responses = []agendastorage.Event{}
|
|
groupsResponse *groupsmanagement.GetGroupsResponse
|
|
eventsResponse *agenda.GetEventsResponse
|
|
groupids = []string{}
|
|
groupsBatchResponse *groupsmanagement.GetGroupsBatchResponse
|
|
)
|
|
// Retrieve accounts in a goroutine
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
accounts, accountsErr = h.services.GetAccounts()
|
|
}()
|
|
// Retrieve beneficiaries in a goroutine
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
beneficiaries, beneficiariesErr = h.services.GetBeneficiaries()
|
|
}()
|
|
// Retrieve bookings in a goroutine
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
bookings, bookingsErr = h.services.GetBookings()
|
|
}()
|
|
// Retrieve groupsRequest in a goroutine
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
request := &groupsmanagement.GetGroupsRequest{
|
|
Namespaces: []string{"parcoursmob_organizations"},
|
|
}
|
|
groupsResponse, groupsResponseErr = h.services.GRPC.GroupsManagement.GetGroups(context.TODO(), request)
|
|
for _, group := range groupsResponse.Groups {
|
|
g := group.ToStorageType()
|
|
groups = append(groups, g)
|
|
}
|
|
sort.Sort(sorting.GroupsByName(groups))
|
|
}()
|
|
// Retrieve Events in a goroutine
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
eventsResponse, eventsResponseErr = h.services.GRPC.Agenda.GetEvents(context.TODO(), &agenda.GetEventsRequest{
|
|
Namespaces: []string{"parcoursmob_dispositifs"},
|
|
})
|
|
for _, e := range eventsResponse.Events {
|
|
groupids = append(groupids, e.Owners...)
|
|
responses = append(responses, e.ToStorageType())
|
|
}
|
|
sort.Sort(sorting.EventsByStartdate(responses))
|
|
}()
|
|
wg.Add(1)
|
|
// Retrieve groupsBatch in a goroutine
|
|
go func() {
|
|
defer wg.Done()
|
|
groupsBatchResponse, groupsBatchErr = h.services.GRPC.GroupsManagement.GetGroupsBatch(context.TODO(), &groupsmanagement.GetGroupsBatchRequest{
|
|
Groupids: groupids,
|
|
})
|
|
groupps := map[string]any{}
|
|
if groupsBatchErr == nil {
|
|
for _, g := range groupsBatchResponse.Groups {
|
|
groupps[g.Id] = g.ToStorageType()
|
|
}
|
|
}
|
|
}()
|
|
wg.Wait()
|
|
if accountsErr != nil || beneficiariesErr != nil || bookingsErr != nil || groupsResponseErr != nil || eventsResponseErr != nil {
|
|
log.Error().
|
|
Any("accounts error", accountsErr).
|
|
Any("beneficiaries error", beneficiariesErr).
|
|
Any("bookings error", bookingsErr).
|
|
Any("groups response error", groupsResponseErr).
|
|
Any("events response error", eventsResponseErr).
|
|
Any("groups batch error", groupsBatchErr).
|
|
Msg("Error in retrieving administration data")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
h.Renderer.Administration(w, r, accounts, beneficiaries, groups, bookings, responses)
|
|
}
|
|
|
|
func (h *ApplicationHandler) AdministrationCreateGroup(w http.ResponseWriter, r *http.Request) {
|
|
if r.Method == "POST" {
|
|
r.ParseForm()
|
|
|
|
if r.FormValue("name") == "" {
|
|
|
|
log.Error().Str("name", r.FormValue("name")).Msg("Invalid name")
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
modules := map[string]any{
|
|
"beneficiaries": r.FormValue("modules.beneficiaries") == "on",
|
|
"journeys": r.FormValue("modules.journeys") == "on",
|
|
"vehicles": r.FormValue("modules.vehicles") == "on",
|
|
"vehicles_management": r.FormValue("modules.vehicles_management") == "on",
|
|
"events": r.FormValue("modules.events") == "on",
|
|
"agenda": r.FormValue("modules.agenda") == "on",
|
|
"groups": r.FormValue("modules.groups") == "on",
|
|
"administration": r.FormValue("modules.administration") == "on",
|
|
"support": r.FormValue("modules.support") == "on",
|
|
"group_module": r.FormValue("modules.group_module") == "on",
|
|
}
|
|
|
|
groupid := uuid.NewString()
|
|
|
|
dataMap := map[string]any{
|
|
"name": r.FormValue("name"),
|
|
"modules": modules,
|
|
}
|
|
|
|
data, err := structpb.NewValue(dataMap)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Cannot create PB struct from data map")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
request_organization := &groupsmanagement.AddGroupRequest{
|
|
Group: &groupsmanagement.Group{
|
|
Id: groupid,
|
|
Namespace: "parcoursmob_organizations",
|
|
Data: data.GetStructValue(),
|
|
},
|
|
}
|
|
|
|
request_role := &groupsmanagement.AddGroupRequest{
|
|
Group: &groupsmanagement.Group{
|
|
Id: groupid + ":admin",
|
|
Namespace: "parcoursmob_roles",
|
|
},
|
|
}
|
|
go func() {
|
|
_, err = h.services.GRPC.GroupsManagement.AddGroup(context.TODO(), request_organization)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Groups management service - AddGroup")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
}()
|
|
// Create the admin role for the organization
|
|
go func() {
|
|
_, err = h.services.GRPC.GroupsManagement.AddGroup(context.TODO(), request_role)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Groups management service - AddGroup")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
}()
|
|
http.Redirect(w, r, fmt.Sprintf("/app/administration/groups/%s", groupid), http.StatusFound)
|
|
return
|
|
}
|
|
h.Renderer.AdministrationCreateGroup(w, r)
|
|
}
|
|
|
|
func (h *ApplicationHandler) AdministrationGroupDisplay(w http.ResponseWriter, r *http.Request) {
|
|
vars := mux.Vars(r)
|
|
groupid := vars["groupid"]
|
|
|
|
request := &groupsmanagement.GetGroupRequest{
|
|
Id: groupid,
|
|
}
|
|
|
|
resp, err := h.services.GRPC.GroupsManagement.GetGroup(context.TODO(), request)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Groups management service - GetGroup")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
groupmembers, admins, err := h.groupmembers(groupid)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("inssue retrieving group members")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
h.Renderer.AdministrationGroupDisplay(w, r, resp.Group.ToStorageType(), groupmembers, admins)
|
|
}
|
|
|
|
func (h *ApplicationHandler) AdministrationGroupInviteAdmin(w http.ResponseWriter, r *http.Request) {
|
|
vars := mux.Vars(r)
|
|
groupid := vars["groupid"]
|
|
var (
|
|
groupresp *groupsmanagement.GetGroupResponse
|
|
accountresp *accounts.GetAccountUsernameResponse
|
|
err error
|
|
)
|
|
groupresp, err = h.services.GRPC.GroupsManagement.GetGroup(context.TODO(), &groupsmanagement.GetGroupRequest{
|
|
Id: groupid,
|
|
Namespace: "parcoursmob_organizations",
|
|
})
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Groups management service - GetGroup")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
r.ParseForm()
|
|
|
|
accountresp, err = h.services.GRPC.MobilityAccounts.GetAccountUsername(context.TODO(), &accounts.GetAccountUsernameRequest{
|
|
Username: r.FormValue("username"),
|
|
Namespace: "parcoursmob",
|
|
})
|
|
if err == nil {
|
|
log.Print("account exists")
|
|
// Account already exists : adding the existing account to admin list
|
|
account := accountresp.Account.ToStorageType()
|
|
// account.Data["groups"] = append(account.Data["groups"].([]any), groupid, groupid)
|
|
account.Data["groups"] = append(account.Data["groups"].([]any), groupid, groupid+":admin")
|
|
|
|
as, _ := accounts.AccountFromStorageType(&account)
|
|
|
|
if _, err = h.services.GRPC.MobilityAccounts.UpdateData(context.TODO(), &accounts.UpdateDataRequest{Account: as}); err != nil {
|
|
log.Error().Err(err).Msg("could not set groups to user account")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
data := map[string]any{
|
|
"group": groupresp.Group.ToStorageType().Data["name"],
|
|
}
|
|
|
|
if err := h.emailing.Send("onboarding.existing_administrator", r.FormValue("username"), data); err != nil {
|
|
log.Error().Err(err).Msg("Cannot send email")
|
|
}
|
|
|
|
} else {
|
|
// Onboard now administrator
|
|
onboarding := map[string]any{
|
|
"username": r.FormValue("username"),
|
|
"group": groupid,
|
|
"admin": true,
|
|
}
|
|
|
|
b := make([]byte, 16)
|
|
if _, err := io.ReadFull(rand.Reader, b); err != nil {
|
|
log.Error().Err(err).Msg("Issue creating random bytes")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
key := base64.RawURLEncoding.EncodeToString(b)
|
|
|
|
h.cache.PutWithTTL("onboarding/"+key, onboarding, 168*time.Hour) // 1 week TTL
|
|
|
|
data := map[string]any{
|
|
"group": groupresp.Group.ToStorageType().Data["name"],
|
|
"key": key,
|
|
}
|
|
|
|
if err := h.emailing.Send("onboarding.new_administrator", r.FormValue("username"), data); err != nil {
|
|
log.Error().Err(err).Msg("Cannot send email")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
}
|
|
|
|
http.Redirect(w, r, fmt.Sprintf("/app/administration/groups/%s", groupid), http.StatusFound)
|
|
}
|
|
|
|
func (h *ApplicationHandler) AdministrationGroupInviteMember(w http.ResponseWriter, r *http.Request) {
|
|
vars := mux.Vars(r)
|
|
groupid := vars["groupid"]
|
|
group, err := h.services.GetGroup(groupid)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Groups management service - GetGroup")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
r.ParseForm()
|
|
accountresp, err := h.services.GRPC.MobilityAccounts.GetAccountUsername(context.TODO(), &accounts.GetAccountUsernameRequest{
|
|
Username: r.FormValue("username"),
|
|
Namespace: "parcoursmob",
|
|
})
|
|
if err == nil {
|
|
account := accountresp.Account.ToStorageType()
|
|
account.Data["groups"] = append(account.Data["groups"].([]any), group.ID)
|
|
|
|
as, _ := accounts.AccountFromStorageType(&account)
|
|
|
|
_, err = h.services.GRPC.MobilityAccounts.UpdateData(
|
|
context.TODO(),
|
|
&accounts.UpdateDataRequest{
|
|
Account: as,
|
|
},
|
|
)
|
|
data := map[string]any{
|
|
"group": group.Data["name"],
|
|
}
|
|
|
|
if err := h.emailing.Send("onboarding.existing_member", r.FormValue("username"), data); err != nil {
|
|
log.Error().Err(err).Msg("error sending email onboarding.existing_member")
|
|
}
|
|
|
|
// http.Redirect(w, r, "/app/group/settings", http.StatusFound)
|
|
// return
|
|
} else {
|
|
// Onboard new administrator
|
|
onboarding := map[string]any{
|
|
"username": r.FormValue("username"),
|
|
"group": group.ID,
|
|
"admin": false,
|
|
}
|
|
|
|
b := make([]byte, 16)
|
|
if _, err := io.ReadFull(rand.Reader, b); err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
key := base64.RawURLEncoding.EncodeToString(b)
|
|
|
|
h.cache.PutWithTTL("onboarding/"+key, onboarding, 168*time.Hour) // 1 week TTL
|
|
data := map[string]any{
|
|
"group": group.Data["name"],
|
|
"key": key,
|
|
}
|
|
|
|
if err := h.emailing.Send("onboarding.new_member", r.FormValue("username"), data); err != nil {
|
|
log.Error().Err(err).Msg("error sending email onboarding.new_member")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
}
|
|
http.Redirect(w, r, "/app/administration/groups/"+group.ID, http.StatusFound)
|
|
}
|
|
|
|
func filteVehicle(r *http.Request, v *fleets.Vehicle) bool {
|
|
g := r.Context().Value(identification.GroupKey)
|
|
if g == nil {
|
|
return false
|
|
}
|
|
|
|
group := g.(storage.Group)
|
|
|
|
for _, n := range v.Administrators {
|
|
if n == group.ID {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func (h ApplicationHandler) AdminStatVehicles(w http.ResponseWriter, r *http.Request) {
|
|
bookings := []fleetsstorage.Booking{}
|
|
administrators := []string{}
|
|
reequest := &fleets.GetVehiclesRequest{
|
|
Namespaces: []string{"parcoursmob"},
|
|
}
|
|
reesp, err := h.services.GRPC.Fleets.GetVehicles(context.TODO(), reequest)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Fleets service - GetVehicles")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
}
|
|
|
|
vehicles := []fleetsstorage.Vehicle{}
|
|
for _, vehiicle := range reesp.Vehicles {
|
|
|
|
v := vehiicle.ToStorageType()
|
|
adminfound := false
|
|
for _, a := range administrators {
|
|
if a == v.Administrators[0] {
|
|
adminfound = true
|
|
break
|
|
}
|
|
}
|
|
if !adminfound {
|
|
administrators = append(administrators, v.Administrators[0])
|
|
}
|
|
|
|
vehicleBookings := []fleetsstorage.Booking{}
|
|
for _, b := range v.Bookings {
|
|
if b.Unavailableto.After(time.Now()) {
|
|
vehicleBookings = append(vehicleBookings, b)
|
|
}
|
|
}
|
|
|
|
v.Bookings = vehicleBookings
|
|
|
|
vehicles = append(vehicles, v)
|
|
|
|
}
|
|
groups := map[string]any{}
|
|
|
|
if len(administrators) > 0 {
|
|
admingroups, err := h.services.GRPC.GroupsManagement.GetGroupsBatch(context.TODO(), &groupsmanagement.GetGroupsBatchRequest{
|
|
Groupids: administrators,
|
|
})
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Groups management service - GetGroupsBatch")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
for _, g := range admingroups.Groups {
|
|
groups[g.Id] = g.ToStorageType()
|
|
}
|
|
|
|
}
|
|
sort.Sort(sorting.VehiclesByLicencePlate(vehicles))
|
|
sort.Sort(sorting.BookingsByStartdate(bookings))
|
|
h.Renderer.AdminStatVehicles(w, r, vehicles, bookings, groups)
|
|
}
|
|
|
|
func (h ApplicationHandler) AdminStatBookings(w http.ResponseWriter, r *http.Request) {
|
|
vehicles := map[string]fleetsstorage.Vehicle{}
|
|
bookings := []fleetsstorage.Booking{}
|
|
|
|
reequest := &fleets.GetVehiclesRequest{
|
|
Namespaces: []string{"parcoursmob"},
|
|
}
|
|
reesp, err := h.services.GRPC.Fleets.GetVehicles(context.TODO(), reequest)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Fleets service - GetVehicles")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
beneficiaries_ids := []string{}
|
|
|
|
for _, vehicle := range reesp.Vehicles {
|
|
|
|
v := vehicle.ToStorageType()
|
|
|
|
for _, b := range v.Bookings {
|
|
bookings = append(bookings, b)
|
|
beneficiaries_ids = append(beneficiaries_ids, b.Driver)
|
|
}
|
|
|
|
vehicles[v.ID] = v
|
|
|
|
}
|
|
|
|
groups := map[string]any{}
|
|
|
|
admingroups, err := h.services.GRPC.GroupsManagement.GetGroups(context.TODO(), &groupsmanagement.GetGroupsRequest{
|
|
Namespaces: []string{"parcoursmob_organizations"},
|
|
})
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Groups management service - GetGroups")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
for _, g := range admingroups.Groups {
|
|
groups[g.Id] = g.ToStorageType()
|
|
}
|
|
|
|
beneficiaries, err := h.services.GRPC.MobilityAccounts.GetAccountsBatch(context.TODO(), &accounts.GetAccountsBatchRequest{
|
|
Accountids: beneficiaries_ids,
|
|
})
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Mobility accounts service - GetAccountsBatch")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
beneficiaries_map := map[string]any{}
|
|
for _, ben := range beneficiaries.Accounts {
|
|
beneficiaries_map[ben.Id] = ben.ToStorageType()
|
|
}
|
|
|
|
sort.Sort(sorting.BookingsByStartdate(bookings))
|
|
h.Renderer.AdminStatBookings(w, r, vehicles, bookings, groups, beneficiaries_map)
|
|
}
|
|
|
|
func (h *ApplicationHandler) members() ([]*accounts.Account, error) {
|
|
resp, err := h.services.GRPC.MobilityAccounts.GetAccounts(context.TODO(), &accounts.GetAccountsRequest{
|
|
Namespaces: []string{"parcoursmob"},
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return resp.Accounts, nil
|
|
}
|
|
|
|
func (h *ApplicationHandler) groupmembers(groupid string) (groupmembers []mobilityaccountsstorage.Account, admins []mobilityaccountsstorage.Account, err error) {
|
|
members, err := h.members()
|
|
if err != nil {
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Cannot get members")
|
|
return nil, nil, err
|
|
}
|
|
}
|
|
|
|
groupmembers = []mobilityaccountsstorage.Account{}
|
|
admins = []mobilityaccountsstorage.Account{}
|
|
|
|
for _, m := range members {
|
|
mm := m.ToStorageType()
|
|
for _, g := range mm.Data["groups"].([]any) {
|
|
if g.(string) == groupid {
|
|
groupmembers = append(groupmembers, mm)
|
|
}
|
|
if g.(string) == groupid+":admin" {
|
|
admins = append(admins, mm)
|
|
}
|
|
}
|
|
}
|
|
|
|
return groupmembers, admins, err
|
|
}
|
|
|
|
func (h ApplicationHandler) AdminStatBeneficaires(w http.ResponseWriter, r *http.Request) {
|
|
beneficiaries, err := h.services.GetBeneficiaries()
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Cannot get beneficiaries")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
cacheid := uuid.NewString()
|
|
h.cache.PutWithTTL(cacheid, beneficiaries, 1*time.Hour)
|
|
h.Renderer.AdminStatBeneficaires(w, r, beneficiaries, cacheid)
|
|
}
|
|
|
|
func (h ApplicationHandler) AdminStatEvents(w http.ResponseWriter, r *http.Request) {
|
|
resp, err := h.services.GRPC.Agenda.GetEvents(context.TODO(), &agenda.GetEventsRequest{
|
|
Namespaces: []string{"parcoursmob_dispositifs"},
|
|
})
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Issue in Agenda service - GetEvents")
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
responses := []agendastorage.Event{}
|
|
|
|
groupids := []string{}
|
|
for _, e := range resp.Events {
|
|
groupids = append(groupids, e.Owners...)
|
|
responses = append(responses, e.ToStorageType())
|
|
}
|
|
|
|
sort.Sort(sorting.EventsByStartdate(responses))
|
|
|
|
groupsresp, err := h.services.GRPC.GroupsManagement.GetGroupsBatch(context.TODO(), &groupsmanagement.GetGroupsBatchRequest{
|
|
Groupids: groupids,
|
|
})
|
|
groups := map[string]any{}
|
|
|
|
if err == nil {
|
|
for _, g := range groupsresp.Groups {
|
|
groups[g.Id] = g.ToStorageType()
|
|
}
|
|
}
|
|
h.Renderer.AdminStatEvents(w, r, responses, groups)
|
|
}
|