Skip to content

A free cryptowallet risk scoring tool with fully explainable scoring.

License

Notifications You must be signed in to change notification settings

januusio/cryptowallet_risk_scoring

Repository files navigation

** Free endpoint currently paused for maintenance **

Cryptowallet Risk Scoring by Januus

A free cryptowallet risk assessment tool, it helps assess if a cryptowallet has transacted with a threat actor, if it belongs to a scammer, or if the wallet address is generally safe. It generates a report on the fraudulent activity of any Ethereum, Bitcoin, or Tron address based on the corresponding identity data and transaction analysis and displays the audit trail for this report in clean JSON.

Google Colab Demonstration

Here is a link to this program running inside of Google Colab. It runs in any web browser, on Google's servers, for free. Or, simply follow along at the terminal.

(Colab written by third party)

Setup

git clone https://github.com/januusio/cryptowallet_risk_scoring
cd cryptowallet_risk_scoring
python3 setup.py install
pip3 install -r requirements.txt
python3 example_default.py

jq (a lightweight CLI JSON processor) is used to show responses.

Data

The dataset spans sanctions, terrorism, petty scammers, crypto heists, malicious smart contract creators, nft thieves, ransomware creators, darknet traders across both OSINT and private datasets. It should be considered the union of existing wallet reporting service data, taggers and scam detectors - in addition to a large dataset unique to Januus. Transaction analysis is performed by default to assess risk by degrees of separation in the output.

Usage

The example files example_default.py, example_json.py, example_reasons.py easily articulate the usage options.

Examples

Here is an address that is a honeypot scammer:

python3 -m januus_riskreport '{"eth_addresses":["0xbb0ea877a85df253ccc312b80c644da31443abfd"]}' | jq .

image

Here is a terrorist who has also sent $40K to two different hackers:

python3 -m januus_riskreport '{"eth_addresses":["0xebfe7a29ea17acb5f6f437e659bd2d472deedc54"]}' | jq .

image

image

Notice that by sending to scammers, the lending_risk is mostly increased. If the terrorist would have received funds from a hacker, the fraudRisk would be the main increase. image

Using the client

The request to Januus' risk scoring service is a single function:

from januus_riskreport.client import riskreport_on_entity

report = riskreport_on_entity( eth_addresses = [...], btc_addresses = [...] )

eth_addresses and btc_addresses are lists of strings. Note, this is a report on a single entity, not a batch request. If an entity has multiple addresses, all may be queried at the same time by passing lists of length > 1. The query must not be empty.

Risk Scoring Package Maintenance

We're committed to supporting this package indefinitely. Fresh risk data, newly identified threat actors and fraudulent wallets are added to the database daily.

Type of Risk Data?

Risk is not one-dimensional. A terrorist organization may pay their bills on time and be very credit worthy; but that doesn’t mean that they should be engaged with. A person who tends to naively fall for phishing scams isn’t necessarily likely to defraud others; but he probably shouldn’t be lent to. This API scores entities along 3 dimensions of risk:

Cryptowallet Reputation Risk

  • What risk does this entity pose to my reputation? Would it cause bad PR to enter a contractual relationship with them?
  • Where to find it:
    • For the overall reputation risk: report.risk_scores.reputation_risk
    • For each reason in report.reasons: reason.offsets.reputation_risk_offset

Cryptowallet Fraud Risk

  • Has this person defrauded others before? Is there a long history of online presence to indicate it’s not a fake (or scam) account? Have they been funded by known scammers?
  • Where to find it:
    • For the overall reputation risk: report.risk_scores.fraud_risk
    • For each reason in report.reasons: reason.offsets.fraud_risk_offset

Cryptowallet Lending Risk

  • Does this entity have concrete signs of credit-worthiness? Do they appear financially responsible? Do they have a record of falling victim to scams in the past?
  • Where to find it:
    • For the overall reputation risk: report.risk_scores.lending_risk
    • For each reason in report.reasons: reason.offsets.lending_risk_offset

