Skip to content

Commit 4e10db9

Browse files
author
Satish Ramachandran
committed
Add external contracts support (import and export) for ACI
1 parent 8dd9e0d commit 4e10db9

File tree

6 files changed

+464
-13
lines changed

6 files changed

+464
-13
lines changed

netctl/commands.go

+46
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,14 @@ var Commands = []cli.Command{
5555
Name: "policy, p",
5656
Usage: "Policy List (separated by commas)",
5757
},
58+
cli.StringFlag{
59+
Name: "consumed-external-contracts, i",
60+
Usage: "Consumed external contracts(separated by commas)",
61+
},
62+
cli.StringFlag{
63+
Name: "provided-external-contracts, e",
64+
Usage: "Provided external contracts(separated by commas)",
65+
},
5866
},
5967
Action: createEndpointGroup,
6068
},
@@ -256,6 +264,44 @@ var Commands = []cli.Command{
256264
},
257265
},
258266
},
267+
{
268+
Name: "external-contracts",
269+
Usage: "External contracts",
270+
Subcommands: []cli.Command{
271+
{
272+
Name: "ls",
273+
Aliases: []string{"list"},
274+
Usage: "List external contracts",
275+
Flags: []cli.Flag{quietFlag, jsonFlag},
276+
Action: listExternalContracts,
277+
},
278+
{
279+
Name: "rm",
280+
Aliases: []string{"delete"},
281+
Usage: "Delete external contracts",
282+
Action: deleteExternalContracts,
283+
},
284+
{
285+
Name: "create",
286+
Usage: "Create external contracts",
287+
Flags: []cli.Flag{
288+
cli.BoolFlag{
289+
Name: "consumed, c",
290+
Usage: "External contracts type - consumed",
291+
},
292+
cli.BoolFlag{
293+
Name: "provided, p",
294+
Usage: "External contracts type - provided",
295+
},
296+
cli.StringFlag{
297+
Name: "contracts, a",
298+
Usage: "Contracts (separated by commas)",
299+
},
300+
},
301+
Action: createExternalContracts,
302+
},
303+
},
304+
},
259305
{
260306
Name: "global",
261307
Usage: "Global information",

netctl/netctl.go

+80-4
Original file line numberDiff line numberDiff line change
@@ -401,11 +401,23 @@ func createEndpointGroup(ctx *cli.Context) {
401401
policies = []string{}
402402
}
403403

404+
consExtContractsGrps := strings.Split(ctx.String("consumed-external-contracts"), ",")
405+
if ctx.String("consumed-external-contracts") == "" {
406+
consExtContractsGrps = []string{}
407+
}
408+
409+
provExtContractsGrps := strings.Split(ctx.String("provided-external-contracts"), ",")
410+
if ctx.String("provided-external-contracts") == "" {
411+
provExtContractsGrps = []string{}
412+
}
413+
404414
errCheck(ctx, getClient(ctx).EndpointGroupPost(&contivClient.EndpointGroup{
405-
TenantName: tenant,
406-
NetworkName: network,
407-
GroupName: group,
408-
Policies: policies,
415+
TenantName: tenant,
416+
NetworkName: network,
417+
GroupName: group,
418+
Policies: policies,
419+
ConsExtContractsGrps: consExtContractsGrps,
420+
ProvExtContractsGrps: provExtContractsGrps,
409421
}))
410422
}
411423

