1 module OSDN; module CLI; module Command
4 puts "#{$0} frs_upload [opts] [target_dir]"
6 puts " -n --dry-run Do noting (use with global -v to inspect)"
7 puts " -p --project=<project> Target project (numeric id or name)"
8 puts " --package=<package-id> Target package (numeric id)"
9 puts " --release=<release-id> Target release (numeric id)"
10 puts " -v --visibility=<public|private|hidden>"
11 puts " Default visibility for newly created items"
12 puts " --force-digest Calc local file digest forcely"
13 puts " --progress Force to show upload progress"
14 puts " --no-progress Force to hide upload progress"
19 opts = GetoptLong.new(
20 [ '--dry-run', '-n', GetoptLong::NO_ARGUMENT ],
21 [ '--project', '-p', GetoptLong::REQUIRED_ARGUMENT ],
22 [ '--package', GetoptLong::REQUIRED_ARGUMENT ],
23 [ '--release', GetoptLong::REQUIRED_ARGUMENT ],
24 [ '--visibility', '-v', GetoptLong::REQUIRED_ARGUMENT ],
25 [ '--force-digest', GetoptLong::NO_ARGUMENT],
26 [ '--progress', GetoptLong::NO_ARGUMENT],
27 [ '--no-progress', GetoptLong::NO_ARGUMENT],
29 opts.each do |opt, arg|
41 unless %w(public private hidden).member?(arg)
42 logger.fatal "Invalid visibility status: #{arg}"
53 @show_progress = false
57 (ARGV.empty? ? ['.'] : ARGV).each do |d|
58 @target_dir = Pathname.new(d)
64 proj_info = api.get_project target_proj # check project existance
66 vars = load_variables(@target_dir)
67 parent_vars = load_variables(@target_dir.parent)
69 if @target_release || vars.release_id ||
70 parent_vars.package_id && !vars.release_id # new release case...
71 process_release(@target_dir)
72 elsif @target_package || vars.package_id
73 process_package(@target_dir)
75 Pathname.glob(@target_dir+'*').sort.each do |pdir|
82 "Upload local file tree and create package/release implicitly."
85 def process_package(pdir)
86 if cur_pkgid = load_variables(pdir).package_id
87 # check package existance on server
89 api.get_package target_proj, target_package(pdir)
90 rescue OSDNClient::ApiError => e
92 err = JSON.parse(e.response_body)
96 if err['status'] == 404
97 logger.warn "Package ##{cur_pkgid} has been deleted on server and local directory '#{pdir}' remains. You can delete the local directory or delete '#{pdir}/.osdn.vars' file to create a package again with new ID."
103 logger.info "Createing new package '#{pdir.basename}'"
105 pinfo = Hashie::Mash.new id: '(dry-run)', name: pdir.basename, url: '(dry-run)'
107 pinfo = api.create_package target_proj, pdir.basename, visibility: @visibility
108 update_variables pdir, package_id: pinfo.id
110 $stdout.puts "New package '#{pinfo.name}' has been created; #{pinfo.url}"
113 Pathname.glob(pdir + '*').sort.each do |rdir|
114 process_release(rdir)
118 def process_release(rdir)
120 logger.warn "Skip normal file '#{rdir}' in release level"
124 vars = load_variables(rdir)
128 rinfo = api.get_release target_proj, target_package(rdir), target_release(rdir)
129 rescue OSDNClient::ApiError => e
131 err = JSON.parse(e.response_body)
135 if err['status'] == 404
136 logger.warn "Release ##{vars.release_id} has been deleted on server and local directory '#{rdir}' remains. You can delete the local directory or delete '#{rdir}/.osdn.vars' file to create a release again with new ID."
142 logger.info "Createing new release '#{rdir.basename}'"
144 rinfo = Hashie::Mash.new id: '(dry-run)', name: rdir.basename, url: '(dry-run)', files: []
146 rinfo = api.create_release target_proj, target_package(rdir), rdir.basename, visibility: @visibility
147 update_variables rdir, release_id: rinfo.id
149 $stdout.puts "New release '#{rinfo.name}' has been created; #{rinfo.url}"
151 Pathname.glob(rdir + '*').sort.each do |file|
152 process_file(file, rdir, rinfo)
156 def process_file(file, rdir, rinfo)
158 logger.error "Skip direcotry #{file}"
162 vars = load_variables(rdir)
164 if !@force_digest && vars.local_file_info &&
165 vars.local_file_info[file.basename.to_s]
166 finfo = vars.local_file_info[file.basename.to_s]
167 if finfo[:size] == file.size && finfo.mtime == file.mtime
168 digests = vars.local_file_info[file.basename.to_s].digests
173 logger.info "Calculating digest for #{file}..."
175 sha256: hexdigest(Digest::SHA256, file),
176 sha1: hexdigest(Digest::SHA1, file),
177 md5: hexdigest(Digest::MD5, file),
179 update_variables rdir, {local_file_info: {file.basename.to_s => {digests: digests, mtime: file.mtime, size: file.size}}}
181 if remote_f = rinfo.files.find { |f| f.name == file.basename.to_s }
182 if digests.find { |type, dig| dig != remote_f.send("digest_#{type}") }
183 logger.error "#{file} was changed from remote file! Please delete remote file before uploading new one."
185 logger.info "Skip already uploaded file '#{file}'"
189 logger.info "Uploading file #{file} (#{file.size} bytes)"
191 finfo = Hashie::Mash.new id: '(dry-run)', url: '(dry-run)'
193 logger.level <= Logger::INFO && @show_progress != false || @show_progress and
194 OSDN::CLI._show_progress = true
196 logger.info "Starting upload #{file}..."
197 finfo = api.create_release_file target_proj, target_package(rdir), target_release(rdir), fio, visibility: @visibility
199 OSDN::CLI._show_progress = false
200 if digests.find { |type, dig| dig != finfo.send("digest_#{type}") }
201 logger.error "File digests are mismatch! Upload file #{file} may be broken! Please check."
203 logger.info "Upload complete."
206 $stdout.puts "New file '#{file}' has been uploaded; #{finfo.url}"
211 @target_proj and return @target_proj
212 vars = load_variables(@target_dir)
213 vars.project && !vars.project.empty? and
215 logger.fatal "No target project is specified."
219 def target_package(dir)
220 @target_package and return @target_package
221 vars = load_variables(dir)
222 vars.package_id && !vars.package_id.to_s.empty? and
223 return vars.package_id
224 logger.fatal "No target package is specified."
228 def target_release(dir)
229 @target_release and return @target_release
230 vars = load_variables(dir)
231 vars.release_id && !vars.release_id.to_s.empty? and
232 return vars.release_id
233 logger.fatal "No target release is specified."
238 OSDNClient::ProjectApi.new