hannesg/uri_template

View on GitHub
lib/uri_template/colon.rb

Summary

Maintainability
A
1 hr
Test Coverage
# -*- encoding : utf-8 -*-
# The MIT License (MIT)
#
# Copyright (c) 2011-2014 Hannes Georg
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
# THE SOFTWARE.

require 'forwardable'

require 'uri_template'
require 'uri_template/utils'

module URITemplate

# A colon based template denotes variables with a colon.
#
# This template type is somewhat compatible with sinatra.
#
# @example
#   tpl = URITemplate::Colon.new('/foo/:bar')
#   tpl.extract('/foo/baz') #=> {'bar'=>'baz'}
#   tpl.expand('bar'=>'boom') #=> '/foo/boom'
#
class Colon

  include URITemplate

  VAR = /(?:\{:(\w+)\}|:(\w+)(?!\w)|\*)/u

  class InvalidValue < StandardError

    include URITemplate::InvalidValue

    attr_reader :variable, :value

    def initialize(variable, value)
      @variable = variable
      @value = value
      super(generate_message())
    end

    class SplatIsNotAnArray < self
    end

  protected

    def generate_message()
      return "The template variable " + variable.inspect + " cannot expand the given value "+ value.inspect
    end

  end

  class Token

    class Variable < self

      include URITemplate::Expression

      attr_reader :name

      def initialize(name)
        @name = name
        @variables = [name]
      end

      def expand(vars)
        return Utils.escape_url(Utils.object_to_param(vars[name]))
      end

      def to_r
        return '([^/]*?)'
      end

      def to_s
        return ":#{name}"
      end

    end

    class Splat < Variable

      SPLAT = 'splat'.freeze

      attr_reader :index

      def initialize(index)
        @index = index
        super(SPLAT)
      end

      def expand(vars)
        var = vars[name]
        if Array === var
          return Utils.escape_uri(Utils.object_to_param(var[index]))
        else
          raise InvalidValue::SplatIsNotAnArray.new(name,var)
        end
      end

      def to_r
        return '(.+?)'
      end

    end

    class Static < self

      include URITemplate::Literal

      def initialize(str)
        @string = str
      end

      def expand(_)
        return @string
      end

      def to_r
        return Regexp.escape(@string)
      end

    end

  end

  attr_reader :pattern

  # Tries to convert the value into a colon-template.
  # @example
  #   URITemplate::Colon.try_convert('/foo/:bar/').pattern #=> '/foo/:bar/'
  #   URITemplate::Colon.try_convert(URITemplate.new(:rfc6570, '/foo/{bar}/')).pattern #=> '/foo/{:bar}/'
  def self.try_convert(x)
    if x.kind_of? String
      return new(x)
    elsif x.kind_of? self
      return x
    elsif x.kind_of? URITemplate::RFC6570 and x.level == 1
      return new( x.pattern.gsub(/\{(.*?)\}/u){ "{:#{$1}}" } )
    else
      return nil
    end
  end

  def initialize(pattern)
    raise ArgumentError,"Expected a String but got #{pattern.inspect}" unless pattern.kind_of? String
    @pattern = pattern
  end

  # Extracts variables from an uri.
  #
  # @param uri [String]
  # @return nil,Hash
  def extract(uri)
    md = self.to_r.match(uri)
    return nil unless md
    result = {}
    splat = []
    self.tokens.select{|tk| tk.kind_of? URITemplate::Expression }.each_with_index do |tk,i|
      if tk.kind_of? Token::Splat
        splat << md[i+1]
        result['splat'] = splat unless result.key? 'splat'
      else
        result[tk.name] = Utils.unescape_url( md[i+1] )
      end
    end
    if block_given?
      return yield(result)
    end
    return result
  end

  def type
    :colon
  end

  def to_r
    @regexp ||= Regexp.new('\A' + tokens.map(&:to_r).join + '\z', Utils::KCODE_UTF8)
  end

  def tokens
    @tokens ||= tokenize!
  end

protected

  def tokenize!
    number_of_splats = 0
    RegexpEnumerator.new(VAR).each(@pattern).map{|x|
      if x.kind_of? String
        Token::Static.new(Utils.escape_uri(x))
      elsif x[0] == '*'
        n = number_of_splats
        number_of_splats = number_of_splats + 1
        Token::Splat.new(n)
      else
        # TODO: when rubinius supports ambigious names this could be replaced with x['name'] *sigh*
        Token::Variable.new(x[1] || x[2])
      end
    }.to_a
  end

end
end