pakyow-core 0.11.3 → 1.0.0.rc1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +5 -5
- data/CHANGELOG.md +0 -0
- data/LICENSE +4 -0
- data/{pakyow-core/README.md → README.md} +3 -4
- data/commands/pakyow +13 -0
- data/lib/pakyow/app/connection/behavior/session.rb +33 -0
- data/lib/pakyow/app/connection/behavior/values.rb +42 -0
- data/lib/pakyow/app/connection/behavior/verifier.rb +34 -0
- data/lib/pakyow/app/connection/session/abstract.rb +24 -0
- data/lib/pakyow/app/connection/session/cookie.rb +56 -0
- data/lib/pakyow/app/connection.rb +57 -0
- data/lib/pakyow/app.rb +219 -0
- data/lib/pakyow/behavior/aspects.rb +45 -0
- data/lib/pakyow/behavior/config.rb +72 -0
- data/lib/pakyow/behavior/endpoints.rb +41 -0
- data/lib/pakyow/behavior/frameworks.rb +45 -0
- data/lib/pakyow/behavior/helpers.rb +107 -0
- data/lib/pakyow/behavior/initializers.rb +19 -0
- data/lib/pakyow/behavior/isolating.rb +69 -0
- data/lib/pakyow/behavior/operations.rb +46 -0
- data/lib/pakyow/behavior/pipeline.rb +58 -0
- data/lib/pakyow/behavior/plugins.rb +148 -0
- data/lib/pakyow/behavior/rescuing.rb +51 -0
- data/lib/pakyow/behavior/restarting.rb +68 -0
- data/lib/pakyow/behavior/sessions.rb +34 -0
- data/lib/pakyow/behavior/verification.rb +47 -0
- data/lib/pakyow/cli.rb +199 -0
- data/lib/pakyow/connection/multipart_input.rb +24 -0
- data/lib/pakyow/connection/multipart_parser.rb +112 -0
- data/lib/pakyow/connection/params.rb +29 -0
- data/lib/pakyow/connection/query_parser.rb +174 -0
- data/lib/pakyow/connection/statuses.rb +119 -0
- data/lib/pakyow/connection.rb +568 -0
- data/lib/pakyow/endpoints.rb +94 -0
- data/lib/pakyow/environment/actions/dispatch.rb +37 -0
- data/lib/pakyow/environment/actions/input_parser.rb +21 -0
- data/lib/pakyow/environment/actions/logger.rb +33 -0
- data/lib/pakyow/environment/actions/normalizer.rb +73 -0
- data/lib/pakyow/environment/behavior/config.rb +166 -0
- data/lib/pakyow/environment/behavior/initializers.rb +21 -0
- data/lib/pakyow/environment/behavior/input_parsing.rb +54 -0
- data/lib/pakyow/environment/behavior/plugins.rb +23 -0
- data/lib/pakyow/environment/behavior/restarting.rb +54 -0
- data/lib/pakyow/environment/behavior/running.rb +129 -0
- data/lib/pakyow/environment/behavior/silencing.rb +23 -0
- data/lib/pakyow/environment/behavior/timezone.rb +21 -0
- data/lib/pakyow/environment/behavior/watching.rb +75 -0
- data/lib/pakyow/environment.rb +381 -0
- data/lib/pakyow/error.rb +273 -0
- data/lib/pakyow/errors.rb +97 -0
- data/lib/pakyow/framework.rb +39 -0
- data/lib/pakyow/generator.rb +138 -0
- data/lib/pakyow/generators/project/default/%dot%env.erb +1 -0
- data/lib/pakyow/generators/project/default/%dot%gitignore +11 -0
- data/lib/pakyow/generators/project/default/%dot%ruby-version.erb +1 -0
- data/lib/pakyow/generators/project/default/Gemfile.erb +28 -0
- data/lib/pakyow/generators/project/default/README.md +14 -0
- data/lib/pakyow/generators/project/default/config/application.rb.erb +17 -0
- data/lib/pakyow/generators/project/default/config/environment.rb +11 -0
- data/lib/pakyow/generators/project/default/config/initializers/application/keep +0 -0
- data/lib/pakyow/generators/project/default/config/initializers/environment/keep +0 -0
- data/lib/pakyow/generators/project/default/database/keep +0 -0
- data/lib/pakyow/generators/project/default/frontend/assets/images/keep +0 -0
- data/lib/pakyow/generators/project/default/frontend/assets/packs/keep +0 -0
- data/lib/pakyow/generators/project/default/frontend/assets/styles/keep +0 -0
- data/lib/pakyow/generators/project/default/frontend/includes/keep +0 -0
- data/lib/pakyow/generators/project/default/frontend/layouts/default.html.erb +19 -0
- data/lib/pakyow/generators/project/default/frontend/pages/keep +0 -0
- data/lib/pakyow/generators/project/default/public/favicon.ico +0 -0
- data/lib/pakyow/generators/project/default/public/robots.txt +1 -0
- data/lib/pakyow/generators/project.rb +29 -0
- data/lib/pakyow/helpers/app.rb +28 -0
- data/lib/pakyow/helpers/connection.rb +45 -0
- data/lib/pakyow/info.rb +26 -0
- data/lib/pakyow/integrations/bootsnap.rb +18 -0
- data/lib/pakyow/integrations/bundler/require.rb +7 -0
- data/lib/pakyow/integrations/bundler/setup.rb +7 -0
- data/lib/pakyow/integrations/dotenv.rb +9 -0
- data/lib/pakyow/loader.rb +31 -0
- data/lib/pakyow/logger/colorizer.rb +38 -0
- data/lib/pakyow/logger/destination.rb +17 -0
- data/lib/pakyow/logger/formatter.rb +22 -0
- data/lib/pakyow/logger/formatters/human.rb +101 -0
- data/lib/pakyow/logger/formatters/json.rb +99 -0
- data/lib/pakyow/logger/formatters/logfmt.rb +50 -0
- data/lib/pakyow/logger/multiplexed.rb +19 -0
- data/lib/pakyow/logger/thread_local.rb +29 -0
- data/lib/pakyow/logger/timekeeper.rb +64 -0
- data/lib/pakyow/logger.rb +125 -0
- data/lib/pakyow/operation.rb +71 -0
- data/lib/pakyow/plugin/helper_caller.rb +20 -0
- data/lib/pakyow/plugin/lookup.rb +33 -0
- data/lib/pakyow/plugin/state.rb +68 -0
- data/lib/pakyow/plugin.rb +367 -0
- data/lib/pakyow/process_manager.rb +53 -0
- data/lib/pakyow/processes/proxy.rb +67 -0
- data/lib/pakyow/processes/server.rb +32 -0
- data/lib/pakyow/rack/compatibility.rb +68 -0
- data/lib/pakyow/task.rb +315 -0
- data/lib/pakyow/tasks/boot.rake +21 -0
- data/lib/pakyow/tasks/create.rake +32 -0
- data/lib/pakyow/tasks/help.rake +17 -0
- data/lib/pakyow/tasks/info/endpoints.rake +72 -0
- data/lib/pakyow/tasks/info.rake +31 -0
- data/lib/pakyow/tasks/irb.rake +11 -0
- data/lib/pakyow/tasks/prelaunch.rake +37 -0
- data/lib/pakyow/types.rb +32 -0
- data/lib/pakyow/validations/acceptance.rb +37 -0
- data/lib/pakyow/validations/email.rb +28 -0
- data/lib/pakyow/validations/inline.rb +32 -0
- data/lib/pakyow/validations/length.rb +44 -0
- data/lib/pakyow/validations/presence.rb +41 -0
- data/lib/pakyow/validations.rb +8 -0
- data/lib/pakyow/validator.rb +81 -0
- data/lib/pakyow/verifier.rb +177 -0
- data/lib/pakyow.rb +5 -0
- metadata +267 -65
- data/pakyow-core/CHANGELOG.md +0 -128
- data/pakyow-core/LICENSE +0 -20
- data/pakyow-core/lib/pakyow/core/app.rb +0 -82
- data/pakyow-core/lib/pakyow/core/app_context.rb +0 -10
- data/pakyow-core/lib/pakyow/core/base.rb +0 -61
- data/pakyow-core/lib/pakyow/core/call_context.rb +0 -171
- data/pakyow-core/lib/pakyow/core/config/app.rb +0 -49
- data/pakyow-core/lib/pakyow/core/config/cookies.rb +0 -4
- data/pakyow-core/lib/pakyow/core/config/logger.rb +0 -34
- data/pakyow-core/lib/pakyow/core/config/reloader.rb +0 -10
- data/pakyow-core/lib/pakyow/core/config/server.rb +0 -10
- data/pakyow-core/lib/pakyow/core/config/session.rb +0 -41
- data/pakyow-core/lib/pakyow/core/config.rb +0 -95
- data/pakyow-core/lib/pakyow/core/errors.rb +0 -16
- data/pakyow-core/lib/pakyow/core/helpers/configuring.rb +0 -142
- data/pakyow-core/lib/pakyow/core/helpers/hooks.rb +0 -106
- data/pakyow-core/lib/pakyow/core/helpers/running.rb +0 -124
- data/pakyow-core/lib/pakyow/core/helpers.rb +0 -61
- data/pakyow-core/lib/pakyow/core/loader.rb +0 -32
- data/pakyow-core/lib/pakyow/core/middleware/logger.rb +0 -146
- data/pakyow-core/lib/pakyow/core/middleware/override.rb +0 -3
- data/pakyow-core/lib/pakyow/core/middleware/reloader.rb +0 -23
- data/pakyow-core/lib/pakyow/core/middleware/req_path_normalizer.rb +0 -49
- data/pakyow-core/lib/pakyow/core/middleware/session.rb +0 -5
- data/pakyow-core/lib/pakyow/core/middleware/static.rb +0 -76
- data/pakyow-core/lib/pakyow/core/multilog.rb +0 -19
- data/pakyow-core/lib/pakyow/core/request.rb +0 -119
- data/pakyow-core/lib/pakyow/core/response.rb +0 -135
- data/pakyow-core/lib/pakyow/core/route_eval.rb +0 -254
- data/pakyow-core/lib/pakyow/core/route_expansion_eval.rb +0 -115
- data/pakyow-core/lib/pakyow/core/route_lookup.rb +0 -41
- data/pakyow-core/lib/pakyow/core/route_merger.rb +0 -76
- data/pakyow-core/lib/pakyow/core/route_module.rb +0 -21
- data/pakyow-core/lib/pakyow/core/route_set.rb +0 -74
- data/pakyow-core/lib/pakyow/core/route_template_defaults.rb +0 -43
- data/pakyow-core/lib/pakyow/core/route_template_eval.rb +0 -72
- data/pakyow-core/lib/pakyow/core/router.rb +0 -181
- data/pakyow-core/lib/pakyow/core.rb +0 -8
- data/pakyow-core/lib/pakyow-core.rb +0 -1
@@ -0,0 +1,45 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/extension"
|
4
|
+
|
5
|
+
module Pakyow
|
6
|
+
module Behavior
|
7
|
+
# Manages {Framework} instances.
|
8
|
+
#
|
9
|
+
module Frameworks
|
10
|
+
extend Support::Extension
|
11
|
+
|
12
|
+
apply_extension do
|
13
|
+
setting :loaded_frameworks, []
|
14
|
+
end
|
15
|
+
|
16
|
+
class_methods do
|
17
|
+
# Includes one or more frameworks into the app class.
|
18
|
+
#
|
19
|
+
def include_frameworks(*frameworks)
|
20
|
+
tap do
|
21
|
+
frameworks.each do |framework_name|
|
22
|
+
include_framework(framework_name)
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
# Includes a framework into the app class.
|
28
|
+
#
|
29
|
+
def include_framework(framework_name)
|
30
|
+
framework_name = framework_name.to_sym
|
31
|
+
Pakyow.frameworks.fetch(framework_name).new(self).boot
|
32
|
+
(config.loaded_frameworks << framework_name).uniq!
|
33
|
+
rescue KeyError => error
|
34
|
+
raise UnknownFramework.build(error, framework: framework_name)
|
35
|
+
end
|
36
|
+
|
37
|
+
# Returns true if +framework+ is loaded.
|
38
|
+
#
|
39
|
+
def includes_framework?(framework_name)
|
40
|
+
config.loaded_frameworks.include?(framework_name.to_sym)
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
@@ -0,0 +1,107 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/extension"
|
4
|
+
require "pakyow/support/makeable"
|
5
|
+
require "pakyow/support/safe_string"
|
6
|
+
|
7
|
+
require "pakyow/helpers/app"
|
8
|
+
require "pakyow/helpers/connection"
|
9
|
+
|
10
|
+
module Pakyow
|
11
|
+
module Helper
|
12
|
+
extend Support::Makeable
|
13
|
+
end
|
14
|
+
|
15
|
+
module Behavior
|
16
|
+
# Maintains a list of helper modules, with code for including helpers of a type into an object.
|
17
|
+
#
|
18
|
+
# Helpers are either global, passive, or active. Global helpers contain utility methods or
|
19
|
+
# methods that need application-level state. Passive helpers can access state on the connection
|
20
|
+
# but never change connection state, which active helpers are solely responsible for doing.
|
21
|
+
#
|
22
|
+
module Helpers
|
23
|
+
extend Support::Extension
|
24
|
+
|
25
|
+
apply_extension do
|
26
|
+
class_state :__included_helpers, default: {}, inheritable: true
|
27
|
+
|
28
|
+
setting :helpers,
|
29
|
+
global: [
|
30
|
+
Pakyow::Helpers::App,
|
31
|
+
Support::SafeStringHelpers
|
32
|
+
],
|
33
|
+
|
34
|
+
passive: [
|
35
|
+
Pakyow::Helpers::Connection
|
36
|
+
],
|
37
|
+
|
38
|
+
active: []
|
39
|
+
|
40
|
+
on "load" do
|
41
|
+
# Helpers are loaded first so that other aspects inherit them.
|
42
|
+
#
|
43
|
+
load_aspect(:helpers)
|
44
|
+
|
45
|
+
self.class.state(:helper).each do |helper|
|
46
|
+
context = if helper.instance_variable_defined?(:@type)
|
47
|
+
helper.instance_variable_get(:@type)
|
48
|
+
else
|
49
|
+
:global
|
50
|
+
end
|
51
|
+
|
52
|
+
self.class.register_helper(context, helper)
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
# Define helpers as stateful after an app is made.
|
57
|
+
#
|
58
|
+
after "make", priority: :high do
|
59
|
+
stateful :helper, Helper
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
class_methods do
|
64
|
+
# Registers a helper module to be loaded on defined endpoints.
|
65
|
+
#
|
66
|
+
def register_helper(context, helper_module)
|
67
|
+
(config.helpers[context.to_sym] << helper_module).uniq!
|
68
|
+
end
|
69
|
+
|
70
|
+
# Includes helpers of a particular context into an object. Global helpers
|
71
|
+
# will automatically be included into active and passive contexts, and
|
72
|
+
# passive helpers will automatically be included into the active context.
|
73
|
+
#
|
74
|
+
def include_helpers(context, object)
|
75
|
+
@__included_helpers[object] = context
|
76
|
+
|
77
|
+
helpers(context.to_sym).each do |helper|
|
78
|
+
object.include helper
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
def included_helper_context(object)
|
83
|
+
@__included_helpers.each_pair do |object_with_helpers, context|
|
84
|
+
return context if object.is_a?(object_with_helpers)
|
85
|
+
end
|
86
|
+
|
87
|
+
nil
|
88
|
+
end
|
89
|
+
|
90
|
+
def helpers(context)
|
91
|
+
case context.to_sym
|
92
|
+
when :global
|
93
|
+
config.helpers[:global]
|
94
|
+
when :passive
|
95
|
+
config.helpers[:global] + config.helpers[:passive]
|
96
|
+
when :active
|
97
|
+
config.helpers[:global] + config.helpers[:passive] + config.helpers[:active]
|
98
|
+
else
|
99
|
+
raise UnknownHelperContext.new_with_message(
|
100
|
+
context: context
|
101
|
+
)
|
102
|
+
end
|
103
|
+
end
|
104
|
+
end
|
105
|
+
end
|
106
|
+
end
|
107
|
+
end
|
@@ -0,0 +1,19 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/extension"
|
4
|
+
|
5
|
+
module Pakyow
|
6
|
+
module Behavior
|
7
|
+
module Initializers
|
8
|
+
extend Support::Extension
|
9
|
+
|
10
|
+
apply_extension do
|
11
|
+
after "make" do
|
12
|
+
Dir.glob(File.join(config.root, "config/initializers/application/**/*.rb")).each do |initializer|
|
13
|
+
class_eval(File.read(initializer), initializer)
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
@@ -0,0 +1,69 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/extension"
|
4
|
+
require "pakyow/support/inflector"
|
5
|
+
|
6
|
+
module Pakyow
|
7
|
+
module Behavior
|
8
|
+
# Helps manage isolated classes for an app.
|
9
|
+
#
|
10
|
+
module Isolating
|
11
|
+
extend Support::Extension
|
12
|
+
|
13
|
+
class_methods do
|
14
|
+
# Creates a subclass within the app's namespace.
|
15
|
+
#
|
16
|
+
# @example
|
17
|
+
# class MyApp < Pakyow::App
|
18
|
+
# isolate Pakyow::Controller do
|
19
|
+
# def self.special_behavior
|
20
|
+
# puts "it works"
|
21
|
+
# end
|
22
|
+
# end
|
23
|
+
# end
|
24
|
+
#
|
25
|
+
# MyApp::Controller.special_behavior
|
26
|
+
# => it works
|
27
|
+
#
|
28
|
+
# Pakyow::Controller
|
29
|
+
# => NoMethodError (undefined method `special_behavior' for Pakyow::Controller:Class)
|
30
|
+
#
|
31
|
+
def isolate(class_to_isolate, &block)
|
32
|
+
isolated_class_name = Support.inflector.demodulize(class_to_isolate.to_s).to_sym
|
33
|
+
|
34
|
+
unless const_defined?(isolated_class_name, false)
|
35
|
+
const_set(isolated_class_name, Class.new(class_to_isolate))
|
36
|
+
end
|
37
|
+
|
38
|
+
isolated(isolated_class_name).tap do |defined_subclass|
|
39
|
+
defined_subclass.class_eval(&block) if block_given?
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
# Returns true if the class name is isolated.
|
44
|
+
#
|
45
|
+
def isolated?(class_name)
|
46
|
+
const_defined?(class_name)
|
47
|
+
end
|
48
|
+
|
49
|
+
# Returns the isolated class, evaluating the block (if provided).
|
50
|
+
#
|
51
|
+
def isolated(class_name, &block)
|
52
|
+
if isolated?(class_name)
|
53
|
+
const_get(class_name).tap do |isolated_class|
|
54
|
+
isolated_class.class_eval(&block) if block_given?
|
55
|
+
end
|
56
|
+
else
|
57
|
+
nil
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
# Convenience method for +Pakyow::App::subclass+.
|
63
|
+
#
|
64
|
+
def isolated(*args, &block)
|
65
|
+
self.class.isolated(*args, &block)
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|
@@ -0,0 +1,46 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/extension"
|
4
|
+
|
5
|
+
require "pakyow/operation"
|
6
|
+
|
7
|
+
module Pakyow
|
8
|
+
module Behavior
|
9
|
+
# Adds support for operations.
|
10
|
+
#
|
11
|
+
module Operations
|
12
|
+
class Lookup
|
13
|
+
def initialize(operations:, app:)
|
14
|
+
operations.each do |operation|
|
15
|
+
define_singleton_method operation.__object_name.name do |values = {}, &block|
|
16
|
+
(block ? Class.new(operation, &block) : operation).new(app: app, **values).perform
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
extend Support::Extension
|
23
|
+
|
24
|
+
apply_extension do
|
25
|
+
on "load" do
|
26
|
+
load_aspect(:operations)
|
27
|
+
end
|
28
|
+
|
29
|
+
attr_reader :operations
|
30
|
+
after "initialize" do
|
31
|
+
@operations = Lookup.new(operations: state(:operation), app: self)
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
class_methods do
|
36
|
+
# Define operations as stateful when an app is defined.
|
37
|
+
#
|
38
|
+
def make(*)
|
39
|
+
super.tap do |new_class|
|
40
|
+
new_class.stateful :operation, Operation
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
46
|
+
end
|
@@ -0,0 +1,58 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/extension"
|
4
|
+
|
5
|
+
module Pakyow
|
6
|
+
module Behavior
|
7
|
+
# Loads default pipeline actions based on included frameworks.
|
8
|
+
#
|
9
|
+
module Pipeline
|
10
|
+
extend Support::Extension
|
11
|
+
|
12
|
+
apply_extension do
|
13
|
+
# We set the priority very low here in case initialize hooks in other frameworks define
|
14
|
+
# state that should be loaded into the pipeline (e.g. controllers).
|
15
|
+
#
|
16
|
+
after "initialize", "initialize.pipeline", priority: -10 do
|
17
|
+
self.class.__pipeline.dup.tap do |pipeline|
|
18
|
+
load_pipeline_defaults(pipeline)
|
19
|
+
@__pipeline = pipeline.callable(self)
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
private
|
25
|
+
|
26
|
+
def load_pipeline_defaults(pipeline)
|
27
|
+
if self.class.includes_framework?(:assets)
|
28
|
+
pipeline.action(Assets::Actions::Public, self)
|
29
|
+
pipeline.action(Assets::Actions::Process)
|
30
|
+
end
|
31
|
+
|
32
|
+
if self.class.includes_framework?(:realtime) && Pakyow.config.realtime.server && !is_a?(Plugin)
|
33
|
+
pipeline.action(Realtime::Actions::Upgrader)
|
34
|
+
end
|
35
|
+
|
36
|
+
if self.class.includes_framework?(:routing) && !Pakyow.env?(:prototype)
|
37
|
+
state(:controller).each do |controller|
|
38
|
+
pipeline.action(controller, self)
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
if instance_variable_defined?(:@plugs)
|
43
|
+
@plugs.each do |plug_instance|
|
44
|
+
pipeline.action(plug_instance)
|
45
|
+
end
|
46
|
+
end
|
47
|
+
|
48
|
+
if self.class.includes_framework?(:presenter)
|
49
|
+
pipeline.action(Presenter::Actions::AutoRender)
|
50
|
+
end
|
51
|
+
|
52
|
+
if self.class.includes_framework?(:routing) && !Pakyow.env?(:prototype) && !is_a?(Plugin)
|
53
|
+
pipeline.action(Routing::Actions::RespondMissing)
|
54
|
+
end
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
@@ -0,0 +1,148 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/errors"
|
4
|
+
|
5
|
+
require "pakyow/support/deep_dup"
|
6
|
+
require "pakyow/support/extension"
|
7
|
+
|
8
|
+
require "pakyow/plugin/lookup"
|
9
|
+
|
10
|
+
module Pakyow
|
11
|
+
module Behavior
|
12
|
+
module Plugins
|
13
|
+
extend Support::Extension
|
14
|
+
|
15
|
+
using Support::DeepDup
|
16
|
+
|
17
|
+
attr_reader :plugs
|
18
|
+
|
19
|
+
def plug(name, instance = :default)
|
20
|
+
@plugs.find { |plug|
|
21
|
+
plug.class.plugin_name == name && plug.class.__object_name.namespace.parts.last == instance
|
22
|
+
}
|
23
|
+
end
|
24
|
+
|
25
|
+
apply_extension do
|
26
|
+
class_state :__plugs, default: [], inheritable: true
|
27
|
+
|
28
|
+
# Create a dynamic helper that allows plugin helpers to be called in context of a specific plug.
|
29
|
+
#
|
30
|
+
on "initialize" do
|
31
|
+
self.class.register_helper :passive, Module.new {
|
32
|
+
Pakyow.plugins.keys.map.each do |plugin_name|
|
33
|
+
define_method plugin_name do |plug = :default|
|
34
|
+
app.plugs.send(plugin_name, plug).helper_caller(
|
35
|
+
app.class.included_helper_context(self),
|
36
|
+
@connection,
|
37
|
+
self
|
38
|
+
)
|
39
|
+
end
|
40
|
+
end
|
41
|
+
}
|
42
|
+
end
|
43
|
+
|
44
|
+
# Setting priority to low gives the app a chance to do any pre-loading
|
45
|
+
# that might affect how plugins are setup.
|
46
|
+
#
|
47
|
+
after "initialize", "load.plugins", priority: :low do
|
48
|
+
@plugs = Plugin::Lookup.new([])
|
49
|
+
|
50
|
+
self.class.__plugs.each do |plug|
|
51
|
+
if self.class.includes_framework?(:presenter)
|
52
|
+
require "pakyow/plugin/helpers/rendering"
|
53
|
+
plug.register_helper :passive, Plugin::Helpers::Rendering
|
54
|
+
end
|
55
|
+
|
56
|
+
# Include frameworks from app.
|
57
|
+
#
|
58
|
+
plug.include_frameworks(
|
59
|
+
*self.class.config.loaded_frameworks
|
60
|
+
)
|
61
|
+
|
62
|
+
# Copy settings from the app config.
|
63
|
+
#
|
64
|
+
plug.config.instance_variable_set(
|
65
|
+
:@__settings, config.__settings.deep_dup.merge(plug.config.__settings)
|
66
|
+
)
|
67
|
+
|
68
|
+
# Copy defaults from the app config.
|
69
|
+
#
|
70
|
+
plug.config.instance_variable_set(
|
71
|
+
:@__defaults, config.__defaults.deep_dup.merge(plug.config.__defaults)
|
72
|
+
)
|
73
|
+
|
74
|
+
# Copy groups from the app config.
|
75
|
+
#
|
76
|
+
plug.config.instance_variable_set(
|
77
|
+
:@__groups, config.__groups.deep_dup.merge(plug.config.__groups)
|
78
|
+
)
|
79
|
+
|
80
|
+
# Override config values that require a specific value.
|
81
|
+
#
|
82
|
+
full_name = [plug.plugin_name]
|
83
|
+
unless plug.__object_name.namespace.parts.last == :default
|
84
|
+
full_name << plug.__object_name.namespace.parts.last
|
85
|
+
end
|
86
|
+
|
87
|
+
plug.config.name = full_name.join("_").to_sym
|
88
|
+
|
89
|
+
# Create a dynamic helper that allows plugin helpers to be called in context of a specific plug.
|
90
|
+
#
|
91
|
+
plug.register_helper :passive, Module.new {
|
92
|
+
Pakyow.plugins.keys.map.each do |plugin_name|
|
93
|
+
define_method plugin_name do |instance_name = :default|
|
94
|
+
app.parent.plugs.send(plugin_name, instance_name).helper_caller(
|
95
|
+
app.class.included_helper_context(self),
|
96
|
+
@connection,
|
97
|
+
self
|
98
|
+
)
|
99
|
+
end
|
100
|
+
end
|
101
|
+
}
|
102
|
+
|
103
|
+
# Finally, create the plugin instance.
|
104
|
+
#
|
105
|
+
@plugs << plug.new(self)
|
106
|
+
end
|
107
|
+
|
108
|
+
@plugs.finalize
|
109
|
+
end
|
110
|
+
|
111
|
+
after "boot" do
|
112
|
+
@plugs.each(&:booted)
|
113
|
+
end
|
114
|
+
end
|
115
|
+
|
116
|
+
class_methods do
|
117
|
+
attr_reader :__plugs
|
118
|
+
|
119
|
+
def plug(plugin_name, at: "/", as: :default, &block)
|
120
|
+
plugin_name = plugin_name.to_sym
|
121
|
+
|
122
|
+
unless plugin = Pakyow.plugins[plugin_name]
|
123
|
+
raise UnknownPlugin.new_with_message(
|
124
|
+
plugin: plugin_name
|
125
|
+
)
|
126
|
+
end
|
127
|
+
|
128
|
+
app = self
|
129
|
+
plug = plugin.make(
|
130
|
+
"plug",
|
131
|
+
within: Support::ObjectNamespace.new(
|
132
|
+
*__object_name.namespace.parts + [plugin_name, as]
|
133
|
+
),
|
134
|
+
mount_path: at,
|
135
|
+
) do
|
136
|
+
# Creates a connection subclass that other frameworks can safely extend.
|
137
|
+
#
|
138
|
+
isolate(app.isolated(:Connection))
|
139
|
+
|
140
|
+
instance_exec(&block) if block
|
141
|
+
end
|
142
|
+
|
143
|
+
@__plugs << plug
|
144
|
+
end
|
145
|
+
end
|
146
|
+
end
|
147
|
+
end
|
148
|
+
end
|
@@ -0,0 +1,51 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/extension"
|
4
|
+
|
5
|
+
module Pakyow
|
6
|
+
module Behavior
|
7
|
+
# Lets an app to be rescued from errors encountered during boot. Once rescued,
|
8
|
+
# an app returns a 500 response with the error that caused it to fail.
|
9
|
+
#
|
10
|
+
module Rescuing
|
11
|
+
extend Support::Extension
|
12
|
+
|
13
|
+
# Error the app was rescued from.
|
14
|
+
#
|
15
|
+
attr_reader :rescued
|
16
|
+
|
17
|
+
# Returns true if the app has been rescued.
|
18
|
+
#
|
19
|
+
def rescued?
|
20
|
+
instance_variable_defined?(:@rescued) && !!@rescued
|
21
|
+
end
|
22
|
+
|
23
|
+
private
|
24
|
+
|
25
|
+
# Enters rescue mode after logging the error.
|
26
|
+
#
|
27
|
+
def rescue!(error)
|
28
|
+
@rescued = error
|
29
|
+
|
30
|
+
performing :rescue do
|
31
|
+
Pakyow.logger.error(error)
|
32
|
+
|
33
|
+
message = <<~ERROR
|
34
|
+
#{self.class} failed to initialize.
|
35
|
+
|
36
|
+
#{error.message}
|
37
|
+
#{error.backtrace.join("\n")}
|
38
|
+
ERROR
|
39
|
+
|
40
|
+
# Override call to always return an errored response.
|
41
|
+
#
|
42
|
+
define_singleton_method :call do |connection|
|
43
|
+
connection.status = 500
|
44
|
+
connection.body = StringIO.new(message)
|
45
|
+
connection.halt
|
46
|
+
end
|
47
|
+
end
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
@@ -0,0 +1,68 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "fileutils"
|
4
|
+
|
5
|
+
require "pakyow/support/silenceable"
|
6
|
+
Pakyow::Support::Silenceable.silence_warnings do
|
7
|
+
require "filewatcher"
|
8
|
+
end
|
9
|
+
|
10
|
+
require "pakyow/support/extension"
|
11
|
+
|
12
|
+
module Pakyow
|
13
|
+
module Behavior
|
14
|
+
# Handles triggering restarts in the parent process.
|
15
|
+
#
|
16
|
+
module Restarting
|
17
|
+
extend Support::Extension
|
18
|
+
|
19
|
+
apply_extension do
|
20
|
+
configurable :process do
|
21
|
+
setting :watched_paths, []
|
22
|
+
setting :excluded_paths, []
|
23
|
+
setting :restartable, false
|
24
|
+
|
25
|
+
defaults :development do
|
26
|
+
setting :restartable, true
|
27
|
+
end
|
28
|
+
|
29
|
+
defaults :prototype do
|
30
|
+
setting :restartable, true
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
after "initialize" do
|
35
|
+
setup_for_restarting
|
36
|
+
end
|
37
|
+
|
38
|
+
# Setting up for restarting even after the app fails to initialize lets
|
39
|
+
# the developer fix the problem and let the server restart on its own.
|
40
|
+
#
|
41
|
+
after "rescue" do
|
42
|
+
setup_for_restarting
|
43
|
+
end
|
44
|
+
end
|
45
|
+
|
46
|
+
def setup_for_restarting
|
47
|
+
if config.process.restartable
|
48
|
+
config.process.watched_paths << File.join(config.src, "**/*.rb")
|
49
|
+
config.process.watched_paths << File.join(config.lib, "**/*.rb")
|
50
|
+
|
51
|
+
# FIXME: this doesn't need to be hardcoded, but instead determined
|
52
|
+
# from the source location when registered with the environment
|
53
|
+
config.process.watched_paths << "./config/application.rb"
|
54
|
+
|
55
|
+
Thread.new do
|
56
|
+
Filewatcher.new(
|
57
|
+
config.process.watched_paths,
|
58
|
+
exclude: config.process.excluded_paths
|
59
|
+
).watch do |_path, _event|
|
60
|
+
FileUtils.mkdir_p "./tmp"
|
61
|
+
FileUtils.touch "./tmp/restart.txt"
|
62
|
+
end
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
@@ -0,0 +1,34 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/extension"
|
4
|
+
require "pakyow/support/inflector"
|
5
|
+
|
6
|
+
module Pakyow
|
7
|
+
module Behavior
|
8
|
+
module Sessions
|
9
|
+
extend Support::Extension
|
10
|
+
|
11
|
+
apply_extension do
|
12
|
+
attr_reader :session_object, :session_options
|
13
|
+
|
14
|
+
after "configure" do
|
15
|
+
if config.session.enabled
|
16
|
+
require "pakyow/app/connection/session/#{config.session.object}"
|
17
|
+
|
18
|
+
@session_object = Pakyow::App::Connection::Session.const_get(
|
19
|
+
Support.inflector.classify(config.session.object)
|
20
|
+
)
|
21
|
+
|
22
|
+
@session_options = if config.session.respond_to?(config.session.object)
|
23
|
+
config.session.public_send(config.session.object)
|
24
|
+
else
|
25
|
+
{}
|
26
|
+
end
|
27
|
+
end
|
28
|
+
rescue LoadError #=> error
|
29
|
+
# TODO: raise an unknown session object error...
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
@@ -0,0 +1,47 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "pakyow/support/class_state"
|
4
|
+
require "pakyow/support/deep_dup"
|
5
|
+
require "pakyow/support/extension"
|
6
|
+
|
7
|
+
require "pakyow/errors"
|
8
|
+
require "pakyow/verifier"
|
9
|
+
|
10
|
+
module Pakyow
|
11
|
+
module Behavior
|
12
|
+
module Verification
|
13
|
+
extend Support::Extension
|
14
|
+
using Support::DeepDup
|
15
|
+
|
16
|
+
def verify(values = nil, &block)
|
17
|
+
unless values
|
18
|
+
if self.class.__object_name_to_verify.nil?
|
19
|
+
raise "Expected values to be passed"
|
20
|
+
else
|
21
|
+
values = public_send(self.class.__object_name_to_verify)
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
original_values = values.deep_dup
|
26
|
+
result = Verifier.new(&block).call(values, context: self)
|
27
|
+
|
28
|
+
unless result.verified?
|
29
|
+
error = InvalidData.new_with_message(:verification)
|
30
|
+
error.context = { object: original_values, result: result }
|
31
|
+
raise error
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
apply_extension do
|
36
|
+
extend Support::ClassState
|
37
|
+
class_state :__object_name_to_verify, inheritable: true
|
38
|
+
end
|
39
|
+
|
40
|
+
class_methods do
|
41
|
+
def verifies(object)
|
42
|
+
@__object_name_to_verify = object
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
46
|
+
end
|
47
|
+
end
|