#encoding: utf-8 require 'pathname' require 'time' require 'nori' require 'nokogiri' require 'httpclient' require "addressable/uri" this = Pathname.new(__FILE__).realpath lib_path = File.expand_path("..", this) $:.unshift(lib_path) $ROOT = File.expand_path("../", lib_path) Dir.glob(File.join(lib_path, '/**/*.rb')).each do |file| require file end module NSYapi class Configuration attr_accessor :username, :password end def self.configure(configuration = NSYapi::Configuration.new) yield configuration if block_given? @@configuration = configuration end def self.configuration # :nodoc: @@configuration ||= NSYapi::Configuration.new end def self.client @client_instance = NSClient.new(configuration.username, configuration.password) unless @client_instance @client_instance end end class NSClient attr_accessor :last_received_raw_xml, :last_received_corrected_xml def initialize(username, password) @client = HTTPClient.new @client.set_auth("http://webservices.ns.nl", username, password) @prices_url = PricesUrl.new("http://webservices.ns.nl/ns-api-prijzen-v3") @last_received_raw_xml = "" @last_received_corrected_xml = "" end def stations parse_stations(get_xml("http://webservices.ns.nl/ns-api-stations-v2")) end def stations_short parse_stations_as_map(get_xml("http://webservices.ns.nl/ns-api-stations-v2")) end def disruptions (query = nil) response_xml = get_xml(disruption_url(query)) raise_error_when_response_is_error(response_xml) parse_disruptions(response_xml) end def prices (opts = {from: nil, to: nil, via: nil, date: nil}) raise MissingParameter, "from and to station is required" if (opts[:from] == nil && opts[:to] == nil) raise MissingParameter, "from station is required" unless opts[:from] raise MissingParameter, "to station is required" unless opts[:to] raise SameDestinationError, "from (#{opts[:from]}) and to (#{opts[:to]}) parameters should not be equal" if opts[:from] == opts[:to] response_xml = get_xml(@prices_url.url(opts)) raise_error_when_response_is_error(response_xml) parse_prices(response_xml) end def parse_prices(response_xml) prices_response = PricesResponse.new (response_xml/'/VervoerderKeuzes/VervoerderKeuze').each do |transporter| prices_response.tariff_units = (transporter/'./Tariefeenheden').text.to_i (transporter/'ReisType').each do |travel_type| prices = [] (travel_type/'ReisKlasse').each do |travel_class| (travel_class/'Korting/Kortingsprijs').each do |price_element| product_price = ProductPrice.new product_price.discount = price_element.attr("name") product_price.train_class = travel_class.attr("klasse") product_price.amount = price_element.attr("prijs").gsub(",", ".").to_f prices << product_price end end name = travel_type.attr('name') prices_response.products[name] = prices end end prices_response end def parse_stations(response_xml) result = [] (response_xml/'/Stations/Station').each do |station| s = Station.new s.code = (station/'./Code').text s.type = (station/'./Type').text s.country = (station/'./Land').text s.short_name = (station/'./Namen/Kort').text s.name = (station/'./Namen/Middel').text s.long_name = (station/'./Namen/Lang').text s.lat = (station/'./Lat').text s.long = (station/'./Lon').text s.uiccode = (station/'./UICCode').text result << s end result end def parse_stations_as_map(response_xml) result = {} (response_xml/'/Stations/Station').each do |station| code = (station/'./Code').text name = (station/'./Namen/Middel').text country = (station/'./Land').text result[code] = [name, country] end result end def parse_disruptions(response_xml) result = {planned: [], unplanned: []} (response_xml/'/Storingen').each do |disruption| (disruption/'Ongepland/Storing').each do |unplanned| unplanned_disruption = UnplannedDisruption.new unplanned_disruption.id = (unplanned/'./id').text unplanned_disruption.trip = (unplanned/'./Traject').text unplanned_disruption.reason = (unplanned/'./Reden').text unplanned_disruption.message = (unplanned/'./Bericht').text unplanned_disruption.datetime_string = (unplanned/'./Datum').text unplanned_disruption.cause = (unplanned/'./Oorzaak').text result[:unplanned] << unplanned_disruption end (disruption/'Gepland/Storing').each do |planned| planned_disruption = PlannedDisruption.new planned_disruption.id = (planned/'./id').text planned_disruption.trip = (planned/'./Traject').text planned_disruption.reason = (planned/'./Reden').text planned_disruption.advice = (planned/'./Advies').text planned_disruption.message = (planned/'./Bericht').text planned_disruption.cause = (planned/'./Oorzaak').text result[:planned] << planned_disruption end end result end def raise_error_when_response_is_error(xdoc) (xdoc/'/error').each do |error| message = (error/'./message').text raise InvalidStationNameError, message end end def get_xml(url) response = @client.get url @last_received_raw_xml = response.content @last_received_corrected_xml = remove_unwanted_whitespace(@last_received_raw_xml) begin Nokogiri.XML(@last_received_corrected_xml) do |config| config.options = Nokogiri::XML::ParseOptions::STRICT end rescue Nokogiri::XML::SyntaxError => e raise UnparseableXMLError.new e end end def remove_unwanted_whitespace content content.gsub /<\s*(\/?)\s*?([a-zA-Z0-9]*)\s*([a-zA-Z0-9]*)\s*>/, '<\1\2\3>' end def disruption_url(query) return "http://webservices.ns.nl/ns-api-storingen?station=#{query}" if query "http://webservices.ns.nl/ns-api-storingen?actual=true" end class PricesResponse attr_accessor :tariff_units, :products def initialize @products = {} @tariff_units = 0 end def enkele_reis products["Enkele reis"] end def dagretour products["Retour"] end end class ProductPrice attr_accessor :discount, :train_class, :amount DISCOUNT_MAP ||= { "20% korting" => "reductie_20", "40% korting" => "reductie_40" }.freeze def type DISCOUNT_MAP.fetch(discount, discount) end end class UnplannedDisruption attr_accessor :id, :trip, :reason, :message, :datetime_string, :cause end class PlannedDisruption attr_accessor :id, :trip, :reason, :advice, :message, :cause end class Station attr_accessor :code, :type, :country, :short_name, :name, :long_name, :uiccode, :synonyms, :lat, :long end class InvalidStationNameError < StandardError end class MissingParameter < StandardError end class UnparseableXMLError < StandardError end class SameDestinationError < StandardError end end