mirror of
https://github.com/jimeh/dotfiles.git
synced 2026-02-19 06:46:40 +00:00
Also fetch data from brew services as JSON instead of parsing the plain text output of `brew services list`.
384 lines
10 KiB
Ruby
Executable File
384 lines
10 KiB
Ruby
Executable File
#!/usr/bin/env ruby
|
|
# frozen_string_literal: true
|
|
|
|
# <xbar.title>Brew Services</xbar.title>
|
|
# <xbar.version>v2.3.0</xbar.version>
|
|
# <xbar.author>Jim Myhrberg</xbar.author>
|
|
# <xbar.author.github>jimeh</xbar.author.github>
|
|
# <xbar.desc>List and manage Homebrew Services</xbar.desc>
|
|
# <xbar.image>https://i.imgur.com/RDfpTLl.png</xbar.image>
|
|
# <xbar.dependencies>ruby</xbar.dependencies>
|
|
# <xbar.abouturl>https://github.com/jimeh/dotfiles/tree/main/xbar</xbar.abouturl>
|
|
#
|
|
# <xbar.var>boolean(VAR_GROUPS=true): List services in started/stopped groups?</xbar.var>
|
|
# <xbar.var>string(VAR_BREW_PATH="/usr/local/bin/brew"): Path to "brew" executable.</xbar.var>
|
|
|
|
# rubocop:disable Style/IfUnlessModifier
|
|
|
|
require 'open3'
|
|
require 'json'
|
|
|
|
module Xbar
|
|
class Printer
|
|
attr_reader :nested_level
|
|
|
|
SUB_STR = '--'
|
|
SEP_STR = '---'
|
|
PARAM_SEP = '|'
|
|
|
|
def initialize(nested_level = 0)
|
|
@nested_level = nested_level
|
|
end
|
|
|
|
def item(label = nil, **props)
|
|
print_item(label, **props) if !label.nil? && !label.empty?
|
|
|
|
yield(sub_printer) if block_given?
|
|
end
|
|
|
|
def separator
|
|
print_item(SEP_STR)
|
|
end
|
|
alias sep separator
|
|
|
|
private
|
|
|
|
def print_item(text, **props)
|
|
props = props.dup
|
|
alt = props.delete(:alt)
|
|
|
|
output = [text]
|
|
unless props.empty?
|
|
props = normalize_props(props)
|
|
output << PARAM_SEP
|
|
output += props.map { |k, v| "#{k}=\"#{v}\"" }
|
|
end
|
|
|
|
$stdout.print(SUB_STR * nested_level, output.join(' '))
|
|
$stdout.puts
|
|
|
|
return if alt.nil? || alt.empty?
|
|
|
|
print_item(alt, **props.merge(alternate: true))
|
|
end
|
|
|
|
def plugin_refresh_uri
|
|
@plugin_refresh_uri ||= 'xbar://app.xbarapp.com/refreshPlugin' \
|
|
"?path=#{File.basename(__FILE__)}"
|
|
end
|
|
|
|
def normalize_props(props = {})
|
|
props = props.dup
|
|
|
|
if props[:shell].is_a?(Array)
|
|
cmd = props[:shell]
|
|
props[:shell] = cmd[0]
|
|
cmd[1..-1].each_with_index do |c, i|
|
|
props["param#{i + 1}".to_sym] = c
|
|
end
|
|
end
|
|
|
|
# Refresh Xbar after shell command has run in terminal
|
|
if props[:terminal] && props[:refresh] && props[:shell]
|
|
props[:refresh] = false
|
|
i = 1
|
|
i += 1 while props.key?("param#{i}".to_sym)
|
|
props["param#{i}".to_sym] = ';'
|
|
props["param#{i + 1}".to_sym] = 'open'
|
|
props["param#{i + 2}".to_sym] = '-jg'
|
|
props["param#{i + 3}".to_sym] = "'#{plugin_refresh_uri}'"
|
|
end
|
|
|
|
props
|
|
end
|
|
|
|
def sub_printer
|
|
@sub_printer || self.class.new(nested_level + 1)
|
|
end
|
|
end
|
|
end
|
|
|
|
module Brew
|
|
class CommandError < StandardError; end
|
|
|
|
class Common
|
|
def self.prefix(value = nil)
|
|
return @prefix if value.nil? || value == ''
|
|
|
|
@prefix = value
|
|
end
|
|
|
|
private
|
|
|
|
def prefix
|
|
self.class.prefix
|
|
end
|
|
|
|
def default_printer
|
|
@default_printer ||= ::Xbar::Printer.new
|
|
end
|
|
|
|
def cmd(*args)
|
|
out, err, s = Open3.capture3(*args)
|
|
raise CommandError, "#{args.join(' ')}: #{err}" if s.exitstatus != 0
|
|
|
|
out
|
|
end
|
|
|
|
def brew_path
|
|
@brew_path ||= ENV.fetch('VAR_BREW_PATH', '/usr/local/bin/brew')
|
|
end
|
|
|
|
def brew_check(printer = nil)
|
|
printer ||= default_printer
|
|
return if File.exist?(brew_path)
|
|
|
|
printer.item("#{prefix}↑:warning:", dropdown: false)
|
|
printer.sep
|
|
printer.item('Homebrew not found', color: 'red')
|
|
printer.item("Executable \"#{brew_path}\" does not exist.")
|
|
printer.sep
|
|
printer.item(
|
|
'Visit https://brew.sh/ for installation instructions',
|
|
href: 'https://brew.sh'
|
|
)
|
|
|
|
exit 0
|
|
end
|
|
end
|
|
|
|
class Service
|
|
attr_reader :name, :status, :user, :file, :exit_code
|
|
|
|
def initialize(args = {})
|
|
@name = args.key?('name') ? args['name'] : args[:name]
|
|
@status = args.key?('status') ? args['status'] : args[:status]
|
|
@user = args.key?('user') ? args['user'] : args[:user]
|
|
@file = args.key?('file') ? args['file'] : args[:file]
|
|
@exit_code = args.key?('exit_code') ? args['exit_code'] : args[:exit_code]
|
|
end
|
|
|
|
def started?
|
|
@started ||= %w[started scheduled].include?(@status.downcase)
|
|
end
|
|
|
|
def stopped?
|
|
@stopped ||= %w[stopped none].include?(@status.downcase)
|
|
end
|
|
|
|
def error?
|
|
@error ||= @status.downcase == 'error'
|
|
end
|
|
|
|
def unknown_status?
|
|
@unknown_status ||= @status.downcase == 'unknown'
|
|
end
|
|
end
|
|
|
|
class Services < Common
|
|
prefix ':bulb:'
|
|
|
|
def run
|
|
printer = default_printer
|
|
|
|
brew_check(printer)
|
|
|
|
printer.item(
|
|
"#{prefix}" \
|
|
"#{started_services.size}/#{services.size - started_services.size}",
|
|
dropdown: false
|
|
)
|
|
printer.sep
|
|
printer.item('Brew Services')
|
|
|
|
printer.item(status_label) do |printer|
|
|
printer.sep
|
|
printer.item(':hourglass: Refresh', refresh: true)
|
|
printer.sep
|
|
if stopped_services.size.positive?
|
|
printer.item(
|
|
"Start All (#{stopped_services.size} services)",
|
|
terminal: false, refresh: true,
|
|
shell: [brew_path, 'services', 'start', '--all']
|
|
)
|
|
else
|
|
printer.item("Start All (#{stopped_services.size} services)")
|
|
end
|
|
if started_services.size.positive?
|
|
printer.item(
|
|
"Stop All (#{started_services.size} services)",
|
|
terminal: false, refresh: true,
|
|
shell: [brew_path, 'services', 'stop', '--all']
|
|
)
|
|
else
|
|
printer.item("Stop All (#{started_services.size} services)")
|
|
end
|
|
if services.size.positive?
|
|
printer.item(
|
|
'Restart All ' \
|
|
"(#{started_services.size + stopped_services.size} services)",
|
|
terminal: false, refresh: true,
|
|
shell: [brew_path, 'services', 'restart', '--all']
|
|
)
|
|
else
|
|
printer.item("Restart All (#{services.size} services)")
|
|
end
|
|
end
|
|
use_groups? ? print_service_groups(printer) : print_services(printer)
|
|
end
|
|
|
|
private
|
|
|
|
def status_label
|
|
label = []
|
|
if started_services.size.positive?
|
|
label << "#{started_services.size} started"
|
|
end
|
|
if stopped_services.size.positive?
|
|
label << "#{stopped_services.size} stopped"
|
|
end
|
|
if errored_services.size.positive?
|
|
label << "#{errored_services.size} error"
|
|
end
|
|
if unknown_status_services.size.positive?
|
|
label << "#{unknown_status_services.size} unknown"
|
|
end
|
|
|
|
label = ['no services available'] if label.empty?
|
|
label.join(' / ')
|
|
end
|
|
|
|
def use_groups?
|
|
ENV.fetch('VAR_GROUPS', 'true') == 'true'
|
|
end
|
|
|
|
def print_service_groups(printer)
|
|
if started_services.size.positive?
|
|
printer.sep
|
|
printer.item("Started (#{started_services.size}):")
|
|
started_services.each do |service|
|
|
print_service(printer, service)
|
|
end
|
|
end
|
|
if stopped_services.size.positive?
|
|
printer.sep
|
|
printer.item("Stopped (#{stopped_services.size}):")
|
|
stopped_services.each do |service|
|
|
print_service(printer, service)
|
|
end
|
|
end
|
|
if errored_services.size.positive?
|
|
printer.sep
|
|
printer.item("Error (#{errored_services.size}):")
|
|
errored_services.each do |service|
|
|
print_service(printer, service)
|
|
end
|
|
end
|
|
if unknown_status_services.size.positive?
|
|
printer.sep
|
|
printer.item("Unknown Status (#{unknown_status_services.size}):")
|
|
unknown_status_services.each do |service|
|
|
print_service(printer, service)
|
|
end
|
|
end
|
|
end
|
|
|
|
def print_services(printer)
|
|
printer.sep
|
|
services.each do |service|
|
|
print_service(printer, service)
|
|
end
|
|
end
|
|
|
|
def print_service(printer, service)
|
|
icon = if service.started?
|
|
':white_check_mark:'
|
|
elsif service.stopped?
|
|
':ballot_box_with_check:'
|
|
elsif service.error?
|
|
':warning:'
|
|
elsif service.unknown_status?
|
|
':question:'
|
|
end
|
|
|
|
printer.item("#{icon} #{service.name}") do |printer|
|
|
if service.stopped? || service.unknown_status?
|
|
printer.item(
|
|
'Start',
|
|
terminal: false, refresh: true,
|
|
shell: [brew_path, 'services', 'start', service.name]
|
|
)
|
|
end
|
|
if service.started? || service.error? || service.unknown_status?
|
|
printer.item(
|
|
'Stop',
|
|
terminal: false, refresh: true,
|
|
shell: [brew_path, 'services', 'stop', service.name]
|
|
)
|
|
printer.item(
|
|
'Restart',
|
|
terminal: false, refresh: true,
|
|
shell: [brew_path, 'services', 'restart', service.name]
|
|
)
|
|
end
|
|
|
|
printer.sep
|
|
printer.item("Status: #{service.status}")
|
|
printer.item("User: #{service.user || '<none>'}")
|
|
if !service.exit_code.nil? && !service.started?
|
|
printer.item("Exit code: #{service.exit_code}")
|
|
end
|
|
|
|
if service.stopped?
|
|
printer.sep
|
|
printer.item('Uninstall') do |printer|
|
|
printer.item('Are you sure?')
|
|
printer.sep
|
|
printer.item(
|
|
'Yes',
|
|
terminal: true, refresh: true,
|
|
shell: [brew_path, 'uninstall', service.name]
|
|
)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def started_services
|
|
@started_services ||= services.select(&:started?)
|
|
end
|
|
|
|
def stopped_services
|
|
@stopped_services ||= services.select(&:stopped?)
|
|
end
|
|
|
|
def errored_services
|
|
@errored_services ||= services.select(&:error?)
|
|
end
|
|
|
|
def unknown_status_services
|
|
@unknown_status_services ||= services.select(&:unknown_status?)
|
|
end
|
|
|
|
def services
|
|
return @services if @services
|
|
|
|
output = cmd(brew_path, 'services', 'list', '--json')
|
|
data = JSON.parse(output)
|
|
|
|
@services = data.each_with_object([]) do |item, memo|
|
|
memo.push(Service.new(item))
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
begin
|
|
Brew::Services.new.run
|
|
rescue StandardError => e
|
|
puts "ERROR: #{e.message}:\n\t#{e.backtrace.join("\n\t")}"
|
|
exit 1
|
|
end
|
|
|
|
# rubocop:enable Style/IfUnlessModifier
|