2022-10-03 12:12:20 +02:00
|
|
|
package contextargs
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"net/http/cookiejar"
|
2023-08-04 20:21:22 +05:30
|
|
|
"sync/atomic"
|
2022-10-03 12:12:20 +02:00
|
|
|
|
2023-06-09 19:52:56 +05:30
|
|
|
"github.com/projectdiscovery/gologger"
|
2023-07-24 16:50:28 +02:00
|
|
|
mapsutil "github.com/projectdiscovery/utils/maps"
|
2022-10-03 12:12:20 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// Context implements a shared context struct to share information across multiple templates within a workflow
|
|
|
|
|
type Context struct {
|
2022-11-09 14:18:56 +01:00
|
|
|
// Meta is the target for the executor
|
|
|
|
|
MetaInput *MetaInput
|
|
|
|
|
|
2022-10-03 12:12:20 +02:00
|
|
|
// CookieJar shared within workflow's http templates
|
|
|
|
|
CookieJar *cookiejar.Jar
|
|
|
|
|
|
|
|
|
|
// Args is a workflow shared key-value store
|
2023-07-24 16:50:28 +02:00
|
|
|
args *mapsutil.SyncLockMap[string, interface{}]
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create a new contextargs instance
|
|
|
|
|
func New() *Context {
|
2023-06-09 19:52:56 +05:30
|
|
|
return NewWithInput("")
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create a new contextargs instance with input string
|
|
|
|
|
func NewWithInput(input string) *Context {
|
2023-06-09 19:52:56 +05:30
|
|
|
jar, err := cookiejar.New(nil)
|
|
|
|
|
if err != nil {
|
2023-08-04 20:21:22 +05:30
|
|
|
gologger.Error().Msgf("contextargs: could not create cookie jar: %s\n", err)
|
|
|
|
|
}
|
|
|
|
|
return &Context{
|
|
|
|
|
MetaInput: &MetaInput{Input: input},
|
|
|
|
|
CookieJar: jar,
|
|
|
|
|
args: &mapsutil.SyncLockMap[string, interface{}]{
|
|
|
|
|
Map: make(map[string]interface{}),
|
|
|
|
|
ReadOnly: atomic.Bool{},
|
|
|
|
|
},
|
2023-06-09 19:52:56 +05:30
|
|
|
}
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set the specific key-value pair
|
|
|
|
|
func (ctx *Context) Set(key string, value interface{}) {
|
2023-07-24 16:50:28 +02:00
|
|
|
_ = ctx.args.Set(key, value)
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ctx *Context) hasArgs() bool {
|
2023-08-04 20:21:22 +05:30
|
|
|
return !ctx.args.IsEmpty()
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-31 18:03:01 +05:30
|
|
|
// Merge the key-value pairs
|
|
|
|
|
func (ctx *Context) Merge(args map[string]interface{}) {
|
|
|
|
|
_ = ctx.args.Merge(args)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add the specific key-value pair
|
|
|
|
|
func (ctx *Context) Add(key string, v interface{}) {
|
|
|
|
|
values, ok := ctx.args.Get(key)
|
|
|
|
|
if !ok {
|
|
|
|
|
ctx.Set(key, v)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If the key exists, append the value to the existing value
|
|
|
|
|
switch v := v.(type) {
|
|
|
|
|
case []string:
|
|
|
|
|
if values, ok := values.([]string); ok {
|
|
|
|
|
values = append(values, v...)
|
|
|
|
|
ctx.Set(key, values)
|
|
|
|
|
}
|
|
|
|
|
case string:
|
|
|
|
|
if values, ok := values.(string); ok {
|
|
|
|
|
tmp := []string{values, v}
|
|
|
|
|
ctx.Set(key, tmp)
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
values, _ := ctx.Get(key)
|
|
|
|
|
ctx.Set(key, []interface{}{values, v})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-03 12:12:20 +02:00
|
|
|
// Get the value with specific key if exists
|
|
|
|
|
func (ctx *Context) Get(key string) (interface{}, bool) {
|
|
|
|
|
if !ctx.hasArgs() {
|
|
|
|
|
return nil, false
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-24 16:50:28 +02:00
|
|
|
return ctx.args.Get(key)
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
2023-08-04 20:21:22 +05:30
|
|
|
func (ctx *Context) GetAll() map[string]interface{} {
|
2022-10-03 12:12:20 +02:00
|
|
|
if !ctx.hasArgs() {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-04 20:21:22 +05:30
|
|
|
return ctx.args.Clone().Map
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ctx *Context) ForEach(f func(string, interface{})) {
|
2023-07-24 16:50:28 +02:00
|
|
|
_ = ctx.args.Iterate(func(k string, v interface{}) error {
|
2022-10-03 12:12:20 +02:00
|
|
|
f(k, v)
|
2023-07-24 16:50:28 +02:00
|
|
|
return nil
|
|
|
|
|
})
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Has check if the key exists
|
|
|
|
|
func (ctx *Context) Has(key string) bool {
|
2023-07-24 16:50:28 +02:00
|
|
|
return ctx.hasArgs() && ctx.args.Has(key)
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (ctx *Context) HasArgs() bool {
|
2023-06-09 19:52:56 +05:30
|
|
|
return !ctx.args.IsEmpty()
|
2022-10-03 12:12:20 +02:00
|
|
|
}
|
2022-11-09 14:18:56 +01:00
|
|
|
|
|
|
|
|
func (ctx *Context) Clone() *Context {
|
|
|
|
|
newCtx := &Context{
|
|
|
|
|
MetaInput: ctx.MetaInput.Clone(),
|
2023-08-31 18:03:01 +05:30
|
|
|
args: ctx.args.Clone(),
|
2022-11-09 14:18:56 +01:00
|
|
|
CookieJar: ctx.CookieJar,
|
|
|
|
|
}
|
|
|
|
|
return newCtx
|
|
|
|
|
}
|