blob: 605d0e4475f66070e22b4a7d2621be8f914a9d1c [file] [log] [blame]
/*
* Copyright 2018-present Open Networking Foundation
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package core
import (
"bytes"
"context"
"fmt"
"os/exec"
"strconv"
"time"
"gerrit.opencord.org/voltha-bbsim/common/logger"
"gerrit.opencord.org/voltha-bbsim/device"
)
// TestManager is the structure for test manager
type TestManager struct {
DhcpServerIP string
Pid []int
testers map[string]map[device.Devkey]*Tester
ctx context.Context
cancel context.CancelFunc
}
// Tester is the structure for Test
type Tester struct {
Type string
Key device.Devkey
Testfunc func(device.Devkey) error
Waitsec int
ctx context.Context
cancel context.CancelFunc
}
// NewTestManager returns new TestManager
func NewTestManager(opt *option) *TestManager {
t := new(TestManager)
t.DhcpServerIP = opt.dhcpservip
return t
}
// CreateTester creates instance of Tester
func (*TestManager) CreateTester(testtype string, opt *option, key device.Devkey, fn func(device.Devkey) error, waitsec int) *Tester {
logger.Debug("CreateTester() called")
t := new(Tester)
t.Type = testtype
t.Key = key
t.Testfunc = fn
t.Waitsec = waitsec
return t
}
// Start does starting action - Blocking Call
func (tm *TestManager) Start() error {
ctx, cancel := context.WithCancel(context.Background())
tm.ctx = ctx
tm.cancel = cancel
tm.testers = make(map[string]map[device.Devkey]*Tester)
tm.testers["AAA"] = map[device.Devkey]*Tester{}
tm.testers["DHCP"] = map[device.Devkey]*Tester{}
logger.Info("TestManager start")
return nil
}
// Stop does stopping action
func (tm *TestManager) Stop() error {
if tm.cancel != nil {
tm.cancel()
}
tm.Initialize()
logger.Debug("TestManager Done")
return nil
}
func (tm *TestManager) StartTester(t *Tester) error {
testtype := t.Type
key := t.Key
waitsec := t.Waitsec
// TODO use timers instead of this crazy loop: https://gobyexample.com/timers
logger.Debug("StartTester type:%s called with key:%v", testtype, key)
child, cancel := context.WithCancel(tm.ctx)
t.ctx = child
t.cancel = cancel
go func() error {
tick := time.NewTicker(time.Second)
counter := 0
defer func() {
tick.Stop()
logger.Debug("Tester type:%s with key %v Done", testtype, key)
}()
L:
for counter < waitsec {
select {
case <-tick.C:
counter++
if counter == waitsec { // TODO: This should be fixed
break L
}
case <-child.Done():
return nil
}
}
err := t.Testfunc(key)
if err != nil {
return err
}
return nil
}()
tm.testers[testtype][key] = t
return nil
}
// StopTester stops the test
func (tm *TestManager) StopTester(testtype string, key device.Devkey) error {
ts := tm.testers[testtype][key]
ts.cancel()
delete(tm.testers[testtype], key)
return nil
}
// Initialize test manager
func (tm *TestManager) Initialize() {
logger.Info("TestManager Initialize () called")
pids := tm.Pid
logger.Debug("Runnig Process: %v", pids)
KillProcesses(pids)
exec.Command("rm", "/var/run/dhcpd.pid").Run() // This is for DHCP server activation
exec.Command("touch", "/var/run/dhcpd.pid").Run() // This is for DHCP server activation
}
// KillProcesses kill process by specified pid
func KillProcesses(pids []int) error {
for _, pname := range pids {
killProcess(pname)
}
return nil
}
func killProcess(pid int) error {
err := exec.Command("kill", strconv.Itoa(pid)).Run()
if err != nil {
logger.Error("Fail to kill %d: %v", pid, err)
return err
}
logger.Info("Successfully killed %d", pid)
return nil
}
func activateWPASupplicant(key device.Devkey) (err error) {
if err = startEAPClient(key.Intfid, key.ID); err != nil {
errmsg := fmt.Sprintf("Failed to activate WPA Supplicant intfid: %d onuid: %d", key.Intfid, key.ID)
logger.Error(errmsg)
}
logger.Info("Successfuly activateWPASupplicant() for intfid:%d onuid:%d", key.Intfid, key.ID)
return nil
}
func activateDHCPClient(key device.Devkey) (err error) {
if err = startDHCPClient(key.Intfid, key.ID); err != nil {
errmsg := fmt.Sprintf("Failed to activate DHCP client intfid: %d onuid: %d", key.Intfid, key.ID)
logger.Error(errmsg)
}
return nil
}
func activateDHCPServer(veth string, serverip string) error {
err := exec.Command("ip", "addr", "add", serverip, "dev", veth).Run()
if err != nil {
logger.Error("Fail to add ip to %s address: %s", veth, err)
return err
}
err = exec.Command("ip", "link", "set", veth, "up").Run()
if err != nil {
logger.Error("Fail to set %s up: %s", veth, err)
return err
}
dhcp := "/usr/local/bin/dhcpd"
conf := "/etc/dhcp/dhcpd.conf"
logfile := "/tmp/dhcplog"
var stderr bytes.Buffer
cmd := exec.Command(dhcp, "-cf", conf, veth, "-tf", logfile)
cmd.Stderr = &stderr
err = cmd.Run()
if err != nil {
logger.Error("Fail to activateDHCP Server (): %s, %s", err, stderr.String())
return err
}
logger.Info("DHCP Server is successfully activated !")
return err
}