@@ -800,3 +812,67 @@ func listServiceLB(ctx *cli.Context) {
800812
}
801813
}
802814
}
815+
816+
func listExternalContracts(ctx *cli.Context) {
817+
argCheck(0, ctx)
818+
819+
extContractsGroups, err := getClient(ctx).ExtContractsGroupList()
820+
errCheck(ctx, err)
821+
822+
if ctx.Bool("json") {
823+
dumpJSONList(ctx, extContractsGroups)
824+
} else if ctx.Bool("quiet") {
825+
contractsGroupNames := ""
826+
for _, extContractsGroup := range *extContractsGroups {
827+
contractsGroupNames += extContractsGroup.ContractsGroupName + "\n"
828+
}
829+
os.Stdout.WriteString(contractsGroupNames)
830+
} else {
831+
writer := tabwriter.NewWriter(os.Stdout, 0, 2, 2, ' ', 0)
832+
defer writer.Flush()
833+
for _, extContractsGroup := range *extContractsGroups {
834+
writer.Write([]byte(fmt.Sprintf(" Name: %s\t\tType: %s\n", extContractsGroup.ContractsGroupName, extContractsGroup.ContractsType)))
835+
writer.Write([]byte(fmt.Sprintf(" Contracts:\n")))
836+
for _, contract := range extContractsGroup.Contracts {
837+
writer.Write([]byte(fmt.Sprintf("\t\t%s\n", contract)))
838+
}
839+
}
840+
}
841+
}
842+
843+
func deleteExternalContracts(ctx *cli.Context) {
844+
argCheck(1, ctx)
845+
contractsGroupName := ctx.Args()[0]
846+
847+
logrus.Infof("Deleting external contracts group %s", contractsGroupName)
848+
errCheck(ctx, getClient(ctx).ExtContractsGroupDelete(contractsGroupName))
849+
850+
}
851+
852+
func createExternalContracts(ctx *cli.Context) {
853+
argCheck(1, ctx)
854+
855+
var contractsType string
856+
if ctx.Bool("provided") && ctx.Bool("consumed") {
857+
errExit(ctx, exitHelp, "Cannot use both provided and consumed", false)
858+
} else if ctx.Bool("provided") {
859+
contractsType = "provided"
860+
} else if ctx.Bool("consumed") {
861+
contractsType = "consumed"
862+
} else {
863+
errExit(ctx, exitHelp, "Either provided or consumed must be specified", false)
864+
}
865+
866+
contracts := strings.Split(ctx.String("contracts"), ",")
867+
if ctx.String("contracts") == "" {
868+
errExit(ctx, exitHelp, "Contracts not provided", false)
869+
}
870+
871+
contractsGroupName := ctx.Args()[0]
872+
873+
errCheck(ctx, getClient(ctx).ExtContractsGroupPost(&contivClient.ExtContractsGroup{
874+
ContractsGroupName: contractsGroupName,
875+
ContractsType: contractsType,
876+
Contracts: contracts,
877+
}))
878+
}

netmaster/objApi/apiController.go

