Class: Utopia::Content

Inherits:
Object
  • Object
show all
Defined in:
lib/utopia/content.rb,
lib/utopia/content/link.rb,
lib/utopia/content/node.rb,
lib/utopia/content/tags.rb,
lib/utopia/content/links.rb,
lib/utopia/content/markup.rb,
lib/utopia/content/document.rb,
lib/utopia/content/response.rb,
lib/utopia/content/namespace.rb

Overview

A middleware which serves dynamically generated content based on markup files.

Defined Under Namespace

Modules: Namespace, Tags Classes: Document, Link, Links, MarkupParser, Node, Response, SymbolicHash, UnbalancedTagError

Constant Summary collapse

INDEX =
'index'.freeze
CONTENT_NAMESPACE =
'content'.freeze
UTOPIA_NAMESPACE =
'utopia'.freeze
DEFERRED_TAG_NAME =
'utopia:deferred'.freeze
CONTENT_TAG_NAME =
'utopia:content'.freeze
XNODE_EXTENSION =

The file extension for markup nodes on disk.

'.xnode'.freeze
Tag =
Trenni::Tag
EXPIRES =

Compatibility with older versions of rack:

'Expires'.freeze
CACHE_CONTROL =
'Cache-Control'.freeze
CONTENT_TYPE =
'Content-Type'.freeze
NO_CACHE =
'no-cache'.freeze

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(app, root: Utopia::default_root, namespaces: {}) ⇒ Content

Returns a new instance of Content.

Parameters:

  • root (String) (defaults to: Utopia::default_root)

    The content root where pages will be generated from.

  • namespaces (Hash<String,Library>) (defaults to: {})

    Tag namespaces for dynamic tag lookup.



44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
# File 'lib/utopia/content.rb', line 44

def initialize(app, root: Utopia::default_root, namespaces: {})
	@app = app
	@root = root
	
	@template_cache = Concurrent::Map.new
	@node_cache = Concurrent::Map.new
	
	@namespaces = namespaces
	
	# Default content namespace for dynamic path based lookup:
	@namespaces[CONTENT_NAMESPACE] ||= self.method(:content_tag)
	
	# The core namespace for utopia specific functionality:
	@namespaces[UTOPIA_NAMESPACE] ||= Tags
end

Instance Attribute Details

#rootObject (readonly)

Returns the value of attribute root.



70
71
72
# File 'lib/utopia/content.rb', line 70

def root
  @root
end

Instance Method Details

#call(env) ⇒ Object



101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
# File 'lib/utopia/content.rb', line 101

def call(env)
	request = Rack::Request.new(env)
	path = Path.create(request.path_info)
	
	# Check if the request is to a non-specific index. This only works for requests with a given name:
	basename = path.basename
	directory_path = File.join(@root, path.dirname.components, basename)

	# If the request for /foo/bar is actually a directory, rewrite it to /foo/bar/index:
	if File.directory? directory_path
		index_path = [basename, INDEX]
		
		return [307, {HTTP::LOCATION => path.dirname.join(index_path).to_s}, []]
	end

	locale = env[Localization::CURRENT_LOCALE_KEY]
	if link = Links.for(@root, path, locale)
		if link.path and node = lookup_node(link.path)
			attributes = request.env.fetch(VARIABLES_KEY, {}).to_hash
			
			return node.process!(request, attributes)
		elsif redirect_uri = link[:uri]
			return [307, {HTTP::LOCATION => redirect_uri}, []]
		end
	end
	
	return @app.call(env)
end

#fetch_template(path) ⇒ Object



72
73
74
75
76
# File 'lib/utopia/content.rb', line 72

def fetch_template(path)
	@template_cache.fetch_or_store(path.to_s) do
		Trenni::MarkupTemplate.load_file(path)
	end
end

#freezeObject



60
61
62
63
64
65
66
67
68
# File 'lib/utopia/content.rb', line 60

def freeze
	return self if frozen?
	
	@root.freeze
	@namespaces.values.each(&:freeze)
	@namespaces.freeze
	
	super
end

#lookup_node(request_path) ⇒ Object

The request_path is an absolute uri path, e.g. /foo/bar. If an xnode file exists on disk for this exact path, it is instantiated, otherwise nil.



88
89
90
91
92
93
94
95
96
97
98
99
# File 'lib/utopia/content.rb', line 88

def lookup_node(request_path)
	name = request_path.last
	name_xnode = name.to_s + XNODE_EXTENSION

	node_path = File.join(@root, request_path.dirname.components, name_xnode)

	if File.exist? node_path
		return Node.new(self, request_path.dirname + name, request_path, node_path)
	end

	return nil
end

#lookup_tag(qualified_name, node) ⇒ Object

Look up a named tag such as <entry /> or <content:page>...



79
80
81
82
83
84
85
# File 'lib/utopia/content.rb', line 79

def lookup_tag(qualified_name, node)
	namespace, name = Trenni::Tag.split(qualified_name)
	
	if library = @namespaces[namespace]
		library.call(name, node)
	end
end