# frozen_string_literal: true
#
# ronin-db-activerecord - ActiveRecord backend for the Ronin Database.
#
# Copyright (c) 2022-2024 Hal Brodigan (postmodern.mod3 at gmail.com)
#
# ronin-db-activerecord is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published
# by the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# ronin-db-activerecord 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 Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with ronin-db-activerecord. If not, see .
#
require 'ronin/db/model'
require 'active_record'
module Ronin
module DB
#
# Represents a TCP or UDP port.
#
class Port < ActiveRecord::Base
include Model
# @!attribute [rw] id
# The primary key of the port.
#
# @return [Integer]
attribute :id, :integer
# @!attribute [rw] protocol
# The protocol of the port (either `'tcp'` / `'udp'`).
#
# @return ["tcp", "udp"]
enum :protocol, {tcp: 'tcp', udp: 'udp'}, default: :tcp
validates :protocol, presence: true
# @!attribute [rw] number
# The port number.
#
# @return [Integer]
attribute :number, :integer
validates :number, presence: true,
numericality: {
greater_than_or_equal_to: 1,
less_than_or_equal_to: 65535
},
uniqueness: {scope: :protocol}
# @!attribute [r] created_at
# Defines the created_at timestamp
#
# @return [Time]
#
# @since 0.2.0
attribute :created_at, :datetime
# @!attribute [rw] open_ports
# The open ports.
#
# @return [Array]
has_many :open_ports, dependent: :destroy
# @!attribute [rw] ip_addresses
# The IP Addresses that that have this port open.
#
# @return [Array]
#
# @since 0.2.0
has_many :ip_addresses, through: :open_ports
# @!attribute [rw] services
# The services that that use this port.
#
# @return [Array]
#
# @since 0.2.0
has_many :services, through: :open_ports
# @!attribute [rw] notes
# The associated notes.
#
# @return [Array]
#
# @since 0.2.0
has_many :notes, dependent: :destroy
#
# Queries all ports with the port number or in the port range.
#
# @param [Integer, Range] number
# The port number or range.
#
# @return [Array]
# The ports with the port number or in the port range.
#
# @api public
#
# @since 0.2.0
#
def self.with_number(number)
where(number: number)
end
#
# Queries all ports with the protocol.
#
# @param [:tcp, ;udp] protocol
# The protocol to search for.
#
# @return [Array]
# The ports that use the protocol.
#
# @api public
#
# @since 0.2.0
#
def self.with_protocol(protocol)
where(protocol: protocol)
end
#
# Queries all ports associated with the service name.
#
# @param [String] name
# The service name to search for.
#
# @return [Array]
# The ports associated with the service name.
#
# @api public
#
# @since 0.2.0
#
def self.with_service_name(name)
joins(open_ports: :service).where(
open_ports: {
ronin_services: {name: name}
}
)
end
#
# Queries all ports associated with the IP address.
#
# @param [String] address
# The IP address to search by.
#
# @return [Array]
# The ports associated with the IP address.
#
# @api public
#
# @since 0.2.0
#
def self.with_ip_address(address)
joins(open_ports: :ip_address).where(
open_ports: {
ronin_ip_addresses: {address: address}
}
)
end
#
# Looks up a port by it's number.
#
# @param [String, Integer] number
# The port number to query.
#
# @return [Port, nil]
# The found port number.
#
# @api public
#
def self.lookup(number)
find_by(number: number)
end
#
# Creates a new Port.
#
# @param [String, Integer] number
# The port number.
#
# @return [Port]
# The new or previously saved port.
#
# @api public
#
def self.import(number)
create(number: number)
end
#
# Converts the port to an integer.
#
# @return [Integer]
# The port number.
#
# @api public
#
def to_i
self.number.to_i
end
#
# Converts the port to a string.
#
# @return [String]
# The port number and protocol.
#
# @api public
#
def to_s
"#{self.number}/#{self.protocol}"
end
end
end
end
require 'ronin/db/open_port'
require 'ronin/db/note'