#!/usr/bin/ruby # # This file is part of centurio.work/ing/commands. # # centurio.work/ing/commands is free software: you can redistribute it and/or # modify it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or (at your # option) any later version. # # centurio.work/ing/commands is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General # Public License for more details. # # You should have received a copy of the GNU General Public License along with # centurio.work/ing/commands (file COPYING in the main directory). If not, see # . require 'rubygems' require 'json' require 'riddl/server' require 'riddl/client' require 'fileutils' require 'nokogiri' require 'sqlite3' require 'net/http' require 'uri' $db = SQLite3::Database.open 'database/stations.db' class GetJsonFiles < Riddl::Implementation def response Dir.chdir( __dir__ + '/json') ret = []; Dir.glob('*').select{ |e| ret.append(e) } Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end class GetJson < Riddl::Implementation def response Dir.chdir( __dir__ + '/json') Riddl::Parameter::Complex.new('list','application/json',File.open(@r[1]).read) rescue Riddl::Parameter::Complex.new('list','application/json', '{"error":"No such file"}') end end class SaveJson < Riddl::Implementation def response Dir.chdir( __dir__ + '/json') File.write(@r.last,(@p[0].value.read)) end end class GetStations < Riddl::Implementation def response result = $db.execute "SELECT * FROM stations ORDER BY station ASC" builder = Nokogiri::XML::Builder.new do |xml| xml.stations { result.each do |row| xml.station(:id => row[0]){ resultstation = $db.execute "SELECT * FROM stationpattern WHERE station = #{row[0]}" resultstation.each do |row| xml.pattern(:id => row[1], :description => row[2], :changed => row[3]){ xml.parts{ resultpattern = $db.execute "SELECT * FROM pattern WHERE internalID = #{row[1]}" resultpattern.each do |row2| xml.send(row2[1], row2[2]) end } } end } end } end #puts builder.to_xml Riddl::Parameter::Complex.new('stations','application/xml',builder.to_xml) end end class CreateStation < Riddl::Implementation def self::createDB(n) #result = $db.execute "SELECT MAX(station) FROM stations" #if(result[0][0] == nil) # $db.execute "INSERT INTO stations (station) VALUES (?)", 0 #else # $db.execute "INSERT INTO stations (station) VALUES (?)", result[0][0]+1 #end #$db.execute ("INSERT INTO stations (station, name) VALUES (?,?)", [n,n]) $db.execute("INSERT OR IGNORE INTO stations (station, name) VALUES (?,?)", [n,n]) end def response CreateStation::createDB(7) end end class GetStation < Riddl::Implementation def self::prepare(id) resultstation = $db.execute "SELECT * FROM stationpattern WHERE station = #{id}" Nokogiri::XML::Builder.new do |xml| xml.station(:id => id){ resultstation.each do |row| xml.pattern(:id => row[1], :description => row[2], :changed => row[3]){ xml.parts{ resultpattern = $db.execute "SELECT * FROM pattern WHERE internalID = #{row[1]}" resultpattern.each do |row2| xml.send(row2[1], row2[2]) end } } end } end end def response builder = GetStation::prepare(@r.last) Riddl::Parameter::Complex.new('station','application/xml',builder.to_xml) end end class CreatePattern < Riddl::Implementation def response doc = Nokogiri::XML(@p[0].value) result = $db.execute "SELECT MAX(internalID) FROM stationpattern" if(result[0][0] == nil) id = 0 else id = result[0][0] +1 end $db.execute("INSERT INTO stationpattern (station, internalID, description, date) VALUES (?,?,?,?)", [@r.last, id, doc.xpath("/*/@description")[0].value, doc.xpath("/*/@changed")[0].value]) doc.xpath(".//parts/*").each do |node| #if(node.text != "") $db.execute("INSERT INTO pattern (internalID, type, value) VALUES (?,?,?)", [id, node.name, node.text]) #end end ret = {:id => id} Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end class GetPattern < Riddl::Implementation def response resultstation = $db.execute "SELECT * FROM stationpattern WHERE station = #{@r[0]} AND internalID = #{@r[1]}" builder = Nokogiri::XML::Builder.new do |xml| resultstation.each do |row| xml.pattern(:id => row[1], :description => row[2], :changed => row[3]){ xml.parts{ resultpattern = $db.execute "SELECT * FROM pattern WHERE internalID = #{row[1]}" resultpattern.each do |row2| xml.send(row2[1], row2[2]) end } } end end Riddl::Parameter::Complex.new('pattern','application/xml',builder.to_xml) end end class UpdatePattern < Riddl::Implementation def response doc = Nokogiri::XML(@p[0].value) $db.execute("Update stationpattern SET description = ?, date = ? WHERE internalID = ?", [doc.xpath("/*/@description")[0].value, doc.xpath("/*/@changed")[0].value, @r[1]]) $db.execute("DELETE FROM pattern WHERE internalID = ?", [@r[1]]) doc.xpath(".//parts/*").each do |node| #if(node.text != "") $db.execute("INSERT INTO pattern (internalID, type, value) VALUES (?,?,?)", [@r[1], node.name, node.text]) #end end end end class DeletePattern < Riddl::Implementation def response $db.execute("DELETE FROM stationpattern WHERE internalID = ?", [@r[1]]) end end class DuplicatePattern < Riddl::Implementation def response #duplicate db etwas mühsam da der eintrag kopiert werden muss und dabei die unique ID verändert werden muss daher geht insert into select nicht so ganz. result = $db.execute "SELECT MAX(internalID) FROM stationpattern" if(result[0][0] == nil) id = 0 else id = result[0][0] +1 end result = $db.execute "SELECT * FROM stationpattern WHERE internalID = #{@r[1]}" result[0][1] = id; $db.execute( "INSERT INTO stationpattern values (?,?,?,?)", result[0]) result = $db.execute "SELECT * FROM pattern WHERE internalID = #{@r[1]}" result.each do |row| row[0] = id; $db.execute( "INSERT INTO pattern values (?,?,?)", row) end #duplicate error #$db.execute "CREATE TABLE IF NOT EXISTS error(internalID INT, error TEXT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" result = $db.execute "SELECT * FROM error WHERE internalID = #{@r[1]}" if(result[0] != nil) result.each do |row| row[0] = id; questionmarks = "" row.length().times{questionmarks = questionmarks + ",?"} questionmarks[0] = "" $db.execute( "INSERT INTO error values (#{questionmarks})", row) end end #duplicate image db #$db.execute "CREATE TABLE IF NOT EXISTS image(internalID INT, imageID INT, language CHARACTER(20), label TEXT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" result = $db.execute "SELECT * FROM image WHERE internalID = #{@r[1]}" if(result[0] != nil) result.each do |row| row[0] = id; questionmarks = "" row.length().times{questionmarks = questionmarks + ",?"} questionmarks[0] = "" $db.execute( "INSERT INTO image values (#{questionmarks})", row) end end #duplicate images pathorigin = File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1]) pathdestination = File.join(File.dirname(__dir__),'images/uploads', @r[0], id.to_s) if (Dir.exist?(pathorigin)) FileUtils.mkdir_p(pathdestination) FileUtils.copy_entry pathorigin, pathdestination end #duplicate Replacement db #$db.execute "CREATE TABLE IF NOT EXISTS replacements(internalID INT, abbreviation TEXT, url TEXT, ordering INT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" result = $db.execute "SELECT * FROM replacements WHERE internalID = #{@r[1]}" if(result[0] != nil) result.each do |row| row[0] = id; questionmarks = "" row.length().times{questionmarks = questionmarks + ",?"} questionmarks[0] = "" $db.execute( "INSERT INTO replacements values (#{questionmarks})", row) end end ret = {:id => id} Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end class SaveError < Riddl::Implementation def response $db.execute "CREATE TABLE IF NOT EXISTS error(internalID INT, error TEXT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" #Delete all, afterwards insert $db.execute("DELETE FROM error WHERE internalID = ?", [@r[1]]) doc = Nokogiri::XML(@p[0].value) doc.xpath(".//reason").each do |node| $db.execute("INSERT INTO error (internalID, error) VALUES (?,?)", [@r[1], node.text]) end end end class GetError < Riddl::Implementation def response result = $db.execute "SELECT * FROM error WHERE internalID = #{@r[1]}" builder = Nokogiri::XML::Builder.new do |xml| xml.error(){ result.each do |row| xml.reason row[1] end } end Riddl::Parameter::Complex.new('error','application/xml',builder.to_xml) end end class SaveReplacement < Riddl::Implementation def response # $db.execute "CREATE TABLE IF NOT EXISTS replacements(internalID INT, abbreviation TEXT, url TEXT, ordering INT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" #Delete all, afterwards insert $db.execute("DELETE FROM replacements WHERE internalID = ?", [@r[1]]) order = 0; doc = Nokogiri::XML(@p[0].value) doc.xpath(".//item").each do |node| $db.execute("INSERT INTO replacements (internalID, abbreviation, url, ordering) VALUES (?,?,?,?)", [@r[1], node.xpath(".//abbreviation").text, node.xpath(".//url").text, order]) order += 1 end end end class GetReplacement < Riddl::Implementation def response result = $db.execute "SELECT * FROM replacements WHERE internalID = #{@r[1]} ORDER BY ordering ASC" builder = Nokogiri::XML::Builder.new do |xml| xml.replacement(){ result.each do |row| xml.item(){ xml.abbreviation row[1] xml.url row[2] } end } end Riddl::Parameter::Complex.new('replacement','application/xml',builder.to_xml) end end class GetImages < Riddl::Implementation def response formatted_no_decl = Nokogiri::XML::Node::SaveOptions::FORMAT + Nokogiri::XML::Node::SaveOptions::NO_DECLARATION #$db.execute "CREATE TABLE IF NOT EXISTS image(internalID INT, imageID INT, language CHARACTER(20), label TEXT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" result = $db.execute "SELECT DISTINCT imageID FROM image WHERE internalID = #{@r[1]} ORDER BY imageID ASC" builder = Nokogiri::XML::Builder.new do |xml| xml.images(){ result.each do |row| xml<< GetImage::prepare(@r[0], @r[1], row[0], @a[0][:appconf]["wasurl"]).to_xml( save_with:formatted_no_decl ) end } end Riddl::Parameter::Complex.new('images','application/xml',builder.to_xml) end end class UploadImage < Riddl::Implementation def response lang = @p[@p.length-1].value #$db.execute "CREATE TABLE IF NOT EXISTS image(internalID INT, imageID INT, language CHARACTER(20), label TEXT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" result = $db.execute "SELECT MAX(imageID) FROM image WHERE internalID = #{@r[1]} and language = '#{lang}'" if(result[0][0] == nil) id = 0 else id = result[0][0] +1 end #puts JSON.pretty_generate(@p) i = 0 while i < @p.length-1 do item = @p[i] if(item != nil && item.name == "files[]") path = File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], id.to_s) FileUtils.mkdir_p(path) #juergen nach alternative fragen readFile = File.read(item.value.inspect.to_s[/Tempfile:(.*?)>/m, 1]) File.open(File.join(path, lang + ".svg"), 'wb') do |file| file.write(readFile.to_s) end $db.execute("INSERT INTO image (internalID, imageID, language, label) VALUES (?,?,?,?)", [@r[1], id, lang, "Label"]) id += 1 end i +=1 end end end class AddExternalImage < Riddl::Implementation def response doc = Nokogiri::XML(@p[0].value) url = doc.xpath("/externalImage/url").text lang = doc.xpath("/externalImage/lang").text result = $db.execute "SELECT MAX(imageID) FROM image WHERE internalID = #{@r[1]} and language = '#{lang}'" if(result[0][0] == nil) id = 0 else id = result[0][0] +1 end #Create link file (we create a file so moving/deleting works always the same and the database does not have to be changed for external files) only thing that had to be changed was imgReplacement.php image = ' ' path = File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], id.to_s) FileUtils.mkdir_p(path) File.open(File.join(path, lang + ".svg"), 'wb') do |file| file.write(image) end #Create DB entry as usual $db.execute("INSERT INTO image (internalID, imageID, language, label) VALUES (?,?,?,?)", [@r[1], id, lang, url]) end end class ReorderImages < Riddl::Implementation def response #remove brackets iter = @p[0].value.read.chop iter[0] ="" iter = iter.split(",").map(&:to_i) i = 0 while i < iter.length do if(i != iter[i]) #swap path = File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], i.to_s) path2 = File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], iter[i].to_s) tmp = File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1]) FileUtils.mv(path + "/" + @r[4] + ".svg", tmp) FileUtils.mv(path2 + "/" + @r[4] + ".svg", path) FileUtils.mv(tmp + "/" + @r[4] + ".svg", path2) #DB Swap result = $db.execute "SELECT MAX(imageID) FROM image" if(result[0][0] == nil) maxImgId = 0 else maxImgId = result[0][0] +1 end #Wegen 999999 fragen #nutze max ID+1 zum tauschen $db.execute("UPDATE image SET imageID = ? WHERE internalID = ? AND imageID = ? AND language = ?", [maxImgId, @r[1], i, @r[4]]) $db.execute("UPDATE image SET imageID = ? WHERE internalID = ? AND imageID = ? AND language = ?", [i, @r[1], iter[i], @r[4]]) $db.execute("UPDATE image SET imageID = ? WHERE internalID = ? AND imageID = ? AND language = ?", [iter[i], @r[1], maxImgId, @r[4]]) iter.map! do |item| if(item == i) iter[i] else item end end end i +=1 end end end class GetImage < Riddl::Implementation def self::prepare(station, pattern, imageID, wasurl) result = $db.execute "SELECT * FROM image WHERE internalID = #{pattern} and imageID = #{imageID}" Nokogiri::XML::Builder.new do |xml| xml.image(:id => imageID){ result.each do |row| xml.variant(:lang => row[2], :label => row[3]){ xml.text(wasurl + "images/uploads/#{station}/#{pattern}/#{imageID}/#{row[3]}.svg") } end } end end def response builder = GetImage::prepare(@r[0], @r[1], @r[3], @a[0][:appconf]["wasurl"]) Riddl::Parameter::Complex.new('image','application/xml',builder.to_xml) end end class UpdateImageLabel < Riddl::Implementation def response $db.execute("UPDATE image SET label = ? WHERE internalID = ? AND imageID = ? AND language = ?", [@p[0].value, @r[1], @r[3], @r[4]]) end end #Not used, prototype for replacing image serverside class GetRealImage < Riddl::Implementation def response #split on "." and tacke [0] to allow for e.g. de-at.svg #would lead to error on de.at.svg //should this be fixed? would also lead to error if everything after last "." would be removed in case of only having de.at #https://centurio.work/customers/evva/was/server/0/0/images/3/de-at?video=xyz img = File.read(File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], @r[3], @r[4].split(".")[0] + ".svg")) if false #Currently replacements are done on the client side if(@p[0].nil?) #puts "Undefined p" else if(@p[0].name == "video" && @p[0].value) xml = Nokogiri.parse img #puts "Width " + xml.xpath("string(//xmlns:image/@width)") #puts "Height " + xml.xpath("string(//xmlns:image/@height)") #puts "Posi " + xml.xpath("string(//xmlns:clipPath/path/@height)") #puts "Posi " + xml.xpath("string(//xmlns:image/following-sibling::clipPath/@id)") #puts xml.xpath("string(//xmlns:text[starts-with(text(), 'url')])").sub("url=", "") # text img = img.sub! "", ' ' else @p.each do |item| unless item.name.nil? || item.value.nil? img.sub! item.name, item.value end end end end end Riddl::Parameter::Complex.new('theRealImage','image/svg+xml',img) end end class DeleteImage < Riddl::Implementation def response File.delete(File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], @r[3] , @r[4] + ".svg")) if File.exist?(File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], @r[3] , @r[4] + ".svg")) $db.execute("DELETE FROM image WHERE internalID = ? AND imageID = ? AND language = ?", [@r[1], @r[3], @r[4]]) result = $db.execute "SELECT MAX(imageID) FROM image WHERE internalID = #{@r[1]} and language = '#{@r[4]}'" if(result[0][0] == nil) id = 0 else id = result[0][0] +1 end #reorder if !end gets deleted if(id == @r[3]) return else cur = @r[3].to_i + 1 prev = @r[3].to_i while cur < id.to_i do $db.execute("UPDATE image SET imageID = ? WHERE internalID = ? AND imageID = ? AND language = ?", [prev, @r[1], cur, @r[4]]) src = File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], cur.to_s, @r[4] + ".svg") target = File.join(File.dirname(__dir__),'images/uploads', @r[0], @r[1], prev.to_s, @r[4] + ".svg") FileUtils.mv(src, target) cur+=1 prev += 1 end end end end class ListSearch < Riddl::Implementation def response ret = { :imagesWIZAll => "/imagesWIZ?pattern=...", :imagesWIZSingle => "/imagesWIZ/0..*/?pattern=...", :errorsWIZ => "/errorsWIZ/" } Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end class SearchImages < Riddl::Implementation def response #https://centurio.work/customers/evva/was/ui/TransformProductCode.php?ProductCode=E.A.PZ.DZ-S.31.31.MP.SEP pattern = JSON.parse(Net::HTTP.get URI(@a[0][:appconf]["wasurl"] + "ui/TransformProductCode.php?ProductCode=" + @p[0].value)) #Get first key dynamicly and then build sql query searchstring = "SELECT " + pattern.keys[0] + ".internalID AS ID" searchstring = searchstring + " FROM " + pattern.map{ |key, val| "pattern " + key}.join(', ') searchstring = searchstring + " WHERE " + pattern.map{ |key, val| pattern.keys[0] + ".internalID == " + key + ".internalID"}.join(' AND ') searchstring = searchstring + " AND " + pattern.map{ |key, val| key + ".type == '" + key + "' AND (" + key + ".value == '"+ val +"' or " + key + ".value == '*')"}.join(' AND ') + " ORDER BY ID ASC" result = $db.execute searchstring; #puts searchstring #Search for images with one or multiple found patternIDs ret = [] if result.length > 0 result.each do |item| result2 = $db.execute "SELECT DISTINCT internalID, imageID FROM image WHERE internalID = #{item[0]} ORDER BY imageID ASC" result2.each do |item2| result3 = $db.execute "SELECT DISTINCT station FROM stationpattern WHERE internalID = #{item2[0]} AND station = #{@r[0]}" if(result3[0] != nil) ret << @r[0] + "/" + item2[0].to_s + "/" + item2[1].to_s end end end end Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end class SearchImagesSingle < Riddl::Implementation def response pattern = JSON.parse(Net::HTTP.get URI(@a[0][:appconf]["wasurl"] + "ui/TransformProductCode.php?ProductCode=" + @p[0].value)) searchstring = "SELECT " + pattern.keys[0] + ".internalID AS ID" searchstring = searchstring + " FROM " + pattern.map{ |key, val| "pattern " + key}.join(', ') searchstring = searchstring + " WHERE " + pattern.map{ |key, val| pattern.keys[0] + ".internalID == " + key + ".internalID"}.join(' AND ') searchstring = searchstring + " AND " + pattern.map{ |key, val| key + ".type == '" + key + "' AND (" + key + ".value == '"+ val +"' or " + key + ".value == '*')"}.join(' AND ') + " ORDER BY ID ASC" result = $db.execute searchstring; count = 0 pattern = 0 image = 0 builder = Nokogiri::XML::Builder.new do |xml| xml.image(:id => @r[3].to_s){ if result.length > 0 result.each do |item| result2 = $db.execute "SELECT DISTINCT internalID, imageID FROM image WHERE internalID = #{item[0]} ORDER BY imageID ASC" result2.each do |item2| result3 = $db.execute "SELECT DISTINCT internalID, imageID, language, label FROM image WHERE internalID = #{item[0]} and imageID =#{item2[1]}" result3.each do |item3| result4 = $db.execute "SELECT DISTINCT station FROM stationpattern WHERE internalID = #{item3[0]} AND station = #{@r[0]}" if(result4[0] != nil) if(count == @r[3].to_i) #ret << @r[0] + "/" + item2[0].to_s + "/" + item2[1].to_s uri = @a[0][:appconf]["wasurl"] + "ui/imageReplacement.php?___image___=" + @r[0].to_s + "/" + item3[0].to_s + "/" + item3[1].to_s + "/" + item3[2].to_s + ".svg" @p.each_with_index do |item, index| if index != 0 uri += "&" + item.name.to_s + "=" + item.value.to_s end end xml.variant(:lang => item3[2].to_s, :label => item3[3].to_s){ xml.text(uri) } end count += 1 end end end end end } end #https://centurio.work/customers/evva/was/ui/imageReplacement.php?___image___=8/23/0/de-at.svg #puts builder.to_xml Riddl::Parameter::Complex.new('image','application/xml',builder.to_xml) end end class SearchErrors < Riddl::Implementation def response pattern = JSON.parse(Net::HTTP.get URI(@a[0][:appconf]["wasurl"] + "ui/TransformProductCode.php?ProductCode=" + @p[0].value)) searchstring = "SELECT " + pattern.keys[0] + ".internalID AS ID" searchstring = searchstring + " FROM " + pattern.map{ |key, val| "pattern " + key}.join(', ') searchstring = searchstring + " WHERE " + pattern.map{ |key, val| pattern.keys[0] + ".internalID == " + key + ".internalID"}.join(' AND ') searchstring = searchstring + " AND " + pattern.map{ |key, val| key + ".type == '" + key + "' AND (" + key + ".value == '"+ val +"' or " + key + ".value == '*')"}.join(' AND ') + " ORDER BY ID ASC" result = $db.execute searchstring; ret = [] if result.length > 0 result.each do |item| result2 = $db.execute "SELECT DISTINCT error FROM error WHERE internalID = #{item[0]} ORDER BY internalID ASC" result2.each do |item2| ret << item2[0].to_s end end end ret = ret.uniq builder = Nokogiri::XML::Builder.new do |xml| xml.reason{ if result.length > 0 ret.each do |item| xml.reason(:lang => "de-at"){ xml.text(item) } end end } end Riddl::Parameter::Complex.new('errors','application/xml',builder.to_xml) end end class GetAssignments < Riddl::Implementation def response ret = {} resultstation = $db.execute "SELECT * FROM assignments" resultstation.each do |row| ret[row[0]] = row[1] end Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end class SaveNewAssignment < Riddl::Implementation def response form = JSON.parse(@p[0].value.read); result = $db.execute "SELECT MAX(ID) FROM assignments" if(result[0][0] == nil) id = 0 else id = result[0][0] +1 end #bring given pattern into correct string (initially pattern could be in different order than required) pattern = []; patternsymlink = []; fullcode = JSON.parse( File.read(File.join(File.dirname(__dir__),"server/json/ProductCode.json"))) fullcode["pattern"].each do |child| pattern.push(form["parts"][child["abk"]]) patternsymlink.push(child["abk"]) end #Create Assignments Symlink #Station3_Ba;DZ-S_Of;MP_ID;12.jpg #img url #form["img"] #img name #pp form #pp pattern #pp patternsymlink link = "Station" + form["station"] pattern.each_with_index {|val, index| if(val != "") link = link + "_" + patternsymlink[index].to_s + ";" + val.to_s end } if(form["assignmentText"] != "") link = link + "_ID;" + form["assignmentText"].to_s end link = link + ".jpg" #pp link #pp @a[0][:appconf]["universalstorageeinzelbilder"] #File.write(File.join(@a[0][:appconf]["universalstorageeinzelbilder"]),File.open("defaultContent/js/ui.js").read) #File.symlink File.join("test.jpg"), File.join(@a[0][:appconf]["universalstorageeinzelbilder"]) imgfile = File.join(form["img"]) symfile = File.join(@a[0][:appconf]["universalstorageeinzelbilder"], "Symlinks", link) #pp imgfile; #pp symfile; if File.file?(imgfile) #file exists if !File.file?(symfile) #symlink should not exist nw = "" + form["img"] nw.slice! @a[0][:appconf]["universalstorageeinzelbilder"].to_s File.symlink "../" + nw, symfile end end #Create Assignemtns DB checkimage = $db.execute "SELECT * FROM assignments WHERE image = '#{form["img"]}'" #if image not stored create new assignment with pattern, otherwise only add pattern if(checkimage[0] == nil) #check if assignment would overlap if (!ListAssignmentSearch::prepare(@a[0][:appconf]["wasurl"], pattern.join('.'), form["station"], form["assignmentText"]).any?) $db.execute("INSERT INTO assignments (ID, image) VALUES (?,?)", [id, form["img"]]) form["parts"].each{ |key,val| $db.execute("INSERT INTO patternassignments (ID, AssID, type, value) VALUES (?,?,?,?)", [id, 0, key, val]) } $db.execute("INSERT INTO patternassignments (ID, AssID, type, value) VALUES (?,?,?,?)", [id, 0, "STATION", form["station"]]) $db.execute("INSERT INTO patternassignments (ID, AssID, type, value) VALUES (?,?,?,?)", [id, 0, "TEXT", form["assignmentText"]]) ret = {:id => id} end Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) else #pp "Im Else" maxID = $db.execute "SELECT MAX(AssID) FROM patternassignments WHERE ID ='#{checkimage[0][0]}'" if(maxID[0][0] == nil) maxID = 0 else maxID = maxID[0][0] +1 end #Search 4 assignment if none found add assignment #pp ListAssignmentSearch::prepare(@a[0][:appconf]["wasurl"], pattern.join('.'), form["station"], form["assignmentText"]) if (!ListAssignmentSearch::prepare(@a[0][:appconf]["wasurl"], pattern.join('.'), form["station"], form["assignmentText"]).any?) form["parts"].each{ |key,val| $db.execute("INSERT INTO patternassignments (ID, AssID, type, value) VALUES (?,?,?,?)", [checkimage[0][0], maxID, key, val]) } $db.execute("INSERT INTO patternassignments (ID, AssID, type, value) VALUES (?,?,?,?)", [checkimage[0][0], maxID, "STATION", form["station"]]) $db.execute("INSERT INTO patternassignments (ID, AssID, type, value) VALUES (?,?,?,?)", [checkimage[0][0], maxID, "TEXT", form["assignmentText"]]) #pp "Added Assignment" ret = {:id => checkimage[0][0]} end Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end end class CreateSymlinks < Riddl::Implementation def response pp "Creating Symlinks" #GETTING ALL Assignments ret = {} ret2 = {} resultstation = $db.execute "SELECT * FROM assignments" resultstation.each do |row| #row[0] pattern ID #row[1] pattern image #if limiting to a specific assignment ID #if row[0] != 627 #164 = battery # next #end #Getting Assignments for ID: ret2 = {} allAssIds = $db.execute "SELECT DISTINCT AssID FROM patternassignments WHERE ID = #{row[0]}" allAssIds.each do |row2| station = $db.execute "SELECT value FROM patternassignments WHERE ID = #{row[0]} AND AssID = #{row2[0]} AND type == 'STATION'" stationvalue = Integer(station[0][0]) singleItem = $db.execute "SELECT type, value FROM patternassignments WHERE ID = #{row[0]} AND AssID = #{row2[0]} AND type != 'STATION'" if(ret2[stationvalue] == nil) ret2[stationvalue] = Hash.new() end ret2[stationvalue][row2[0]] = Hash.new() singleItem.each do |it| ret2[stationvalue][row2[0]][it[0]] = it[1] end link = "Station" + stationvalue.to_s ret2[stationvalue][row2[0]].each do |key, value| if(value != "") if(key != "TEXT") link = link + "_" + key.to_s + ";" + value.to_s else link = link + "_" + "ID" + ";" + value.to_s end end end link = link + ".jpg" #pp link imgfile = File.join(row[1]) symfile = File.join(@a[0][:appconf]["universalstorageeinzelbilder"], "Symlinks", link) #pp "imgfile " + imgfile #pp "symfile " + symfile #Create Symlinks for each Assignment if File.file?(imgfile) #file exists #pp "File exists" if !File.symlink?(symfile) #symlink should not exist pp "Creating Symlink" nw = "" + row[1] nw.slice! @a[0][:appconf]["universalstorageeinzelbilder"].to_s #pp "../" + nw #pp symfile #pp "actual link creation"; # nw = actual image, symfile = symlink location File.symlink "../" + nw, symfile else #pp "Symlink exists" end end end end #Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret2)) end end class GetAssignment < Riddl::Implementation def response ret = {} allAssIds = $db.execute "SELECT DISTINCT AssID FROM patternassignments WHERE ID = #{@r[1]}" allAssIds.each do |row| station = $db.execute "SELECT value FROM patternassignments WHERE ID = #{@r[1]} AND AssID = #{row[0]} AND type == 'STATION'" stationvalue = Integer(station[0][0]) #old where pattern is stored as string #singleItem = $db.execute "SELECT value FROM patternassignments WHERE ID = #{@r[1]} AND AssID = #{row[0]} AND type != 'STATION'" #if(ret[stationvalue] == nil) # ret[stationvalue] = Hash.new() #end #ret[stationvalue][row[0]] = singleItem.join('.') singleItem = $db.execute "SELECT type, value FROM patternassignments WHERE ID = #{@r[1]} AND AssID = #{row[0]} AND type != 'STATION'" if(ret[stationvalue] == nil) ret[stationvalue] = Hash.new() end ret[stationvalue][row[0]] = Hash.new() singleItem.each do |it| ret[stationvalue][row[0]][it[0]] = it[1] end end Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end class RenameAssignmentImagePath < Riddl::Implementation def response $db.execute("UPDATE assignments SET image = '#{JSON.parse(@p[0].value.read)["url"]}' WHERE ID = #{@r[1]}") end end class DeleteAssignment < Riddl::Implementation def response $db.execute("DELETE FROM assignments WHERE ID = ?", [@r[1]]) end end class DeleteSingleAssignment < Riddl::Implementation def response $db.execute("DELETE FROM patternassignments WHERE AssID = ? AND ID = ?", [@r[2], @r[1]]) end end class ListAssignmentSearch < Riddl::Implementation def self::prepare(appconf, pat, pat_station, pat_value) #https://centurio.work/customers/evva/was/ui/TransformProductCode.php?ProductCode=E.A.PZ.DZ-S.31.31.MP.SEP pattern = JSON.parse(Net::HTTP.get URI(appconf + "ui/TransformProductCode.php?ProductCode=" + pat)) #remove empty entries from pattern pattern.each do |key, value| if(value == "") if(key == "Zi") elsif (key == "Kf") else pattern.delete(key) end end end emptypatternparts = [] #fullcode = JSON.parse( File.read(File.join("json" , "ProductCode.json"))) #puts File.join(File.dirname(__dir__),"server/json/ProductCode.json") fullcode = JSON.parse( File.read(File.join(File.dirname(__dir__),"server/json/ProductCode.json"))) fullcode["pattern"].each do |key| if(!pattern.has_key?(key["abk"]) ) emptypatternparts.push(key["abk"]) end end #$db.execute "CREATE TABLE IF NOT EXISTS patternassignments(ID INT, AssID INT, type CHARACTER(20), value CHARACTER(20), FOREIGN KEY(ID) REFERENCES assignments(ID) ON DELETE CASCADE)" #Get first key dynamicly and then build sql query if(pattern.keys[0] != nil) searchstring = "SELECT " + pattern.keys[0] + ".ID AS ID" searchstring = searchstring + " FROM " + pattern.map{ |key, val| "patternassignments " + key}.join(', ') searchstring = searchstring + ", " + emptypatternparts.map{ |key| "patternassignments " + key}.join(', ') searchstring = searchstring + " , patternassignments STATION , patternassignments ValText" searchstring = searchstring + " WHERE " + pattern.map{ |key, val| pattern.keys[0] + ".AssID == " + key + ".AssID"}.join(' AND '); searchstring = searchstring + " AND " + pattern.map{ |key, val| pattern.keys[0] + ".ID == " + key + ".ID"}.join(' AND ') searchstring = searchstring + " AND " + emptypatternparts.map{ |key| pattern.keys[0] + ".AssID == " + key + ".AssID"}.join(' AND ') searchstring = searchstring + " AND " + emptypatternparts.map{ |key| pattern.keys[0] + ".ID == " + key + ".ID"}.join(' AND ') searchstring = searchstring + " AND " + emptypatternparts.map{ |key| key + ".type == '" + key + "' AND " + key + ".value == ''" }.join(' AND ') searchstring = searchstring + " AND " + pattern.keys[0] + ".AssID == STATION.AssID" searchstring = searchstring + " AND " + pattern.keys[0] + ".AssID == ValText.AssID" searchstring = searchstring + " AND " + pattern.keys[0] + ".ID == STATION.ID" searchstring = searchstring + " AND " + pattern.keys[0] + ".ID == ValText.ID" searchstring = searchstring + " AND STATION.type == 'STATION'" + " AND STATION.value == '" + pat_station + "'" searchstring = searchstring + " AND ValText.type == 'TEXT'" + " AND ValText.value == '" + pat_value + "'" searchstring = searchstring + " AND " + pattern.map{ |key, val| #In case a value has the option "KeineAngabe" empty values are allowed if(val.start_with?("KeineAngabe")) key + ".type == '" + key + "' AND (" + key + ".value == '"+ val +"' or " + key + ".value == '*' or " + key + ".value == '')" else key + ".type == '" + key + "' AND (" + key + ".value == '"+ val +"' or " + key + ".value == '*')" end }.join(' AND ') + " ORDER BY ID ASC" #puts searchstring; else searchstring = "SELECT * FROM " searchstring = searchstring + "patternassignments STATION , patternassignments ValText" searchstring = searchstring + " WHERE" searchstring = searchstring + " STATION.type == 'STATION'" + " AND STATION.value == '" + pat_station + "'" searchstring = searchstring + " AND ValText.type == 'TEXT'" + " AND ValText.value == '" + pat_value + "'" searchstring = searchstring + " AND ValText.AssID == STATION.AssID" searchstring = searchstring + " ORDER BY ID ASC" end #pp searchstring result = $db.execute searchstring; #count entries #result2 = $db.execute "SELECT COUNT(*) FROM patternassignments"; #pp result2; #Search for images with one or multiple found patternIDs ret = [] if result.length > 0 result.each do |item| result2 = $db.execute "SELECT DISTINCT image FROM assignments WHERE ID = #{item[0]}" result2.each do |item2| ret << item2[0] end end end return ret end def response ret = ListAssignmentSearch::prepare(@a[0][:appconf]["wasurl"], @p[0].value, @p[1].value, @p[2].value) Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end class ListAssignmentSearchMultiple < Riddl::Implementation def response #pattern station text pattern = JSON.parse(@p[0].value) station = JSON.parse(@p[1].value) text = JSON.parse(@p[2].value) ret = [] pattern.each_with_index {|val, index| #puts "#{val} => #{index} : #{station[index]} : #{text[index]}" ret = ret.push(ListAssignmentSearch::prepare(@a[0][:appconf]["wasurl"], val, station[index], text[index])) } #ret = ListAssignmentSearch::prepare(@a[0][:appconf]["wasurl"], @p[0].value, @p[1].value, @p[2].value) Riddl::Parameter::Complex.new('list','application/json',JSON::pretty_generate(ret)) end end def createDB(opts) $db.execute("PRAGMA foreign_keys=ON"); $db.execute "CREATE TABLE IF NOT EXISTS stations(station INT, name VARCHAR(20), PRIMARY KEY(station))" opts[:appconf]["stations"].each { |n| CreateStation::createDB(n) } #vor DB struktur umstellung #$db.execute "CREATE TABLE IF NOT EXISTS station(station INT, patternID INT, pattern CHARACTER(256), description TEXT, date TEXT, P0 VARCHAR(10), P1 VARCHAR(10), P2 VARCHAR(10), P3 VARCHAR(10), P4 VARCHAR(10), P5 VARCHAR(10), P6 VARCHAR(10), P7 VARCHAR(10), P8 VARCHAR(10), P9 VARCHAR(10), P10 VARCHAR(10), PRIMARY KEY(station,patternID), UNIQUE(station, patternID))" #$db.execute "CREATE TABLE IF NOT EXISTS error(station INT, patternID INT, error TEXT, FOREIGN KEY(station, patternID) REFERENCES station(station, patternID) ON DELETE CASCADE)" #$db.execute "CREATE TABLE IF NOT EXISTS replacements(station INT, patternID INT, abbreviation TEXT, url TEXT, ordering INT, FOREIGN KEY(station, patternID) REFERENCES station(station, patternID) ON DELETE CASCADE)" #$db.execute "CREATE TABLE IF NOT EXISTS image(station INT, patternID INT, imageID INT, language CHARACTER(20), label TEXT, FOREIGN KEY(station, patternID) REFERENCES station(station, patternID) ON DELETE CASCADE,PRIMARY KEY(station,patternID,imageID,language))" $db.execute "CREATE TABLE IF NOT EXISTS stationpattern(station INT, internalID INT, description TEXT, date TEXT, PRIMARY KEY(internalID), UNIQUE(internalID))" $db.execute "CREATE TABLE IF NOT EXISTS pattern(internalID INT, type CHARACTER(20), value CHARACTER(20), FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" $db.execute "CREATE TABLE IF NOT EXISTS error(internalID INT, error TEXT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" $db.execute "CREATE TABLE IF NOT EXISTS replacements(internalID INT, abbreviation TEXT, url TEXT, ordering INT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" $db.execute "CREATE TABLE IF NOT EXISTS image(internalID INT, imageID INT, language CHARACTER(20), label TEXT, FOREIGN KEY(internalID) REFERENCES stationpattern(internalID) ON DELETE CASCADE)" $db.execute "CREATE TABLE IF NOT EXISTS assignments(ID INT, image TEXT, PRIMARY KEY(ID) UNIQUE (ID))" $db.execute "CREATE TABLE IF NOT EXISTS patternassignments(ID INT, AssID INT, type CHARACTER(20), value CHARACTER(20), FOREIGN KEY(ID) REFERENCES assignments(ID) ON DELETE CASCADE)" end server = Riddl::Server.new(File.join(__dir__,'/was.xml'), :host => 'localhost') do |opts| accessible_description true cross_site_xhr true createDB(opts) # opts[:appconf] on resource do run GetStations if get run CreateStation if post on resource 'cutimage' do run CutImage if get #should add deleteReplacement end on resource 'json' do run GetJsonFiles if get on resource '.*' do run SaveJson if put 'list' run GetJson if get end #should add deleteReplacement end on resource '\d+' do run GetStation if get run CreatePattern if post 'pattern' on resource '\d+' do run GetPattern if get run UpdatePattern if put 'pattern' run DeletePattern if delete run DuplicatePattern if put 'patternID' on resource 'error' do run SaveError if put 'error' run GetError if get #should add deleteerror end on resource 'replacement' do run SaveReplacement if put 'replacement' run GetReplacement if get #should add deleteReplacement end on resource 'images' do run GetImages, opts if get #run CreateImage if post #'image' run AddExternalImage if post 'externalImage' run UploadImage if post on resource 'reorder' do on resource do run ReorderImages if put 'orderlist' end end on resource '\d+' do run GetImage, opts if get on resource do run UpdateImageLabel if put 'label' run GetRealImage if get run DeleteImage if delete end end end end on resource 'search' do run ListSearch if get on resource 'imagesWIZ' do run SearchImages, opts if get on resource '\d+' do run SearchImagesSingle, opts if get end end on resource 'errorsWIZ' do run SearchErrors, opts if get end end end on resource 'assignments' do run GetAssignments if get run SaveNewAssignment, opts if post 'list' on resource 'search' do run ListAssignmentSearch, opts if get end on resource 'searchMultiple' do run ListAssignmentSearchMultiple, opts if post end on resource 'createSymlinks' do run CreateSymlinks, opts if get end on resource '\d+' do run GetAssignment if get run RenameAssignmentImagePath if put 'list' run DeleteAssignment if delete on resource '\d+' do run DeleteSingleAssignment if delete end end end end end.loop!