Retriving a Risk Score Value Only

  • Use report.risk_scores.combined_risk.

All of the above dimensions are aggregated to produce a combined risk score. This is probably the most important single score. The only guarantee it makes as to certain failure is the following: If the report.risk_scores.combined_risk is failing (>= 60), then one or more of the above dimensions must also have had a failing score. Each reason also has a combined_risk_offset. In the rare circumstance that an entity has multiple reasons for certain failure, the combined risk offsets may not be evenly distributed.

Core Concepts

Januus' Risk Scale

The risk score that’s returned does not represent a probability; it’s a carefully calculated grade with nuanced ranges of meaning. It’s bounded between 0 and 100; where 0 is impossibly low risk (zero) and 100 is impossibly high risk. The neutral score is 30 and 60 is failing. You can think of it like an A+/A-/B+/B/B-/C+/C/C-/D+/D/D-/F grade from primary school (except from-low-to-high, because low risk equates to a good grade.)

Here’s a brief summary:

Score Description
> 25 Good
30 Neutral
> 40 A troublingly high score
>= 60 A failing score

A more detailed breakdown:

Score Description
0 Impossibly low risk (zero)
<10 Very few entities will have a score this outstanding, almost risk-free based on seen datapoints.
10-20 An extremely good score.
20-25 A good score. In this range, the improved score is still very meaningful.
25-27 Starting to become only slightly better than neutral. Usually in this range, there might not be much data available about an entity other than date verification.
27-33 Neutral.
33-35 Slightly worse than neutral. This is the realm where a risk factor starts to become more than just noise.
35-40 A slightly worrying score. Perhaps this wallet has transacted with bad actors, but at a low enough proportion that a meaningful link cannot be established.
40-50 A bad score.
50-60 A very bad score. Because it becomes exponentially more difficult to reach 60 without a reason for certain failure, this range probably signifies the presence of several unrelated risk factors. Especially as the score inches toward toward and past 57ish, then it becomes nearly as meaningful as a failing score in the low 60’s.
==60 A failing score; do not have dealings with this entity. This is the exact point where risk becomes so extreme that an irredeemably failing score is given. Only a hand-full of concrete reasons in our rule system will allow the score to hit this threshold. Once at 60, no number of safe datapoints can ever push the score below this mark. It’s guaranteed.
70-80 This score signifies an entity that one should divert considerable resources to avoid. Although, it’s possible that it was even higher, but some datapoints (or ‘reasons’) had a negative offset, pushing it closer toward 60.
>80 A supremely high risk. In this range, usually the wallet has been directly and credibly marked as malicious or has an overwhelming number of direct neighbors of whom this is true.
==100 An impossibly high risk score.

What makes the risk score explainable?

In the response, we provide an array of reasons ( in report.reasons ) that explain our score for a given entity. Each reason given provides an offset for a very approximate estimate of how much it impacted the overall score. A risk factor will have a positive offset and a datapoint that gives credibility will have a negative offset.

The Report

The generated report will be an instance of the RiskReport class:

class RiskReport:
    risk_scores: RiskScores
    reasons: List[RiskReason]

The RiskScores Object

The RiskScores object contains the three fundamental scores of risk, as well as a combined score. These are all explained above in the 'Concepts' section.

class RiskScores:
    combined_risk: float
    fraud_risk: float
    reputation_risk: float
    lending_risk: float

White-box Reasoning

What about the reasons: List[RiskReason] of our risk report? Each Reason is a manifestation of a single component that contributed to the score.

class RiskReason:

    # An English explanation of the reason
    explanation: str
    
    # A RiskElaboration gives machine-readable details about this Reason, or component of the score
    risk_elaboration: RiskElaboration
    
    # This is json data giving details of the reason.
    # There are a finite number of labels, and each label has it's own corresponding RiskElaboration
    label: str
    
    # A RiskOffsets object is an approximation of how much this Reason contributed to each dimension of the top-level RiskScore
    offsets: RiskOffsets 

