Skip to content

Commit

Permalink
Merge pull request #46 from passivetotal/asi-docs
Browse files Browse the repository at this point in the history
ASI and vuln doc updates and bug fixes, plus whois history
  • Loading branch information
aeetos authored Mar 11, 2022
2 parents 8b95fe4 + 0dc72e4 commit cece042
Show file tree
Hide file tree
Showing 16 changed files with 524 additions and 57 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -14,3 +14,4 @@ libpassivetotal.egg-info/
pip-wheel-metadata
docs/_build
*.vscode
!.readthedocs.yaml
11 changes: 11 additions & 0 deletions .readthedocs.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@

version: 2

sphinx:
configuration: docs/conf.py

python:
version: 3.7
install:
- requirements: requirements.txt
- requirements: docs/requirements.txt
35 changes: 35 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,40 @@
# Changelog

## v2.5.9

#### Enhancements

- Significant improvements to the Attack Surface Intelligence (ASI) documentation. Added
class references for ASI, CTI and vulnerability intelligence to ensure the docs and links
generated properly. Introduced a new Sphinx module to help generate inline table-of-contents
for complex classes. Corrected typos in docstrings and ensured consistent type references
when methods returned RecordList-type objects.
- Implemented new config files for readthedocs to align with current documentation practices.
- New `whois_history` property of `Hostname` and `IPAddress` entities gives direct access
to historical Whois (ownership) records. Includes more consistent implementation of
RecordList functionality and better pandas dataframe support for both historical Whois and
field-level Whois searches.
- New `impacted_attack_surfaces` property of vulnerability articles (`VulnArticle`) filters
the list of third-party vendors to only those with at least one observation. The Illuminate
API returns all attack surfaces associated with an API key regardless of whether they are
impacted; the complete list is still available in the `attack_surfaces` property. Also updated
the `info` view of the Pandas dataframe on a vulnerability article so the `impacts` column
shows the count of impacted attack surfaces.


#### Bug Fixes

- Correctly sum insight and observation counts when accessing Attack Surface Insights
(ASIs) across multiple severity levels. Previously the `active_insight_count`,
`total_insight_count`, and `total_observations` properties of the `all_active_insights`
record list were only counting high-priority insights.
- Fixed issue that caused an exception when trying to generate a dictionary view of an
AttackSurfaceComponent (detection).
- Removed reference to non-existant field in `VulnArticle` that was causing an exception when
rendering a vulnerability article as a dictionary with the `as_dict` property.
- Handle vuln articles with no impacted assets without raising an exception.



## v2.5.8

