feat(xbar): switch to xbar (BitBar replacement) and re-write my scripts

This is a complete rewrite of my old scripts, and a wholesale
replacement of BitBar's "brew-services" script with a more feature
packed variant.
This commit is contained in:
2021-10-31 22:39:25 +00:00
parent deaa9aebc8
commit 0f71533a8c
8 changed files with 742 additions and 277 deletions

209
xbar/brew-cask-updates.1h.rb Executable file
View File

@@ -0,0 +1,209 @@
#!/usr/bin/env ruby
# frozen_string_literal: true
# <xbar.title>Brew Cask Updates</xbar.title>
# <xbar.version>v2.0.0</xbar.version>
# <xbar.author>Jim Myhrberg</xbar.author>
# <xbar.author.github>jimeh</xbar.author.github>
# <xbar.desc>Show outdated Homebrew casks</xbar.desc>
# <xbar.image>https://i.imgur.com/aAD0pqO.png</xbar.image>
# <xbar.dependencies>ruby</xbar.dependencies>
#
# <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)
output = [text]
unless props.empty?
output << PARAM_SEP
output += props.map { |k, v| "#{k}=\"#{v}\"" }
end
$stdout.print(SUB_STR * nested_level, output.join(' '))
$stdout.puts
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
def brew_update
cmd(brew_path, 'update')
rescue CommandError => e
# Continue as if nothing happened when brew update fails, as it likely
# to be due to another update process is already running.
end
end
class Cask
attr_reader :name, :installed_version, :latest_version
def initialize(attributes = {})
@name = attributes['name']
@installed_version = attributes['installed_versions']
@latest_version = attributes['current_version']
end
alias current_version installed_version
end
class CaskUpdates < Common
prefix ':tropical_drink:'
def run
printer = default_printer
brew_check(printer)
brew_update
printer.item("#{prefix}#{casks.size}", dropdown: false)
printer.sep
printer.item('Brew Cask Updates')
printer.item("#{casks.size} outdated") do |printer|
printer.sep
printer.item(':hourglass: Refresh', refresh: true)
end
print_casks(printer)
printer.sep
printer.item('Refresh', refresh: true)
end
private
def print_casks(printer)
return unless casks.size.positive?
printer.item(
'Upgrade all casks',
terminal: true, refresh: true,
shell: brew_path, param1: 'upgrade'
)
printer.sep
printer.item('Upgrade:')
casks.each do |cask|
printer.item(cask.name) do |printer|
printer.item(
'Upgrade',
terminal: true, refresh: true, shell: brew_path,
param1: 'upgrade', param2: '--cask', param3: cask.name
)
printer.item(
"Upgrade (#{cask.current_version}#{cask.latest_version})",
alternate: true, terminal: true, refresh: true,
shell: brew_path, param1: 'upgrade', param2: '--cask',
param3: cask.name
)
printer.sep
printer.item("Installed: #{cask.installed_version}")
printer.item("Latest: #{cask.latest_version}")
printer.sep
printer.item('Uninstall') do |printer|
printer.item('Are you sure?')
printer.sep
printer.item(
'Yes',
terminal: true, refresh: true,
shell: brew_path, param1: 'uninstall',
param2: '--cask', param3: cask.name
)
end
end
end
end
def casks
@casks ||= JSON.parse(
cmd(brew_path, 'outdated', '--cask', '--json')
)['casks'].map { |line| Cask.new(line) }
end
end
end
begin
Brew::CaskUpdates.new.run
rescue StandardError => e
puts "ERROR: #{e.message}:\n\t#{e.backtrace.join("\n\t")}"
exit 1
end
# rubocop:enable Style/IfUnlessModifier

270
xbar/brew-formula-updates.1h.rb Executable file
View File

