diff --git a/README.md b/README.md
index 820d85d9a..d62100895 100644
--- a/README.md
+++ b/README.md
@@ -110,6 +110,7 @@ Commands:
public-suffix-list
quote
refang
+ defang
rot
sha1
sha256
@@ -508,6 +509,13 @@ $ ronin refang hxxps://www[.]evil[.]com/foo/bar/baz
https://www.evil.com/foo/bar/baz
```
+De-fangs a URL:
+
+```shell
+$ ronin defang https://www.evil.com/foo/bar/baz
+hxxps://www[.]evil[.]com/foo/bar/baz
+```
+
Query the ASN of an IP address:
```shell
diff --git a/gemspec.yml b/gemspec.yml
index 191ff9002..c0ab1ea58 100644
--- a/gemspec.yml
+++ b/gemspec.yml
@@ -79,6 +79,7 @@ generated_files:
- man/ronin-public-suffix-list.1
- man/ronin-quote.1
- man/ronin-refang.1
+ - man/ronin-defang.1
- man/ronin-rot.1
- man/ronin-sha1.1
- man/ronin-sha256.1
diff --git a/lib/ronin/cli/commands/defang.rb b/lib/ronin/cli/commands/defang.rb
new file mode 100644
index 000000000..bc3271230
--- /dev/null
+++ b/lib/ronin/cli/commands/defang.rb
@@ -0,0 +1,84 @@
+# frozen_string_literal: true
+#
+# Copyright (c) 2006-2023 Hal Brodigan (postmodern.mod3 at gmail.com)
+#
+# Ronin is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ronin is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ronin. If not, see .
+#
+
+require_relative '../value_processor_command'
+
+require 'ronin/support/network/defang'
+
+module Ronin
+ class CLI
+ module Commands
+ #
+ # De-fangs a URL, hostname, or IP address.
+ #
+ # ## Usage
+ #
+ # ronin defang [options] [{URL | HOST | IP} ...]
+ #
+ # ## Options
+ #
+ # -f, --file FILE Optional file to read values from
+ # -h, --help Print help information
+ #
+ # ## Arguments
+ #
+ # [URL | HOST | IP ...] A URL, hostname, or IP address
+ #
+ # ## Examples
+ #
+ # ronin defang https://www.evil.com/foo/bar/baz
+ # ronin defang www.example.com
+ # ronin defang 192.168.1.1
+ # ronin defang --file urls.txt
+ #
+ # @since 2.2.0
+ #
+ class Defang < ValueProcessorCommand
+
+ usage '[options] [{URL | HOST | IP} ...]'
+
+ argument :value, required: false,
+ repeats: true,
+ usage: 'URL | HOST | IP',
+ desc: 'A refanged URL, hostname, or IP address'
+
+ examples [
+ 'https://www.evil.com/foo/bar/baz',
+ 'www.example.com',
+ '192.168.1.1',
+ '--file urls.txt'
+ ]
+
+ description 'Defangs a URLs, hostnames, or IP addresses'
+
+ man_page 'ronin-defang.1'
+
+ #
+ # Defangs a URL, hostname, or IP address.
+ #
+ # @param [String] value
+ # The value to defang.
+ #
+ def process_value(value)
+ puts Support::Network::Defang.defang(value)
+ end
+
+ end
+ end
+ end
+end
diff --git a/man/ronin-defang.1.md b/man/ronin-defang.1.md
new file mode 100644
index 000000000..f9370e6d6
--- /dev/null
+++ b/man/ronin-defang.1.md
@@ -0,0 +1,59 @@
+# ronin-defang 1 "2025-01-01" Ronin "User Manuals"
+
+## NAME
+
+ronin-defang - Defangs a URLs, hostnames, or IP addresses
+
+## SYNOPSIS
+
+`ronin defang` [*options*] [{*URL* \| *HOST* \| *IP*} ...]
+
+## DESCRIPTION
+
+De-fangs URL(s), hostname(s), or IP address(es).
+
+## ARGUMENTS
+
+*URL*
+: A URL argument to defang
+ (ex: `https://www.evil.com/foo/bar/baz`).
+
+*HOST*
+: A hostname argument to defang (ex: `www.example.com`).
+
+*IP*
+: A IP address argument to defang (ex: `192.168.1.1`).
+
+## OPTIONS
+
+`-f`, `--file` *FILE*
+: The optional file to read values from.
+
+`-h`, `--help`
+: Print help information.
+
+## EXAMPLES
+
+De-fangs a URL:
+
+ ronin defang https://www.evil.com/foo/bar/baz
+
+De-fangs a hostname:
+
+ ronin defang www.example.com
+
+De-fangs a IP address:
+
+ ronin defang 192.168.1.1
+
+De-fangs a file of URLs, hostnames, or IP addresses:
+
+ ronin defang --file urls.txt
+
+## AUTHOR
+
+Postmodern
+
+## SEE ALSO
+
+[ronin-refang](ronin-refang.1.md)
diff --git a/spec/cli/commands/defang_spec.rb b/spec/cli/commands/defang_spec.rb
new file mode 100644
index 000000000..54e5a2b14
--- /dev/null
+++ b/spec/cli/commands/defang_spec.rb
@@ -0,0 +1,42 @@
+require 'spec_helper'
+require 'ronin/cli/commands/defang'
+require_relative 'man_page_example'
+
+describe Ronin::CLI::Commands::Defang do
+ include_examples "man_page"
+
+ describe "#process_value" do
+ context "when given a refanged URL value" do
+ let(:refanged) { 'https://www.evil.com/foo/bar/baz' }
+ let(:defanged) { 'hxxps[://]www[.]evil[.]com/foo/bar/baz' }
+
+ it "must print the de-fanged URL" do
+ expect {
+ subject.process_value(refanged)
+ }.to output("#{defanged}#{$/}").to_stdout
+ end
+ end
+
+ context "when given a refanged hostname value" do
+ let(:refanged) { 'www.example.com' }
+ let(:defanged) { 'www[.]example[.]com' }
+
+ it "must print the de-fanged hostname" do
+ expect {
+ subject.process_value(refanged)
+ }.to output("#{defanged}#{$/}").to_stdout
+ end
+ end
+
+ context "when given a refanged IP address value" do
+ let(:refanged) { '192.168.1.1' }
+ let(:defanged) { '192[.]168[.]1[.]1' }
+
+ it "must print the de-fanged IP address" do
+ expect {
+ subject.process_value(refanged)
+ }.to output("#{defanged}#{$/}").to_stdout
+ end
+ end
+ end
+end
diff --git a/spec/cli/commands/refang_spec.rb b/spec/cli/commands/refang_spec.rb
index 7d3f3bc1f..d234a131b 100644
--- a/spec/cli/commands/refang_spec.rb
+++ b/spec/cli/commands/refang_spec.rb
@@ -7,7 +7,7 @@
describe "#process_value" do
context "when given a defanged URL value" do
- let(:defanged) { 'hxxps://www[.]evil[.]com/foo/bar/baz' }
+ let(:defanged) { 'hxxps[://]www[.]evil[.]com/foo/bar/baz' }
let(:refanged) { 'https://www.evil.com/foo/bar/baz' }
it "must print the re-fanged URL" do