+170-3
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ import (
2222
"github.com/contiv/netplugin/netmaster/intent"
2323
"github.com/contiv/netplugin/netmaster/master"
2424
"github.com/contiv/netplugin/utils"
25+
"github.com/contiv/netplugin/utils/extContracts"
2526
"github.com/contiv/objdb/modeldb"
2627
"strconv"
2728
"strings"
@@ -57,9 +58,8 @@ func NewAPIController(router *mux.Router, storeURL string) *APIController {
5758
contivModel.RegisterRuleCallbacks(ctrler)
5859
contivModel.RegisterTenantCallbacks(ctrler)
5960
contivModel.RegisterBgpCallbacks(ctrler)
60-
contivModel.RegisterBgpCallbacks(ctrler)
6161
contivModel.RegisterServiceLBCallbacks(ctrler)
62-
62+
contivModel.RegisterExtContractsGroupCallbacks(ctrler)
6363
// Register routes
6464
contivModel.AddRoutes(router)
6565

@@ -343,6 +343,12 @@ func endpointGroupCleanup(endpointGroup *contivModel.EndpointGroup) {
343343
policy.Write()
344344
}
345345

346+
// Cleanup any external contracts
347+
err = extContracts.CleanupExternalContracts(endpointGroup)
348+
if err != nil {
349+
return err
350+
}
351+
346352
// Remove the endpoint group from network and tenant link sets.
347353
nwObjKey := endpointGroup.TenantName + ":" + endpointGroup.NetworkName
348354
network := contivModel.FindNetwork(nwObjKey)
@@ -360,6 +366,107 @@ func endpointGroupCleanup(endpointGroup *contivModel.EndpointGroup) {
360366
// FIXME: hack to allocate unique endpoint group ids
361367
var globalEpgID = 1
362368

369+
/*
370+
// Some utility functions to work with the external contracts
371+
func extContractsGrpDeregister(epg *contivModel.EndpointGroup, contractsGrp *contivModel.ExtContractsGroup, contractType string) error {
372+
if contractsGrp == nil {
373+
errStr := fmt.Sprintf("%s External contracts group not found", contractType)
374+
log.Errorf(errStr)
375+
return core.Errorf(errStr)
376+
}
377+
378+
modeldb.RemoveLinkSet(&contractsGrp.LinkSets.EndpointGroups, epg)
379+
if contractType == "provided" {
380+
modeldb.RemoveLinkSet(&epg.LinkSets.ProvExtContractsGrps, contractsGrp)
381+
} else if contractType == "consumed" {
382+
modeldb.RemoveLinkSet(&epg.LinkSets.ConsExtContractsGrps, contractsGrp)
383+
}
384+
// Links broken, update the contracts group object.
385+
err := contractsGrp.Write()
386+
if err != nil {
387+
return err
388+
}
389+
390+
return nil
391+
}
392+
393+
func extContractsGrpValidateAndRegister(epg *contivModel.EndpointGroup, contractsGrp *contivModel.ExtContractsGroup, contractType string) error {
394+
if contractsGrp == nil {
395+
errStr := fmt.Sprintf("%s External contracts group not found", contractType)
396+
log.Errorf(errStr)
397+
return core.Errorf(errStr)
398+
}
399+
400+
if strings.ToLower(contractsGrp.ContractsType) != contractType {
401+
errStr := fmt.Sprintf("Incorrect type for contract group: %v", contractsGrp)
402+
log.Errorf(errStr)
403+
return core.Errorf(errStr)
404+
}
405+
406+
// Establish the necessary links.
407+
// TBD: Error handling. In the loop, we are bailing out
408+
// on failure. How do we take care of already established
409+
// links?
410+
modeldb.AddLinkSet(&contractsGrp.LinkSets.EndpointGroups, epg)
411+
if contractType == "provided" {
412+
modeldb.AddLinkSet(&epg.LinkSets.ProvExtContractsGrps, contractsGrp)
413+
} else if contractType == "consumed" {
414+
modeldb.AddLinkSet(&epg.LinkSets.ConsExtContractsGrps, contractsGrp)
415+
}
416+
417+
// Links made, write the policy set object.
418+
err := contractsGrp.Write()
419+
if err != nil {
420+
return err
421+
}
422+
423+
return nil
424+
}
425+
426+
func cleanupExternalContracts(endpointGroup *contivModel.EndpointGroup) error {
427+
// Cleanup consumed external contracts
428+
for _, consExtContractsGrp := range endpointGroup.ConsExtContractsGrps {
429+
contractsGrp := contivModel.FindExtContractsGroup(consExtContractsGrp)
430+
err = extContractsGrpDeregister(endpointGroup, contractsGrp, "consumed")
431+
if err != nil {
432+
return err
433+
}
434+
}
435+
436+
// Cleanup provided external contracts
437+
for _, provExtContractsGrp := range endpointGroup.ProvExtContractsGrps {
438+
contractsGrp := contivModel.FindExtContractsGroup(provExtContractsGrp)
439+
err = extContractsGrpDeregister(endpointGroup, contractsGrp, "provided")
440+
if err != nil {
441+
return err
442+
}
443+
}
444+
445+
return nil
446+
}
447+
448+
func setupExternalContracts(endpointGroup *contivModel.EndpointGroup, consContractsGrps, provContractsGrps string[]) error {
449+
// Validate presence and register consumed external contracts
450+
for _, consExtContractsGrp := range consContractsGrps {
451+
contractsGrp := contivModel.FindExtContractsGroup(consExtContractsGrp)
452+
err = extContractsGrpValidateAndRegister(endpointGroup, contractsGrp, "consumed")
453+
if err != nil {
454+
return err
455+
}
456+
}
457+
458+
// Validate presence and register provided external contracts
459+
for _, provExtContractsGrp := range provContractsGrps {
460+
contractsGrp := contivModel.FindExtContractsGroup(provExtContractsGrp)
461+
err = extContractsGrpValidateAndRegister(endpointGroup, contractsGrp, "provided")
462+
if err != nil {
463+
return err
464+
}
465+
}
466+
467+
return nil
468+
}
469+
*/
363470
// EndpointGroupCreate creates end point group
364471
func (ac *APIController) EndpointGroupCreate(endpointGroup *contivModel.EndpointGroup) error {
365472
log.Infof("Received EndpointGroupCreate: %+v", endpointGroup)
@@ -391,6 +498,12 @@ func (ac *APIController) EndpointGroupCreate(endpointGroup *contivModel.Endpoint
391498
return err
392499
}
393500

501+
// Setup external contracts this EPG might have.
502+
err = extContracts.SetupExternalContracts(endpointGroup, endpointGroup.ConsExtContractsGrps, endpointGroup.ProvExtContractsGrps)
503+
if err != nil {
504+
return err
505+
}
506+
394507
// for each policy create an epg policy Instance
395508
for _, policyName := range endpointGroup.Policies {
396509
policyKey := endpointGroup.TenantName + ":" + policyName
@@ -512,10 +625,27 @@ func (ac *APIController) EndpointGroupUpdate(endpointGroup, params *contivModel.
512625
}
513626
}
514627
}
515-
516628
// Update the policy list
517629
endpointGroup.Policies = params.Policies
518630

631+
// For the external contracts, we can keep the update simple. Remove
632+
// all that we have now, and update the epg with the new list.
633+
// Step 1: Cleanup existing external contracts.
634+
err := extContracts.CleanupExternalContracts(endpointGroup)
635+
if err != nil {
636+
return err
637+
}
638+
// Step 2: Add contracts from the update.
639+
// Consumed contracts
640+
err = extContracts.SetupExternalContracts(endpointGroup, params.ConsExtContractsGrps, params.ProvExtContractsGrps)
641+
if err != nil {
642+
return err
643+
}
644+
645+
// Update the epg itself with the new contracts groups.
646+
endpointGroup.ConsExtContractsGrps = params.ConsExtContractsGrps
647+
endpointGroup.ProvExtContractsGrps = params.ProvExtContractsGrps
648+
519649
// if there is an associated app profiles, update that as well
520650
profKey := endpointGroup.Links.AppProfile.ObjKey
521651
profObj := contivModel.FindAppProfile(profKey)
@@ -1135,3 +1265,40 @@ func validatePorts(ports []string) bool {
11351265
}
11361266
return true
11371267
}
1268+
1269+
func isExtContractsGroupUsed(contractsGroup *contivModel.ExtContractsGroup) bool {
1270+
if len(contractsGroup.LinkSets.EndpointGroups) > 0 {
1271+
return true
1272+
}
1273+
1274+
return false
1275+
}
1276+
1277+
// ExtContractsGroupCreate creates a new group of external contracts
1278+
func (ac *APIController) ExtContractsGroupCreate(contractsGroup *contivModel.ExtContractsGroup) error {
1279+
log.Infof("Received ExtContractsGroupCreate: %+v", contractsGroup)
1280+
1281+
return nil
1282+
}
1283+
1284+
// ExtContractsGroupUpdate updates an existing group of contract sets
1285+
func (ac *APIController) ExtContractsGroupUpdate(contractsGroup, params *contivModel.ExtContractsGroup) error {
1286+
log.Infof("Received ExtContractsGroupUpdate: %+v, params: %+v", contractsGroup, params)
1287+
log.Errorf("Error: external contracts update not supported: %s", contractsGroup.ContractsGroupName)
1288+
1289+
return core.Errorf("external contracts update not supported")
1290+
}
1291+
1292+
// ExtContractsGroupDelete deletes an existing external contracts group
1293+
func (ac *APIController) ExtContractsGroupDelete(contractsGroup *contivModel.ExtContractsGroup) error {
1294+
log.Infof("Received ExtContractsGroupDelete: %+v", contractsGroup)
1295+
1296+
// At this moment, we let the external contracts to be deleted only
1297+
// if there are no consumers of this external contracts group
1298+
if isExtContractsGroupUsed(contractsGroup) == true {
1299+
log.Errorf("Error: External contracts groups is being used: %s", contractsGroup.ContractsGroupName)
1300+
return core.Errorf("External contracts group is in-use")
1301+
}
1302+
1303+
return nil
1304+
}

0 commit comments

Comments
 (0)