class RiskOffsets:
    combined_risk_offset: float
    fraud_risk_offset: float
    lending_risk_offset: float
    reputation_risk_offset: float
    

The ideas of RiskOffsets are, again, covered above in the 'Concepts' section.

Labels & subclasses of RiskElaboration

Each reason will have a label. For each reason, the subclass of reason.risk_elaboration can be known by its label. If you would like a new label added, please contact us and we’d love to look into its feasibility and usefulness. Here are the current labels:

"sent-to-bad-actor"
"funded-by-bad-actor"
"bad-zero-valued-txs"
"is-bad-actor"
"date-verification"

And here are the corresponding classes

class RiskElaboration:
  pass
  
class IsBadActor(RiskElaboration):
    """
    The queried entity is a known scammer. 
    The layout of `RiskAccountDetails` will be publicly expanded in the future, but for now it's JSON. 
    """
    risk_details: List[RiskAccountDetails]

class BadNeighborDetails:
    sender: str
    recipient: str
    total_usd: float
    risk_details: RiskAccountDetails

class SentToBadActor(RiskElaboration):
    """
    The entity sent money to a scammer.
    The 'recipient' is a wallet that has received funds from the queried entity.
    """
    how_many_recipients: int
    how_many_bad_recipients: int
    bad_recipient_details: List[ScammyNeighborDetails]

class FundedByBadActor(RiskElaboration):
    """
    The entity has received funds from a scammer.
    A wallet that has sent money to the queried entity is a 'funder'
    """
    how_many_funders: int
    how_many_bad_funders: int
    bad_funder_details: List[ScammyNeighborDetails]


class BadZeroValuedTxs(RiskElaboration):
    "This class shows that the queried entity has had one or more zero-valued transactions with a scammer."
    how_many_neighbors: int
    how_many_bad_neighbors: int
    bad_neighbor_details: List[Dict[str,str]]

Date verification is also important, as newly created entities tend to not have as much known about them or credit history.

class VerifiedDates(RiskElaboration):
    verified_dates: List[VerifiedDate]
    
class VerifiedDate:
    date: str
    source: str
    weight: float

Rate Limits

We ask that you generally not generate more than 30 queries per minute.

Performance / Latency

95% of addresses will be queried in roughly 1 second or under. Latency of up to 30 seconds is uncommon.

Full Example Output

