[
    {
        "description": "This metric employs an Input Crossover, combining the perturbed input from proxy 1 and the gradient standard deviation calculation from proxy 4.",
        "code": "import torch\n\ndef heuristic_2(model, inputs, targets):\n    # Generate small random noise\n    noise = torch.randn_like(inputs) * 0.01\n    perturbed_inputs = inputs + noise\n    perturbed_inputs.requires_grad_(True)\n\n    # Compute the perturbed output\n    output = model(perturbed_inputs)\n\n    gradients = torch.autograd.grad(output, perturbed_inputs, grad_outputs=torch.ones_like(output), create_graph=True, retain_graph=True)[0]\n\n    if gradients is None:\n        return 0.0\n\n    std_dev_grad = torch.std(gradients).item()\n\n    return std_dev_grad",
        "score": 0.8289392226981206
    },
    {
        "description": "This metric examines the sensitivity of the output to small perturbations in the input. It's based on the idea that a robust network should exhibit some level of invariance to noise. We add small random noise to the input images and measure the change in the output logits. A smaller change in the output indicates greater robustness. The score is calculated as the negative of the average L2 norm of the difference between the original output and the perturbed output across the batch.",
        "code": "import torch\n\ndef heuristic_5(model, inputs, targets):\n    # Generate small random noise\n    noise = torch.randn_like(inputs) * 0.01\n\n    # Perturb the input\n    perturbed_inputs = inputs + noise\n\n    # Compute the original and perturbed outputs\n    with torch.no_grad():\n        original_output = model(inputs)\n        perturbed_output = model(perturbed_inputs)\n\n    # Calculate the L2 norm of the difference between the outputs\n    output_diff = original_output - perturbed_output\n    sensitivity = torch.norm(output_diff, p=2, dim=1).mean()\n\n    # Return the negative sensitivity as the score (lower is better)\n    return (-sensitivity).cpu().item()",
        "score": 0.7816322611349211
    },
    {
        "description": "This metric calculates the sum of max value of weights and gradients of the model.",
        "code": "import torch\nimport torch.nn as nn\n\ndef heuristic_10(model, inputs, targets):\n    max_abs_weight = 0.0\n    max_abs_grad = 0.0\n    num_conv_layers = 0\n\n    for module in model.modules():\n        if isinstance(module, nn.Conv2d):\n            max_abs_weight += torch.max(torch.abs(module.weight)).item()\n            num_conv_layers += 1\n\n    inputs.requires_grad_(True)\n    output = model(inputs)\n\n    gradients = torch.autograd.grad(output, inputs, grad_outputs=torch.ones_like(output), create_graph=True, retain_graph=True)[0]\n\n    if gradients is not None:\n        max_abs_grad = torch.max(torch.abs(gradients)).item()\n\n    return max_abs_weight + max_abs_grad",
        "score": 0.7754108865123127
    },
    {
        "description": "This metric computes the product of the maximum absolute values of weights and the variance of the gradient. This combines information about the weights and gradients to provide a different view of the model's behavior.",
        "code": "import torch\nimport torch.nn as nn\n\ndef heuristic_9(model, inputs, targets):\n    max_abs_weight = 0.0\n    for module in model.modules():\n        if isinstance(module, nn.Conv2d):\n            max_abs_weight = max(max_abs_weight, torch.max(torch.abs(module.weight)).item())\n\n    inputs.requires_grad_(True)\n    output = model(inputs)\n\n    gradients = torch.autograd.grad(output, inputs, grad_outputs=torch.ones_like(output), create_graph=True, retain_graph=True)[0]\n\n    if gradients is None:\n        return 0.0\n    else:\n        grad_variance = torch.var(gradients).item()\n        return max_abs_weight * grad_variance",
        "score": 0.6886953558188454
    },
    {
        "description": "This metric uses Operation Crossover, combining the concept of output variance (used in proxy 3 & 4) with the absolute mean gradient (used in proxy 5). It calculates the absolute mean gradient of the output, then divides it by the variance of the original output.",
        "code": "import torch\n\ndef heuristic_8(model, inputs, targets):\n    inputs.requires_grad_(True)\n    output = model(inputs)\n    output.mean().backward()\n    gradients = inputs.grad\n    mean_abs_grad = torch.abs(gradients).mean().item()\n\n    output_variance = torch.var(output).item()\n\n    if output_variance == 0:\n        return 0.0\n    else:\n        return mean_abs_grad / output_variance",
        "score": 0.6864754104616312
    }
]