Expand Down
117 changes: 114 additions & 3 deletions docs/analyzer.rst
Original file line number Diff line number Diff line change
Expand Up @@ -72,8 +72,8 @@ IP Analysis
:inherited-members:
Module Reference
----------------
Analyzer Module Reference
-------------------------
.. automodule:: passivetotal.analyzer
:members:
Expand All @@ -98,6 +98,8 @@ directly inform security research and may guide subsequent searches.
Whois Records
-------------
Domain Name Whois
^^^^^^^^^^^^^^^^^
The `whois` property for host names returns the DomainWhois record for
the registered domain name portion of the host name.
Expand All @@ -113,7 +115,50 @@ the API tries to normalize and parse the data into fields, your code should alwa
prepared for missing or malformed data. Access the `raw` record for the API response
directly as a Python dict or use the `record` property to get the raw Whois response.
The RiskIQ PassiveTotal API can search Whois records by field to find related
IP Whois Records
^^^^^^^^^^^^^^^^
The `whois` property is also available for IP addresses. Some of the fields are different
than domain whois records, but most are the same.
.. code-block:: python
>>> from passivetotal import analyzer
>>> analyzer.init()
>>> print(analyzer.IPAddress('160.69.1.37').whois.organization)
PACCAR, Inc.
Using Whois Fields
^^^^^^^^^^^^^^^^^^
Fields in the IP and Hostname Whois record are returned as `WhoisField` objects to faciliate
field-level searching. Cast the field as a string if you need to get the actual value:
.. code-block:: python
>>> org = analyzer.IPAddress('160.69.1.37').whois.organization
>>> org
WhoisField('organization','PACCAR, Inc.')
>>> print(org)
Paccar, Inc.
>>> org_str = str(org)
>>> org_str
Paccar, Inc.
You can also start with a field name and a string and search for it directly to find domains
or IPs associated with a name, email address, or other supported field type.
.. code-block:: python
>>> from passivetotal.analyzer.whois import WhoisField
>>> WhoisField('email','domains@riskiq.com').records.domains
Search Whois Records
^^^^^^^^^^^^^^^^^^^^
The RiskIQ PassiveTotal API can search Whois records by field to find related
domain names with the same contact information. Use the `records` property of
supported fields (any property that returns type `WhoisField`).
Expand All @@ -123,11 +168,65 @@ supported fields (any property that returns type `WhoisField`).
>>> analyzer.Hostname('riskiq.net').whois.organization.records.domains
{Hostname('riskiq.com'), Hostname('riskiq.net'), Hostname('riskiqeg.com')}
The `records` property returns the same type of list-like object that other analyzer
objects return, so you can filter, sort, and convert to pandas DataFrames as needed.
See below for reference.
Historical Whois Records
^^^^^^^^^^^^^^^^^^^^^^^^
Historical whois records may be available for domain names and IP addresses. Access the
`whois_history` property of an `IPAddress` or `Hostname` object to obtain a list of historically
observed records as a standard analyzer `RecordList`.
.. code-block:: python
>>> from passivetotal import analyzer
>>> for record in analyzer.Hostname('passivetotal.org').whois_history:
print(record.last_seen, record.registrant_email)
2022-02-11 19:16:29.334000-08:00 passivetotal.org-registrant@anonymised.email
2021-04-05 09:13:10.330000-07:00 passivetotal.org-registrant@anonymised.email
2021-04-04 08:40:41.295000-07:00 passivetotal.org-registrant@anonymised.email
2021-04-03 16:25:42.455000-07:00 passivetotal.org-registrant@anonymised.email
2021-04-06 08:42:07.273000-07:00 passivetotal.org-registrant@anonymised.email
2021-03-04 15:45:41.026000-08:00 passivetotal.org-Registrant@anonymised.email
2021-03-03 13:56:05.605000-08:00 passivetotal.org-registrant@anonymised.email
2020-10-06 02:41:38.359000-07:00 passivetotal.org-registrant@anonymised.email
2020-10-06 14:01:33.575000-07:00 passivetotal.org-registrant@anonymised.email
2020-08-21 04:20:43.757000-07:00 passivetotal.org-Registrant@anonymised.email
2020-06-20 03:57:01.411000-07:00 abuse@comlaude.com
2020-03-16 15:22:46.043000-07:00 abuse@comlaude.com
2020-03-17 06:57:02.899000-07:00 abuse@comlaude.com
2019-09-29 07:45:02.096000-07:00 domains@riskiq.com
In this example, we accessed only two fields of each record, but the complete Whois record
remains available. Consider using the `as_dict` or `as_df` properties of the `whois_history`
object to get the complete list as a Python dictionary or a Pandas dataframe.
.. code-block:: python
>>> analyzer.Hostname('passivetotal.org').whois_history.as_dict
{'records': [{'admin': {'email': 'passivetotal.org-admin@anonymised.email'},
'billing': {},
'registrant': {'country': 'US',
'email': 'passivetotal.org-registrant@anonymised.email',
...
Whois Object Reference
^^^^^^^^^^^^^^^^^^^^^^
.. autoclass:: passivetotal.analyzer.whois.DomainWhois
:members:
:inherited-members:
.. autoclass:: passivetotal.analyzer.whois.IPWhois
:members:
:inherited-members:
.. autoclass:: passivetotal.analyzer.whois.WhoisField
:members:
:inherited-members:
Expand All @@ -136,6 +235,18 @@ supported fields (any property that returns type `WhoisField`).
:members:
:inherited-members:
.. autoclass:: passivetotal.analyzer.whois.HistoricalWhoisRecords
:members:
:inherited-members:
.. autoclass:: passivetotal.analyzer.whois.HistoricalDomainWhois
:members:
:inherited-members:
.. autoclass:: passivetotal.analyzer.whois.HistoricalIPWhois
:members:
:inherited-members:
Threat Intel Articles
Expand Down
11 changes: 10 additions & 1 deletion docs/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,10 +31,17 @@
# ones.
extensions = [
'sphinx.ext.autodoc',
'sphinx.ext.autosummary',
'sphinx.ext.intersphinx',
'sphinx.ext.todo',
'sphinx.ext.coverage',
'sphinx.ext.ifconfig',
'autoclasstoc',
]

autoclasstoc_sections = [
'public-attrs',
'public-methods',
]

# Add any paths that contain templates here, relative to this directory.
Expand Down Expand Up @@ -294,7 +301,9 @@
import os
on_rtd = os.environ.get('READTHEDOCS', None) == 'True'

if not on_rtd: # only import and set the theme if we're building docs locally
if on_rtd:
html_theme = 'default'
else:
import sphinx_rtd_theme
html_theme = 'sphinx_rtd_theme'
html_theme_path = [sphinx_rtd_theme.get_html_theme_path()]
Expand Down
Loading

0 comments on commit cece042

Please sign in to comment.