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"
15 puts " --bwlimit=RATE Limit bandwidth (in KB)"
20 opts = GetoptLong.new(
21 [ '--dry-run', '-n', GetoptLong::NO_ARGUMENT ],
22 [ '--project', '-p', GetoptLong::REQUIRED_ARGUMENT ],
23 [ '--package', GetoptLong::REQUIRED_ARGUMENT ],
24 [ '--release', GetoptLong::REQUIRED_ARGUMENT ],
25 [ '--visibility', '-v', GetoptLong::REQUIRED_ARGUMENT ],
26 [ '--force-digest', GetoptLong::NO_ARGUMENT],
27 [ '--progress', GetoptLong::NO_ARGUMENT],
28 [ '--no-progress', GetoptLong::NO_ARGUMENT],
29 [ '--bwlimit', GetoptLong::REQUIRED_ARGUMENT ],
31 opts.each do |opt, arg|
43 unless %w(public private hidden).member?(arg)
44 logger.fatal "Invalid visibility status: #{arg}"
55 @show_progress = false
58 OSDN::CLI._rate_limit = arg.to_i * 1024
62 (ARGV.empty? ? ['.'] : ARGV).each do |d|
63 @target_dir = Pathname.new(d)
69 proj_info = api.get_project target_proj # check project existance
71 vars = load_variables(@target_dir)
72 parent_vars = load_variables(@target_dir.parent)
74 if @target_release || vars.release_id ||
75 parent_vars.package_id && !vars.release_id # new release case...
76 process_release(@target_dir)
77 elsif @target_package || vars.package_id
78 process_package(@target_dir)
80 Pathname.glob(@target_dir+'*').sort.each do |pdir|
87 "Upload local file tree and create package/release implicitly."
90 def process_package(pdir)
91 if cur_pkgid = load_variables(pdir).package_id
92 # check package existance on server
94 api.get_package target_proj, target_package(pdir)
95 rescue OSDNClient::ApiError => e
97 err = JSON.parse(e.response_body)
101 if err['status'] == 404
102 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."
108 logger.info "Createing new package '#{pdir.basename}'"
110 pinfo = Hashie::Mash.new id: '(dry-run)', name: pdir.basename, url: '(dry-run)'
112 pinfo = api.create_package target_proj, pdir.basename, visibility: @visibility
113 update_variables pdir, package_id: pinfo.id
115 $stdout.puts "New package '#{pinfo.name}' has been created; #{pinfo.url}"
118 Pathname.glob(pdir + '*').sort.each do |rdir|
119 process_release(rdir)
123 def process_release(rdir)
125 logger.warn "Skip normal file '#{rdir}' in release level"
129 vars = load_variables(rdir)
133 rinfo = api.get_release target_proj, target_package(rdir), target_release(rdir)
134 rescue OSDNClient::ApiError => e
136 err = JSON.parse(e.response_body)
140 if err['status'] == 404
141 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."
147 logger.info "Createing new release '#{rdir.basename}'"
149 rinfo = Hashie::Mash.new id: '(dry-run)', name: rdir.basename, url: '(dry-run)', files: []
151 rinfo = api.create_release target_proj, target_package(rdir), rdir.basename, visibility: @visibility
152 update_variables rdir, release_id: rinfo.id
154 $stdout.puts "New release '#{rinfo.name}' has been created; #{rinfo.url}"
156 Pathname.glob(rdir + '*').sort.each do |file|
157 process_file(file, rdir, rinfo)
161 def process_file(file, rdir, rinfo)
163 logger.error "Skip direcotry #{file}"
167 vars = load_variables(rdir)
169 if !@force_digest && vars.local_file_info &&
170 vars.local_file_info[file.basename.to_s]
171 finfo = vars.local_file_info[file.basename.to_s]
172 if finfo[:size] == file.size && finfo.mtime == file.mtime
173 digests = vars.local_file_info[file.basename.to_s].digests
178 logger.info "Calculating digest for #{file}..."
180 sha256: hexdigest(Digest::SHA256, file),
181 sha1: hexdigest(Digest::SHA1, file),
182 md5: hexdigest(Digest::MD5, file),
184 update_variables rdir, {local_file_info: {file.basename.to_s => {digests: digests, mtime: file.mtime, size: file.size}}}
186 if remote_f = rinfo.files.find { |f| f.name == file.basename.to_s }
187 if digests.find { |type, dig| dig != remote_f.send("digest_#{type}") }
188 logger.error "#{file} was changed from remote file! Please delete remote file before uploading new one."
190 logger.info "Skip already uploaded file '#{file}'"
194 logger.info "Uploading file #{file} (#{file.size} bytes)"
196 finfo = Hashie::Mash.new id: '(dry-run)', url: '(dry-run)'
198 logger.level <= Logger::INFO && @show_progress != false || @show_progress and
199 OSDN::CLI._show_progress = true
201 logger.info "Starting upload #{file}..."
202 finfo = api.create_release_file target_proj, target_package(rdir), target_release(rdir), fio, visibility: @visibility
204 OSDN::CLI._show_progress = false
205 if digests.find { |type, dig| dig != finfo.send("digest_#{type}") }
206 logger.error "File digests are mismatch! Upload file #{file} may be broken! Please check."
208 logger.info "Upload complete."
211 $stdout.puts "New file '#{file}' has been uploaded; #{finfo.url}"
216 @target_proj and return @target_proj
217 vars = load_variables(@target_dir)
218 vars.project && !vars.project.empty? and
220 logger.fatal "No target project is specified."
224 def target_package(dir)
225 @target_package and return @target_package
226 vars = load_variables(dir)
227 vars.package_id && !vars.package_id.to_s.empty? and
228 return vars.package_id
229 logger.fatal "No target package is specified."
233 def target_release(dir)
234 @target_release and return @target_release
235 vars = load_variables(dir)
236 vars.release_id && !vars.release_id.to_s.empty? and
237 return vars.release_id
238 logger.fatal "No target release is specified."
243 OSDNClient::ProjectApi.new