python3 -m januus_riskreport '{"eth_addresses":["0x08d32b0da63e2C3bcF8019c9c5d849d7a9d791e6"]}' | jq . 
{
  "reasons": [
    {
      "explanation": "The entity has been associated with multiple dates between 2017-02-14 and 2023-09-22",
      "label": "date-verification",
      "offsets": {
        "combinedRiskOffset": -2.5368404,
        "fraudRiskOffset": -2.8950028,
        "lendingRiskOffset": 0,
        "reputationRiskOffset": -1.7997159
      },
      "riskElaboration": {
        "verifiedDates": [
          {
            "date": "2017-02-14",
            "source": "The day '0x08d32b0da63e2c3bcf8019c9c5d849d7a9d791e6' was first seen on the blockchain.",
            "weight": 0.7461
          },
          {
            "date": "2023-09-22",
            "source": "The day '0x08d32b0da63e2c3bcf8019c9c5d849d7a9d791e6' was last seen on the blockchain.",
            "weight": 0.0228
          }
        ]
      }
    },
    {
      "explanation": "The entity has sent a total of $117.73446 to 1 risk vector(s). This could mean that they are a victim of fraud, or have knowingly funded a threat actor.",
      "label": "sent-to-bad-actor",
      "offsets": {
        "combinedRiskOffset": 0.005155486,
        "fraudRiskOffset": 0.0029025616,
        "lendingRiskOffset": 0.005628586,
        "reputationRiskOffset": 0.002870712
      },
      "riskElaboration": {
        "badRecipientDetails": [
          {
            "recipient": "0xe63b94071ddfef2acea8a1048a158b573b605f20",
            "riskDetails": {
              "address": "0xe63b94071ddfef2acea8a1048a158b573b605f20",
              "riskFactors": [
                {
                  "actorType": "scammer",
                  "involvedRiskActivity": {
                    "categoryName": "phishing",
                    "description": "Tricks the victim into paying a certain amount of money to the actor.",
                    "rating": "moderate",
                    "subCategory": "scam"
                  },
                  "mediaPlatformUsed": "Unknown"
                }
              ]
            },
            "sender": "0x08d32b0da63e2c3bcf8019c9c5d849d7a9d791e6",
            "totalUsd": 117.73445892333984
          }
        ],
        "howManyBadRecipients": "1",
        "howManyRecipients": "3246",
        "totalUsd": 117.73445892333984
      }
    },
    {
      "explanation": "The entity has received a total of $128.235 from 1 scammer(s) or threat actor(s).",
      "label": "funded-by-bad-actor",
      "offsets": {
        "combinedRiskOffset": 0.0022822178,
        "fraudRiskOffset": 0.002160931,
        "lendingRiskOffset": 0,
        "reputationRiskOffset": 0.0021372193
      },
      "riskElaboration": {
        "badFunderDetails": [
          {
            "recipient": "0x08d32b0da63e2c3bcf8019c9c5d849d7a9d791e6",
            "riskDetails": {
              "address": "0xe63b94071ddfef2acea8a1048a158b573b605f20",
              "riskFactors": [
                {
                  "actorType": "scammer",
                  "involvedRiskActivity": {
                    "categoryName": "phishing",
                    "description": "Tricks the victim into paying a certain amount of money to the actor.",
                    "rating": "moderate",
                    "subCategory": "scam"
                  },
                  "mediaPlatformUsed": "Unknown"
                }
              ]
            },
            "sender": "0xe63b94071ddfef2acea8a1048a158b573b605f20",
            "totalUsd": 128.23500061035156
          }
        ],
        "howManyBadFunders": "1",
        "howManyFunders": "3762"
      }
    },
    {
      "explanation": "The entity has had blockchain transactions with a known threat. However, the total value sent/received was $0 so it may be part of a automatically executed contract. Or, if it's BTC, it could imply they belong to the same wallet.",
      "label": "bad-zero-valued-txs",
      "offsets": {
        "combinedRiskOffset": 0.0026444746,
        "fraudRiskOffset": 0.0015002476,
        "lendingRiskOffset": 0,
        "reputationRiskOffset": 0.0029676585
      },
      "riskElaboration": {
        "badNeighborDetails": [
          {
            "neighbor": "0xa6a8dc7904911b23347989eac761e8322db67018",
            "riskDetails": {
              "address": "0xa6a8dc7904911b23347989eac761e8322db67018",
              "riskFactors": [
                {
                  "actorType": "scammer",
                  "involvedRiskActivity": {
                    "categoryName": "phishing",
                    "description": "Tricks the victim into paying a certain amount of money to the actor.",
                    "rating": "moderate",
                    "subCategory": "scam"
                  },
                  "mediaPlatformUsed": "Unknown"
                }
              ]
            }
          },
          {
            "neighbor": "0x9007a0421145b06a0345d55a8c0f0327f62a2224",
            "riskDetails": {
              "address": "0x9007a0421145b06a0345d55a8c0f0327f62a2224",
              "riskFactors": [
                {
                  "actorType": "hacker",
                  "involvedRiskActivity": {
                    "categoryName": "hacking",
                    "description": "Utilizing vulnerabilities in systems to gain unauthorized access, often with the intent of compromising data or executing malicious actions.",
                    "rating": "high",
                    "subCategory": "exploit"
                  },
                  "mediaPlatformUsed": "Unknown"
                }
              ]
            }
          }
        ],
        "howManyBadNeighbors": "2",
        "howManyNeighbors": "4515"
      }
    }
  ],
  "riskScores": {
    "combinedRisk": 27.473242,
    "fraudRisk": 27.11156,
    "lendingRisk": 30.005629,
    "reputationRisk": 28.20826
  }
}