@@ -0,0 +1,270 @@
#!/usr/bin/env ruby
# frozen_string_literal: true
# <xbar.title>Brew Formula Updates</xbar.title>
# <xbar.version>v2.0.0</xbar.version>
# <xbar.author>Jim Myhrberg</xbar.author>
# <xbar.author.github>jimeh</xbar.author.github>
# <xbar.desc>Show outdated Homebrew formulas</xbar.desc>
# <xbar.image>https://i.imgur.com/6PC6OPg.png</xbar.image>
# <xbar.dependencies>ruby</xbar.dependencies>
#
# <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)
output = [text]
unless props.empty?
output << PARAM_SEP
output += props.map { |k, v| "#{k}=\"#{v}\"" }
end
$stdout.print(SUB_STR * nested_level, output.join(' '))
$stdout.puts
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
def brew_update
cmd(brew_path, 'update')
rescue CommandError => e
# Continue as if nothing happened when brew update fails, as it likely
# to be due to another update process is already running.
end
end
class Formula
attr_reader :name, :installed_versions, :latest_version,
:pinned, :pinned_version
def initialize(attributes = {})
@name = attributes['name']
@installed_versions = attributes['installed_versions']
@latest_version = attributes['current_version']
@pinned = attributes['pinned']
@pinned_version = attributes['pinned_version']
end
def current_version
installed_versions.last
end
end
class FormulaUpdates < Common
prefix ':beer:'
def run
printer = default_printer
brew_check(printer)
brew_update
printer.item("#{prefix}#{formulas.size}", dropdown: false)
printer.sep
printer.item('Brew Formula Updates')
pinned_msg = " / #{pinned.size} pinned" if pinned.size.positive?
printer.item(
"#{formulas.size} outdated#{pinned_msg}"
) do |printer|
printer.sep
printer.item(':hourglass: Refresh', refresh: true)
end
print_formulas(printer)
print_pinned(printer)
printer.sep
printer.item('Refresh', refresh: true)
end
private
def print_formulas(printer)
return unless formulas.size.positive?
printer.item(
'Upgrade all formula',
terminal: true, refresh: true, shell: brew_path, param1: 'upgrade'
)
printer.sep
printer.item('Upgrade:')
formulas.each do |formula|
printer.item(formula.name) do |printer|
printer.item(
'Upgrade',
terminal: true, refresh: true,
shell: brew_path, param1: 'upgrade', param2: formula.name
)
printer.item(
"Upgrade (#{formula.current_version}#{formula.latest_version})",
alternate: true, terminal: true, refresh: true,
shell: brew_path, param1: 'upgrade', param2: formula.name
)
printer.sep
printer.item("Installed: #{formula.installed_versions.join(', ')}")
printer.item("Latest: #{formula.latest_version}")
printer.sep
printer.item(
'Pin',
terminal: false, refresh: true,
shell: brew_path, param1: 'pin', param2: formula.name
)
printer.item(
"Pin (to #{formula.current_version})",
alternate: true, terminal: false, refresh: true,
shell: brew_path, param1: 'pin', param2: formula.name
)
printer.item('Uninstall') do |printer|
printer.item('Are you sure?')
printer.sep
printer.item(
'Yes',
terminal: true, refresh: true,
shell: brew_path, param1: 'uninstall', param2: formula.name
)
end
end
end
end
def print_pinned(printer)
return unless pinned.size.positive?
printer.sep
printer.item('Pinned:')
pinned.each do |formula|
printer.item(formula.name) do |printer|
printer.item('Upgrade')
printer.item(
"Upgrade (#{formula.current_version}#{formula.latest_version})",
alternate: true
)
printer.sep
printer.item("Pinned: #{formula.pinned_version}")
if formula.installed_versions.size > 1
printer.item("Installed: #{formula.installed_versions.join(', ')}")
end
printer.item("Latest: #{formula.latest_version}")
printer.sep
printer.item(
'Unpin',
terminal: false, refresh: true,
shell: brew_path, param1: 'unpin', param2: formula.name
)
printer.item('Uninstall') do |printer|
printer.item('Are you sure?')
printer.sep
printer.item(
'Yes',
terminal: true, refresh: true,
shell: brew_path, param1: 'uninstall', param2: formula.name
)
end
end
end
end
def formulas
@formulas ||= outdated.reject(&:pinned)
end
def pinned
@pinned ||= outdated.select(&:pinned)
end
def outdated
@outdated ||= JSON.parse(
cmd(brew_path, 'outdated', '--formula', '--json')
)['formulae'].map { |line| Formula.new(line) }
end
end
end
begin
Brew::FormulaUpdates.new.run
rescue StandardError => e
puts "ERROR: #{e.message}:\n\t#{e.backtrace.join("\n\t")}"
exit 1
end
# rubocop:enable Style/IfUnlessModifier

