forked from cloudfoundry/java-buildpack
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathapp_dynamics_agent.rb
More file actions
198 lines (164 loc) · 7.87 KB
/
app_dynamics_agent.rb
File metadata and controls
198 lines (164 loc) · 7.87 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
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
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
# frozen_string_literal: true
# Cloud Foundry Java Buildpack
# Copyright 2013-2020 the original author or authors.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
require 'fileutils'
require 'java_buildpack/component/versioned_dependency_component'
require 'java_buildpack/framework'
module JavaBuildpack
module Framework
# Encapsulates the functionality for enabling zero-touch AppDynamics support.
class AppDynamicsAgent < JavaBuildpack::Component::VersionedDependencyComponent
def initialize(context)
super(context)
@logger = JavaBuildpack::Logging::LoggerFactory.instance.get_logger AppDynamicsAgent
end
# (see JavaBuildpack::Component::BaseComponent#compile)
def compile
download_zip(false, @droplet.sandbox, 'AppDynamics Agent')
# acessor for resources dir through @droplet?
resources_dir = Pathname.new(File.expand_path('../../../resources', __dir__)).freeze
default_conf_dir = resources_dir + @droplet.component_id + 'defaults'
copy_appd_default_configuration(default_conf_dir)
override_default_config_if_applicable
@droplet.copy_resources
end
# (see JavaBuildpack::Component::BaseComponent#release)
def release
credentials = @application.services.find_service(FILTER, 'host-name')['credentials']
java_opts = @droplet.java_opts
java_opts.add_javaagent(@droplet.sandbox + 'javaagent.jar')
application_name java_opts, credentials
tier_name java_opts, credentials
node_name java_opts, credentials
account_access_key java_opts, credentials
account_name java_opts, credentials
host_name java_opts, credentials
port java_opts, credentials
ssl_enabled java_opts, credentials
unique_host_name java_opts
end
protected
# (see JavaBuildpack::Component::VersionedDependencyComponent#supports?)
def supports?
@application.services.one_service? FILTER, 'host-name'
end
private
CONFIG_FILES = %w[logging/log4j2.xml logging/log4j.xml app-agent-config.xml controller-info.xml
service-endpoint.xml transactions.xml custom-interceptors.xml
custom-activity-correlation.xml].freeze
FILTER = /app[-]?dynamics/.freeze
private_constant :CONFIG_FILES, :FILTER
def application_name(java_opts, credentials)
name = credentials['application-name'] || @configuration['default_application_name'] ||
@application.details['application_name']
java_opts.add_system_property('appdynamics.agent.applicationName', name.to_s)
end
def account_access_key(java_opts, credentials)
account_access_key = credentials['account-access-key'] || credentials.dig('account-access-secret', 'secret')
java_opts.add_system_property 'appdynamics.agent.accountAccessKey', account_access_key if account_access_key
end
def account_name(java_opts, credentials)
account_name = credentials['account-name']
java_opts.add_system_property 'appdynamics.agent.accountName', account_name if account_name
end
def host_name(java_opts, credentials)
host_name = credentials['host-name']
raise "'host-name' credential must be set" unless host_name
java_opts.add_system_property 'appdynamics.controller.hostName', host_name
end
def node_name(java_opts, credentials)
name = credentials['node-name'] || @configuration['default_node_name']
java_opts.add_system_property('appdynamics.agent.nodeName', name.to_s)
end
def port(java_opts, credentials)
port = credentials['port']
java_opts.add_system_property 'appdynamics.controller.port', port if port
end
def ssl_enabled(java_opts, credentials)
ssl_enabled = credentials['ssl-enabled']
java_opts.add_system_property 'appdynamics.controller.ssl.enabled', ssl_enabled if ssl_enabled
end
def tier_name(java_opts, credentials)
name = credentials['tier-name'] || @configuration['default_tier_name'] ||
@application.details['application_name']
java_opts.add_system_property('appdynamics.agent.tierName', name.to_s)
end
def unique_host_name(java_opts)
name = @configuration['default_unique_host_name'] || @application.details['application_name']
java_opts.add_system_property('appdynamics.agent.uniqueHostId', name.to_s)
end
# Copy default configuration present in resources folder of app_dynamics_agent ver* directories present in sandbox
#
# @param [Pathname] default_conf_dir the 'defaults' directory present in app_dynamics_agent resources.
# @return [Void]
def copy_appd_default_configuration(default_conf_dir)
return unless default_conf_dir.exist?
Dir.glob(@droplet.sandbox + 'ver*') do |target_directory|
FileUtils.cp_r "#{default_conf_dir}/.", target_directory
end
end
# Check if configuration file exists on the server before download
# @param [ResourceURI] uri URI of the remote configuration server
# @param [ConfigFileName] conf_file Name of the configuration file
# @return [Boolean] returns true if files exists on path specified by APPD_CONF_HTTP_URL, false otherwise
def check_if_resource_exists(resource_uri, conf_file)
# check if resource exists on remote server
begin
response = Net::HTTP.start(resource_uri.host, resource_uri.port) do |http|
http.request_head(resource_uri)
end
rescue StandardError => e
@logger.error { "Request failure: #{e.message}" }
return false
end
case response
when Net::HTTPSuccess
true
when Net::HTTPRedirection
location = response['location']
@logger.info { "redirected to #{location}" }
check_if_resource_exists(location, conf_file)
else
@logger.info { "Could not retrieve #{resource_uri}. Code: #{response.code} Message: #{response.message}" }
false
end
end
# Check for configuration files on a remote server. If found, copy to conf dir under each ver* dir
# @return [Void]
def override_default_config_if_applicable
return unless @application.environment['APPD_CONF_HTTP_URL']
JavaBuildpack::Util::Cache::InternetAvailability.instance.available(
true, 'The AppDynamics remote configuration download location is always accessible'
) do
agent_root = @application.environment['APPD_CONF_HTTP_URL'].chomp('/') + '/java/'
@logger.info { "Downloading override configuration files from #{agent_root}" }
CONFIG_FILES.each do |conf_file|
uri = URI(agent_root + conf_file)
# `download()` uses retries with exponential backoff which is expensive
# for situations like 404 File not Found. Also, `download()` doesn't expose
# an api to disable retries, which makes this check necessary to prevent
# long install times.
next unless check_if_resource_exists(uri, conf_file)
download(false, uri.to_s) do |file|
Dir.glob(@droplet.sandbox + 'ver*') do |target_directory|
FileUtils.cp_r file, target_directory + '/conf/' + conf_file
end
end
end
end
end
end
end
end