# == Schema Information # # Table name: projects # # id :integer not null, primary key # name :string(255) not null # timeout :integer default(3600), not null # created_at :datetime # updated_at :datetime # token :string(255) # default_ref :string(255) # path :string(255) # always_build :boolean default(FALSE), not null # polling_interval :integer # public :boolean default(FALSE), not null # ssh_url_to_repo :string(255) # gitlab_id :integer # allow_git_fetch :boolean default(TRUE), not null # email_recipients :string(255) default(""), not null # email_add_pusher :boolean default(TRUE), not null # email_only_broken_builds :boolean default(TRUE), not null # skip_refs :string(255) # coverage_regex :string(255) # shared_runners_enabled :boolean default(FALSE) # generated_yaml_config :text # module Ci class Project < ActiveRecord::Base extend Ci::Model include Ci::ProjectStatus has_many :commits, ->() { order(:committed_at) }, dependent: :destroy, class_name: 'Ci::Commit' has_many :builds, through: :commits, dependent: :destroy, class_name: 'Ci::Build' has_many :runner_projects, dependent: :destroy, class_name: 'Ci::RunnerProject' has_many :runners, through: :runner_projects, class_name: 'Ci::Runner' has_many :web_hooks, dependent: :destroy, class_name: 'Ci::WebHook' has_many :events, dependent: :destroy, class_name: 'Ci::Event' has_many :variables, dependent: :destroy, class_name: 'Ci::Variable' has_many :triggers, dependent: :destroy, class_name: 'Ci::Trigger' # Project services has_many :services, dependent: :destroy, class_name: 'Ci::Service' has_one :hip_chat_service, dependent: :destroy, class_name: 'Ci::HipChatService' has_one :slack_service, dependent: :destroy, class_name: 'Ci::SlackService' has_one :mail_service, dependent: :destroy, class_name: 'Ci::MailService' accepts_nested_attributes_for :variables, allow_destroy: true # # Validations # validates_presence_of :name, :timeout, :token, :default_ref, :path, :ssh_url_to_repo, :gitlab_id validates_uniqueness_of :gitlab_id validates :polling_interval, presence: true, if: ->(project) { project.always_build.present? } scope :public_only, ->() { where(public: true) } before_validation :set_default_values class << self include Ci::CurrentSettings def base_build_script <<-eos git submodule update --init ls -la eos end def parse(project) params = { name: project.name_with_namespace, gitlab_id: project.id, path: project.path_with_namespace, default_ref: project.default_branch || 'master', ssh_url_to_repo: project.ssh_url_to_repo, email_add_pusher: current_application_settings.add_pusher, email_only_broken_builds: current_application_settings.all_broken_builds, } project = Ci::Project.new(params) project.build_missing_services project end def from_gitlab(user, scope = :owned, options) opts = user.authenticate_options opts.merge! options projects = Ci::Network.new.projects(opts.compact, scope) if projects projects.map { |pr| OpenStruct.new(pr) } else [] end end def already_added?(project) where(gitlab_id: project.id).any? end def unassigned(runner) joins("LEFT JOIN #{Ci::RunnerProject.table_name} ON #{Ci::RunnerProject.table_name}.project_id = #{Ci::Project.table_name}.id " \ "AND #{Ci::RunnerProject.table_name}.runner_id = #{runner.id}"). where('#{Ci::RunnerProject.table_name}.project_id' => nil) end def ordered_by_last_commit_date last_commit_subquery = "(SELECT project_id, MAX(committed_at) committed_at FROM #{Ci::Commit.table_name} GROUP BY project_id)" joins("LEFT JOIN #{last_commit_subquery} AS last_commit ON #{Ci::Project.table_name}.id = last_commit.project_id"). order("CASE WHEN last_commit.committed_at IS NULL THEN 1 ELSE 0 END, last_commit.committed_at DESC") end def search(query) where("LOWER(#{Ci::Project.table_name}.name) LIKE :query", query: "%#{query.try(:downcase)}%") end end def any_runners? if runners.active.any? return true end shared_runners_enabled && Ci::Runner.shared.active.any? end def set_default_values self.token = SecureRandom.hex(15) if self.token.blank? end def tracked_refs @tracked_refs ||= default_ref.split(",").map{|ref| ref.strip} end def valid_token? token self.token && self.token == token end def no_running_builds? # Get running builds not later than 3 days ago to ignore hangs builds.running.where("updated_at > ?", 3.days.ago).empty? end def email_notification? email_add_pusher || email_recipients.present? end def web_hooks? web_hooks.any? end def services? services.any? end def timeout_in_minutes timeout / 60 end def timeout_in_minutes=(value) self.timeout = value.to_i * 60 end def coverage_enabled? coverage_regex.present? end # Build a clone-able repo url # using http and basic auth def repo_url_with_auth auth = "gitlab-ci-token:#{token}@" url = gitlab_url + ".git" url.sub(/^https?:\/\//) do |prefix| prefix + auth end end def available_services_names %w(slack mail hip_chat) end def build_missing_services available_services_names.each do |service_name| service = services.find { |service| service.to_param == service_name } # If service is available but missing in db # we should create an instance. Ex `create_gitlab_ci_service` service = self.send :"create_#{service_name}_service" if service.nil? end end def execute_services(data) services.each do |service| # Call service hook only if it is active begin service.execute(data) if service.active && service.can_execute?(data) rescue => e logger.error(e) end end end def gitlab_url File.join(GitlabCi.config.gitlab_server.url, path) end def setup_finished? commits.any? end end end