252
xbar/brew-services.10m.rb Executable file
View File

@@ -0,0 +1,252 @@
#!/usr/bin/env ruby
# frozen_string_literal: true
# <xbar.title>Brew Services</xbar.title>
# <xbar.version>v2.0.0</xbar.version>
# <xbar.author>Jim Myhrberg</xbar.author>
# <xbar.author.github>jimeh</xbar.author.github>
# <xbar.desc>List and manage Brew Services</xbar.desc>
# <xbar.image>https://i.imgur.com/cAVfsvF.png</xbar.image>
# <xbar.dependencies>ruby</xbar.dependencies>
#
# <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)
output = [text]
unless props.empty?
output << PARAM_SEP
output += props.map { |k, v| "#{k}=\"#{v}\"" }
end
$stdout.print(SUB_STR * nested_level, output.join(' '))
$stdout.puts
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
def self.from_line(line)
name, status, user, _plist = line.split
new(name: name, status: status, user: user)
end
def initialize(name:, status:, user: nil)
@name = name
@status = status
@user = user
end
def started?
@started ||= @status.downcase == 'started'
end
def stopped?
!started?
end
end
class Services < Common
prefix ':bulb:'
def run
printer = default_printer
brew_check(printer)
printer.item(
"#{prefix}#{started_services.size}/#{stopped_services.size}",
dropdown: false
)
printer.sep
printer.item('Brew Services')
printer.item(
"#{started_services.size} started / " \
"#{stopped_services.size} stopped"
) do |printer|
printer.sep
printer.item(':hourglass: Refresh', refresh: true)
end
printer.sep
use_groups? ? print_service_groups(printer) : print_services(printer)
end
private
def use_groups?
ENV.fetch('VAR_GROUPS', 'true') == 'true'
end
def print_service_groups(printer)
printer.item('Started:')
started_services.each do |service|
print_service(printer, service)
end
printer.sep
printer.item('Stopped:')
stopped_services.each do |service|
print_service(printer, service)
end
end
def print_services(printer)
services.each do |service|
print_service(printer, service)
end
end
def print_service(printer, service)
label = if service.started?
":white_check_mark: #{service.name}"
else
":ballot_box_with_check: #{service.name}"
end
printer.item(label) do |printer|
if service.started?
printer.item(
'Stop',
terminal: false, refresh: true, shell: brew_path,
param1: 'services', param2: 'stop', param3: service.name
)
else
printer.item(
'Start',
terminal: false, refresh: true, shell: brew_path,
param1: 'services', param2: 'start', param3: service.name
)
end
printer.sep
printer.item("State: #{service.status}")
printer.item("User: #{service.user || '<none>'}")
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, param1: 'uninstall',
param2: service.name
)
end
end
end
end
def started_services
@started_services ||= services.select(&:started?)
end
def stopped_services
@stopped_services ||= services.reject(&:started?)
end
def services
@services ||= cmd(
brew_path, 'services', 'list'
).each_line.each_with_object([]) do |line, memo|
next if line.match(/^Name\s+Status\s+User\s+Plist$/)
memo.push(Service.from_line(line))
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