require 'open3' require_relative 'gitlab_init' require_relative 'gitlab_metrics' class GitlabCustomHook attr_reader :vars def initialize(repo_path, key_id) @repo_path = repo_path @vars = { 'GL_ID' => key_id } end def pre_receive(changes) GitlabMetrics.measure("pre-receive-hook") do find_hooks('pre-receive').all? do |hook| call_receive_hook(hook, changes) end end end def post_receive(changes) GitlabMetrics.measure("post-receive-hook") do find_hooks('post-receive').all? do |hook| call_receive_hook(hook, changes) end end end def update(ref_name, old_value, new_value) GitlabMetrics.measure("update-hook") do find_hooks('update').all? do |hook| system(vars, hook, ref_name, old_value, new_value) end end end private def find_hooks(hook_name) [ hook_file(hook_name, @repo_path), hook_file(hook_name, ROOT_PATH) ].compact end def call_receive_hook(hook, changes) # Prepare the hook subprocess. Attach a pipe to its stdin, and merge # both its stdout and stderr into our own stdout. stdin_reader, stdin_writer = IO.pipe hook_pid = spawn(vars, hook, in: stdin_reader, err: :out) stdin_reader.close # Submit changes to the hook via its stdin. begin IO.copy_stream(StringIO.new(changes), stdin_writer) rescue Errno::EPIPE # It is not an error if the hook does not consume all of its input. end # Close the pipe to let the hook know there is no further input. stdin_writer.close Process.wait(hook_pid) $?.success? end def hook_file(hook_type, repo_path) hook_path = File.join(repo_path.strip, 'custom_hooks') hook_file = "#{hook_path}/#{hook_type}" hook_file if File.executable?(hook_file) end end