require 'pinpress/constants'
require 'pinpress/template'
# The PinPress module, which wraps everything
# in this gem.
module PinPress
extend self
class << self
# Stores whether initalization has completed.
# @return [Boolean]
attr_reader :initialized
# Stores whether verbose output is turned on.
# @return [Boolean]
attr_accessor :verbose
end
# Determines whether an invalid combination of linking options
# (auto and manual, via either a switch or a config parameter)
# has been given.
# @param [Hash] opts
# @return [Bool]
def conflicting_link_opts?(opts)
auto_link_flag = opts[:a]
auto_link_conf = configuration.pinpress.auto_link
manual_link_flag = opts[:l]
manual_link_conf = configuration.pinpress.manual_link
((auto_link_conf && manual_link_conf) || (auto_link_flag && manual_link_flag))
end
# Determines which URL linking options to set. There are some
# basic rules:
# 1. You can't specify both switches or both config options simultaneously.
# 2. Switches will take priority.
# 3. Config options are a last resort
# @param [Hash] opts_hash
# @return [Hash]
def determine_link_opts(opts_hash)
opts = {}
if opts_hash[:a_switch] || opts_hash[:m_switch]
opts.merge!(auto_link: true) if opts_hash[:a_switch]
opts.merge!(link: true) if opts_hash[:m_switch]
else
opts.merge!(auto_link: true) if opts_hash[:a_config]
opts.merge!(link: true) if opts_hash[:m_config]
end
opts
end
# Grabs Pinboard data (passed on passed options) and yields a block that
# allows the user to act upon that returned data.
# @param [Hash] pinboard_opts
# @yield pins
# @raise StandardError if Pinboard client fails
def execute_template(pinboard_opts)
begin
client = Pinboard::Client.new(token: configuration.pinpress.api_token)
pins = client.posts(pinboard_opts)
if pins.empty?
messenger.warn('No data matching your arguments...')
else
yield pins
end
rescue StandardError => e
messenger.debug(e.to_s)
puts e.to_s
raise "Pinboard API failed; are you sure you've run " \
" `pinpress init` (and that your API key is correct)?"
end
end
# Returns a template Hash from the configuration file.
# @param [String] template_name The name of the template to get
# @param [Fixnum] template_type The template type
# @return [Hash]
def get_template(template_name, template_type)
case template_type
when PinPress::Template::TYPE_PIN
configuration.pin_templates.find { |t| t.keys[0] == template_name.to_sym}.values[0]
when PinPress::Template::TYPE_TAG
configuration.tag_templates.find { |t| t.keys[0] == template_name.to_sym}.values[0]
end
end
# "Initializes" a passed template:
# 1. If the template exists, returns it.
# 2. If not, return a default template (if it exists).
# 3. Throw an exception if #1 and #2 fail.
# @param [String] explicit_template A template name passed in via the CLI
# @param [Fixnum] template_type The template type
# @return [Hash]
# @raise StandardError if no explicit or default template is found
def init_template(explicit_template, template_type)
pin_t_sym = :default_pin_template
tag_t_sym = :default_tag_template
s = (template_type == PinPress::Template::TYPE_PIN ? pin_t_sym : tag_t_sym)
default_template = configuration.pinpress[s]
if explicit_template && PinPress.is_template?(explicit_template, template_type)
messenger.debug("Using explicit template: #{ explicit_template }")
return explicit_template, PinPress.get_template(explicit_template, template_type)
elsif PinPress.is_template?(default_template, template_type)
messenger.debug("Using default template: #{ default_template }")
return default_template, PinPress.get_template(default_template, template_type)
else
fail 'Invalid template specified and/or no default template defined'
end
end
# Initializes PinPress by downloading and
# collecting all necessary items and info.
# @param [Boolean] from_scratch
# @return [void]
def init(from_scratch = false)
messenger.section('INITIALIZING...')
if from_scratch
configuration.reset
# Add initial configuration info.
configuration.add_section(:pinpress)
configuration.add_section(:links)
configuration.pinpress = {
config_location: configuration.config_path,
default_pin_template: 'pinpress_default',
default_tag_template: 'pinpress_default',
log_level: 'WARN',
version: PinPress::VERSION
}
# Add a single default pin and tag template
default_pin_template = {
pinpress_default: {
opener: '
',
item: '- <%= href %>
',
closer: '
'
}
}
default_tag_template = {
pinpress_default: {
item: '<%= tag %> (<%= count %>),'
}
}
configuration.pin_templates = [default_pin_template]
configuration.tag_templates = [default_tag_template]
end
# Run through installation prompts and ingest the results into
# the configuration.
pm = CLIUtils::Prefs.new(PinPress::PREF_FILES['INIT'], configuration)
pm.ask
configuration.ingest_prefs(pm)
m = "Configuration values after pref collection: #{ configuration.data }"
messenger.debug(m)
configuration.save
@initialized = true
end
# Determines whether a template exists in the configuration file.
# @param [String] template_name The name of the template to search for
# @param [Fixnum] template_type The template type
def is_template?(template_name, template_type)
case template_type
when PinPress::Template::TYPE_PIN
templates = configuration.pin_templates
when PinPress::Template::TYPE_TAG
templates = configuration.tag_templates
end
!templates.find { |t| t.keys[0] == template_name.to_sym }.nil?
end
def link_urls_in_desc(name, description, method)
fail "Unknown link creation methdo: #{ method.to_s }" unless [:AUTO, :MANUAL].include? method
urls = URI.extract(description, ['http', 'https'])
urls.each do |u|
link_text = nil
pin_id = Digest::MD5.hexdigest(description + u)
# I don't get why, but URL.extract is loose enough to include
# weird characters. This is my evolving regex to handle those.
u.sub!(/[()\.]+$/, '')
if configuration.links.send(pin_id)
# First, check the configuration file to see if we've stored
# this URL before (so that we can grab the saved value).
link_text = configuration.links.send(pin_id).link_text
else
if method == :AUTO
# If the configuration file doesn't have an entry for this
# link, no worries; create one.
link_text = u
elsif method == :MANUAL
# If the configuration file doesn't have an entry for this
# link, no worries; create one.
CLIUtils::PrettyIO.wrap = false
messenger.section('URL FOUND!')
messenger.info("URL:\t\t#{ u }")
messenger.info("TITLE:\t#{ name }")
messenger.info("POSITION:\t..." + description.scan(/.{0,40}#{ u }.{0,40}/)[0] + '...')
until !link_text.nil?
link_text = messenger.prompt('What do you want the link text to say?')
messenger.warn('Please provide some link text.') if link_text.nil?
end
CLIUtils::PrettyIO.wrap = true
end
# Store this newly created link info back in the configuration
# file.
configuration.links.merge!(pin_id => {
title: name,
url: u,
link_text: link_text
})
end
description.sub!(u, "#{ link_text }")
end
description
end
# Present a list of installed templates to the user
# @return [void]
def list_templates
%w(pin tag).each do |type|
templates = configuration.send("#{ type }_templates")
messenger.section("AVAILABLE #{ type.upcase } TEMPLATES:")
if templates
templates.each_with_index do |template, index|
template_name, template = template.first
puts "#{ index + 1 }.\tName: ".blue + "#{ template_name }"
puts "Opener:".blue.rjust(22) + "\t#{ template[:opener] }".truncate(80)
puts "Item:".blue.rjust(22) + "\t#{ template[:item] }".truncate(80)
puts "Closer:".blue.rjust(22) + "\t#{ template[:closer] }".truncate(80)
end
else
messenger.warn('No templates defined...')
end
end
end
# Helper method to merge command line options that are relevant for both pin
# and tag requests.
# @param [Hash] options
# @raise StandardError if an invalid combo of linking options is given
# @return [Hash]
def merge_common_options(options, template_name, template_type)
case template_type
when PinPress::Template::TYPE_PIN
section = configuration.pin_templates.find { |t| t.keys[0] == template_name.to_sym}.values[0]
when PinPress::Template::TYPE_TAG
section = configuration.tag_templates.find { |t| t.keys[0] == template_name.to_sym}.values[0]
end
opts = {}
if options[:n]
opts.merge!(results: options[:n])
elsif section.default_num_results
opts.merge!(results: section.default_num_results)
end
if options[:t]
opts.merge!(tag: options[:t])
elsif section.default_tags
opts.merge!(tag: section.default_tags.join(','))
end
# These options are PinPress-related, not necessarily Pinboard-related;
# for the sake of convenience, they're included here.
# Auto-linking and prompting for link text don't go together, so make
# sure to let the user know if they include both.
if conflicting_link_opts?(options)
fail "You can't specify (a) both the `-a` and `-l` switches or " \
"(b) both the `auto_link` and `manual_link` configuration options."
else
link_opts = determine_link_opts({
a_switch: options[:a],
m_switch: options[:l],
a_config: configuration.pinpress.auto_link,
m_config: configuration.pinpress.manual_link
})
opts.merge!(link_opts) if link_opts
end
opts
end
# Creates text output from pin data (based upon a passed template).
# @param [Hash] template The template to use
# @param [Hash] opts CLI options to use
# @return [String]
def pin_yield(template, opts)
output = ''
PinPress.execute_template(opts) do |data|
output += template[:opener] if template[:opener]
data.each do |i|
name = HTMLEntities.new.encode(i[:description])
if opts[:link]
desc = link_urls_in_desc(name, i[:extended], :MANUAL)
elsif opts[:auto_link]
desc = link_urls_in_desc(name, i[:extended], :AUTO)
else
desc = i[:extended]
end
href = i[:href]
description = name
extended = desc
tag = i[:tag]
time = i[:time]
replace = i[:replace]
shared = i[:shared]
toread = i[:toread]
output += ERB.new(template[:item]).result(binding)
end
output += template[:closer] if template[:closer]
end
output
end
# Creates text output from tag data (based upon a passed template).
# @param [Hash] template The template to use
# @param [Hash] opts CLI options to use
# @return [String]
def tag_yield(template, opts)
output = ''
PinPress.execute_template(opts) do |data|
tags = []
data.each { |i| tags += i[:tag] }
tags = tags.uniq.map do |t|
{ tag: t, count: tags.count(t) }
end
output += template[:opener] if template[:opener]
tags.each do |t|
unless t[:tag] == opts[:tag]
tag = t[:tag]
count = t[:count]
output += ERB.new(template.item).result(binding)
end
end
output += template[:closer] if template[:closer]
end
output
end
# Notifies the user that the config file needs to be
# re-done and does it.
# @return [void]
def update_config_file
m = "This version needs to make some config changes. Don't worry; " \
"when prompted, your current values for existing config options " \
"will be presented (so it'll be easier to fly through the upgrade)."
messenger.info(m)
messenger.prompt('Press enter to continue')
PinPress.init(true)
end
end