diff --git a/ghostwriter/modules/custom_serializers.py b/ghostwriter/modules/custom_serializers.py index afbb405b5..370009765 100644 --- a/ghostwriter/modules/custom_serializers.py +++ b/ghostwriter/modules/custom_serializers.py @@ -27,6 +27,7 @@ Finding, Report, ReportFindingLink, + ReportObservationLink, ReportTemplate, ) from ghostwriter.rolodex.models import ( @@ -220,6 +221,16 @@ def get_severity_color_hex(self, obj): return obj.severity.color_hex +class ObservationLinkSerializer(TaggitSerializer, CustomModelSerializer): + """Serialize :model:`reporting:ObservationLinkSerializer` entries.""" + + tags = TagListSerializerField() + + class Meta: + model = ReportObservationLink + fields = "__all__" + + class ReportTemplateSerializer(CustomModelSerializer): """Serialize :model:`reporting:ReportTemplate` entries.""" @@ -238,6 +249,7 @@ class ReportSerializer(TaggitSerializer, CustomModelSerializer): total_findings = SerializerMethodField("get_total_findings") findings = FindingLinkSerializer(source="reportfindinglink_set", many=True, exclude=["id", "report"]) + observations = ObservationLinkSerializer(source="reportobservationlink_set", many=True, exclude=["id", "report"]) tags = TagListSerializerField() @@ -734,6 +746,13 @@ class ReportDataSerializer(CustomModelSerializer): "report", ], ) + observations = ObservationLinkSerializer( + source="reportobservationlink_set", + many=True, + exclude=[ + "report", + ], + ) docx_template = ReportTemplateSerializer( exclude=[ "upload_date", diff --git a/ghostwriter/modules/linting_utils.py b/ghostwriter/modules/linting_utils.py index 97dfe8edc..5c46b2191 100644 --- a/ghostwriter/modules/linting_utils.py +++ b/ghostwriter/modules/linting_utils.py @@ -354,6 +354,13 @@ "tags": ["tag1", "tag2", "tag3"], }, ], + "observations": [ + { + "id": 1, + "title": "test observation", + "description": "", + } + ], "docx_template": { "id": 1, "document": "/media/template_oxnfkmX.docx", diff --git a/ghostwriter/modules/reportwriter.py b/ghostwriter/modules/reportwriter.py index f9aaa2971..60616892a 100644 --- a/ghostwriter/modules/reportwriter.py +++ b/ghostwriter/modules/reportwriter.py @@ -1572,10 +1572,11 @@ def render_subdocument(section, finding, p_style=None): return self.sacrificial_doc return None + p_style = self.report_queryset.docx_template.p_style + # Findings for finding in context["findings"]: logger.info("Processing %s", finding["title"]) - p_style = self.report_queryset.docx_template.p_style # Create ``RichText()`` object for a colored severity category finding["severity_rt"] = RichText(finding["severity"], color=finding["severity_color"]) finding["cvss_score_rt"] = RichText(finding["cvss_score"], color=finding["severity_color"]) diff --git a/ghostwriter/reporting/forms.py b/ghostwriter/reporting/forms.py index 8c236c5f0..3c2af30dc 100644 --- a/ghostwriter/reporting/forms.py +++ b/ghostwriter/reporting/forms.py @@ -35,6 +35,7 @@ Observation, Report, ReportFindingLink, + ReportObservationLink, ReportTemplate, Severity, ) @@ -1029,3 +1030,48 @@ def __init__(self, *args, **kwargs): ), ), ) + + +class ReportObservationLinkUpdateForm(forms.ModelForm): + """ + Update an individual :model:`reporting.ReportObservationLink` associated with an + individual :model:`reporting.Report`. + """ + + class Meta: + model = ReportObservationLink + exclude = ( + "report", + "position", + ) + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + for field in self.fields: + self.fields[field].widget.attrs["autocomplete"] = "off" + self.fields["title"].widget.attrs["placeholder"] = "Observation Title" + self.fields["description"].widget.attrs["placeholder"] = "What is this ..." + self.fields["tags"].widget.attrs["placeholder"] = "ATT&CK:T1555, privesc, ..." + + self.helper = FormHelper() + self.helper.form_show_labels = True + self.helper.form_method = "post" + self.helper.form_id = "report-observation-form" + self.helper.layout = Layout( + Row( + Column("title", css_class="form-group col-md-6 mb-0"), + Column("tags", css_class="form-group col-md-6 mb-0"), + css_class="form-row", + ), + Field("description", css_class="enable-evidence-upload"), + ButtonHolder( + Submit("submit_btn", "Submit", css_class="btn btn-primary col-md-4"), + HTML( + """ + + """ + ), + ), + ) diff --git a/ghostwriter/reporting/migrations/0044_reportobservationlink.py b/ghostwriter/reporting/migrations/0044_reportobservationlink.py new file mode 100644 index 000000000..3431e21b2 --- /dev/null +++ b/ghostwriter/reporting/migrations/0044_reportobservationlink.py @@ -0,0 +1,37 @@ +# Generated by Django 3.2.19 on 2023-10-31 17:21 + +from django.conf import settings +import django.core.validators +from django.db import migrations, models +import django.db.models.deletion +import taggit.managers + + +class Migration(migrations.Migration): + + dependencies = [ + ('taggit', '0005_auto_20220424_2025'), + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ('reporting', '0043_observation'), + ] + + operations = [ + migrations.CreateModel( + name='ReportObservationLink', + fields=[ + ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('title', models.CharField(help_text='Enter a title for this observation that will appear in the reports', max_length=255, verbose_name='Title')), + ('position', models.IntegerField(default=1, validators=[django.core.validators.MinValueValidator(1)], verbose_name='Report Position')), + ('description', models.TextField(blank=True, help_text='Provide a description for this observation that introduces it', null=True, verbose_name='Description')), + ('added_as_blank', models.BooleanField(default=False, help_text='Identify an observation that was created for this report instead of copied from the library', verbose_name='Added as Blank')), + ('assigned_to', models.ForeignKey(blank=True, help_text='Assign the task of editing this observation to a specific operator - defaults to the operator that added it to the report', null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)), + ('report', models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to='reporting.report')), + ('tags', taggit.managers.TaggableManager(blank=True, help_text='A comma-separated list of tags.', through='taggit.TaggedItem', to='taggit.Tag', verbose_name='Tags')), + ], + options={ + 'verbose_name': 'Report observation', + 'verbose_name_plural': 'Report observations', + 'ordering': ['report', 'position'], + }, + ), + ] diff --git a/ghostwriter/reporting/models.py b/ghostwriter/reporting/models.py index 6258fdd22..c366588d7 100644 --- a/ghostwriter/reporting/models.py +++ b/ghostwriter/reporting/models.py @@ -687,3 +687,46 @@ def __str__(self): def get_absolute_url(self): return reverse("reporting:observation_detail", args=[str(self.id)]) + + +class ReportObservationLink(models.Model): + + title = models.CharField( + "Title", + max_length=255, + help_text="Enter a title for this observation that will appear in the reports", + ) + position = models.IntegerField( + "Report Position", + default=1, + validators=[MinValueValidator(1)], + ) + description = models.TextField( + "Description", + null=True, + blank=True, + help_text="Provide a description for this observation that introduces it", + ) + added_as_blank = models.BooleanField( + "Added as Blank", + default=False, + help_text="Identify an observation that was created for this report instead of copied from the library", + ) + tags = TaggableManager(blank=True) + # Foreign Keys + report = models.ForeignKey("Report", on_delete=models.CASCADE, null=True) + assigned_to = models.ForeignKey( + settings.AUTH_USER_MODEL, + on_delete=models.SET_NULL, + null=True, + blank=True, + help_text="Assign the task of editing this observation to a specific operator - defaults to the operator that added it to the report", + ) + + class Meta: + ordering = ["report", "position"] + verbose_name = "Report observation" + verbose_name_plural = "Report observations" + + def __str__(self): + return self.title diff --git a/ghostwriter/reporting/templates/reporting/local_observation_edit.html b/ghostwriter/reporting/templates/reporting/local_observation_edit.html new file mode 100644 index 000000000..35451af58 --- /dev/null +++ b/ghostwriter/reporting/templates/reporting/local_observation_edit.html @@ -0,0 +1,124 @@ +{% extends "base_generic.html" %} + +{% load crispy_forms_tags %} + +{% load bleach_tags %} + +{% load custom_tags %} + +{% block pagetitle %}Finding Entry{% endblock %} + +{% block breadcrumbs %} + +{% endblock %} + +{% block content %} +
Ghostwriter supports several template keywords you may utilize to format text and insert various pieces of + information. Begin typing @ to open the autocomplete dialog for keywords.
+Keyword | +Usage | +
---|---|
{{.client}} | + {% endverbatim %} ++ {% if reportobservationlink.report.project.client.short_name %} + This keyword will be replaced with the client's short name, " + {{ reportobservationlink.report.project.client.short_name }}." + {% else %} + No short name is set for this client, so this keyword will be replaced by the full name, " + {{ reportobservationlink.report.project.client.name }}." + {% endif %} + | +
{{.project_type}} | + {% endverbatim %} +This keyword will be replaced with the project type in + lowercase, {{ reportobservationlink.report.project.project_type|lower }}. + | +
{{.project_start}} | + {% endverbatim %} +This keyword will be replaced with the project's start date in + {% settings_value "DATE_FORMAT" %} + format: {{ reportobservationlink.report.project.start_date|date:"DATE_FORMAT" }} | +
{{.project_end}} | + {% endverbatim %} +This keyword will be replaced with the project's end date in + {% settings_value "DATE_FORMAT" %} + format: {{ reportobservationlink.report.project.end_date|date:"DATE_FORMAT" }} | +
{{.caption}} | +Start a line of text with this keyword to make it a caption. This is + intended to follow a code block. + | +
For additional formatting, utilize the WYSIWIG HTML formatting to apply bold, italic, code, inline code, and other text styles.
+These styles will carry over to the Word and PowerPoint reports. See the documentaiton for more + details.
+Use this form to edit "{{ reportobservationlink.title }}" just for this report.
+When the finding is complete and ready for review, make sure you check the box down below.
+ + + {% if form.errors %} + + {% endif %} + + + {% crispy form form.helper %} +{% endblock %} + +{% block morescripts %} + {% comment %} Include the reusable delete confirmation modal and related scripts {% endcomment %} + {% include "confirm_delete_modal.html" %} +{% endblock %} diff --git a/ghostwriter/reporting/templates/reporting/observation_list.html b/ghostwriter/reporting/templates/reporting/observation_list.html index 31f12a389..39227d965 100644 --- a/ghostwriter/reporting/templates/reporting/observation_list.html +++ b/ghostwriter/reporting/templates/reporting/observation_list.html @@ -27,12 +27,38 @@No findings have been added to this report yet.
{% endif %} + +Attach observations by searching and then clicking the button next to your desired + observation.
+ + + + + + + {% if report.reportobservationlink_set.all %} ++ | Observation | +Options | +
---|---|---|
+ | + {{ observation.title }} + + | +
+
+
+
+
+ |
+
No observations have been added to this report yet.
+ {% endif %} +