nuclei/v2/pkg/templates/compile.go

133 lines
3.9 KiB
Go
Raw Normal View History

2020-04-03 21:20:32 +00:00
package templates
import (
"fmt"
2020-04-03 21:20:32 +00:00
"os"
2020-12-29 13:18:13 +00:00
"github.com/goccy/go-yaml"
"github.com/pkg/errors"
"github.com/projectdiscovery/nuclei/v2/pkg/protocols"
2020-12-29 11:03:25 +00:00
"github.com/projectdiscovery/nuclei/v2/pkg/protocols/dns"
"github.com/projectdiscovery/nuclei/v2/pkg/protocols/http"
"github.com/projectdiscovery/nuclei/v2/pkg/workflows"
2020-04-03 21:20:32 +00:00
)
2020-06-29 12:13:08 +00:00
// Parse parses a yaml request template file
func Parse(file string, options *protocols.ExecuterOptions) (*Template, error) {
2020-04-03 21:20:32 +00:00
template := &Template{}
f, err := os.Open(file)
if err != nil {
return nil, err
}
err = yaml.NewDecoder(f).Decode(template)
if err != nil {
return nil, err
}
2020-06-26 12:37:55 +00:00
defer f.Close()
// Setting up variables regarding template metadata
options.TemplateID = template.ID
options.TemplateInfo = template.Info
options.TemplatePath = file
2020-07-31 15:13:51 +00:00
2020-12-29 11:03:25 +00:00
// We don't support both http and dns in a single template
if len(template.RequestsDNS) > 0 && len(template.RequestsHTTP) > 0 {
return nil, fmt.Errorf("both http and dns requests for %s", template.ID)
}
2020-06-29 12:13:08 +00:00
// If no requests, and it is also not a workflow, return error.
2020-12-29 13:18:13 +00:00
if len(template.RequestsDNS)+len(template.RequestsHTTP)+len(template.Workflows) == 0 {
return nil, fmt.Errorf("no requests defined for %s", template.ID)
2020-06-26 12:37:55 +00:00
}
2020-04-03 21:20:32 +00:00
// Compile the workflow request
2020-12-29 13:18:13 +00:00
if len(template.Workflows) > 0 {
if err := template.compileWorkflow(options); err != nil {
return nil, errors.Wrap(err, "could not compile workflow")
}
2020-12-29 14:48:59 +00:00
template.Workflow.Compile(options)
}
// Compile the requests found
for _, request := range template.RequestsDNS {
template.TotalRequests += request.Requests()
}
for _, request := range template.RequestsHTTP {
template.TotalRequests += request.Requests()
2020-04-03 21:20:32 +00:00
}
2020-12-29 11:03:25 +00:00
if len(template.RequestsDNS) > 0 {
template.Executer = dns.NewExecuter(template.RequestsDNS, options)
err = template.Executer.Compile()
2020-12-29 11:03:25 +00:00
}
if len(template.RequestsHTTP) > 0 {
template.Executer = http.NewExecuter(template.RequestsHTTP, options)
err = template.Executer.Compile()
2020-12-29 11:03:25 +00:00
}
if err != nil {
return nil, errors.Wrap(err, "could not compile request")
}
2020-12-29 13:18:13 +00:00
if template.Executer != nil {
if err := template.Executer.Compile(); err != nil {
return nil, errors.Wrap(err, "could not compile template executer")
}
}
2020-04-03 21:20:32 +00:00
return template, nil
}
// compileWorkflow compiles the workflow for execution
func (t *Template) compileWorkflow(options *protocols.ExecuterOptions) error {
for _, workflow := range t.Workflows {
if err := t.parseWorkflow(workflow, options); err != nil {
return err
}
}
return nil
}
// parseWorkflow parses and compiles all templates in a workflow recursively
func (t *Template) parseWorkflow(workflow *workflows.WorkflowTemplate, options *protocols.ExecuterOptions) error {
if err := t.parseWorkflowTemplate(workflow, options); err != nil {
return err
}
for _, subtemplates := range workflow.Subtemplates {
if err := t.parseWorkflow(subtemplates, options); err != nil {
return err
}
}
for _, matcher := range workflow.Matchers {
for _, subtemplates := range matcher.Subtemplates {
if err := t.parseWorkflow(subtemplates, options); err != nil {
return err
}
}
}
return nil
}
// parseWorkflowTemplate parses a workflow template creating an executer
func (t *Template) parseWorkflowTemplate(workflow *workflows.WorkflowTemplate, options *protocols.ExecuterOptions) error {
opts := protocols.ExecuterOptions{
Output: options.Output,
Options: options.Options,
Progress: options.Progress,
Catalogue: options.Catalogue,
RateLimiter: options.RateLimiter,
ProjectFile: options.ProjectFile,
}
paths, err := options.Catalogue.GetTemplatePath(workflow.Template)
if err != nil {
return errors.Wrap(err, "could not get workflow template")
}
if len(paths) != 1 {
return errors.Wrap(err, "invalid number of templates matched")
}
template, err := Parse(paths[0], &opts)
if err != nil {
return errors.Wrap(err, "could not parse workflow template")
}
workflow.Executer = template.Executer
return nil
}