image.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. //云端图片操作工具
  2. const EventEmitter = require('events')
  3. const api = require('./api.js')
  4. const env = require('./env.js')
  5. const config = require('./config.js')
  6. const logger = require('./logger.js')('image.js')
  7. const downloadLogger = require('./logger.js')('download')
  8. const upyun = require('./upyun.js')
  9. const PromisePool = require('./promise-pool.js')
  10. const fs = require('fs')
  11. const path = require('path')
  12. const readline = require('readline')
  13. const sizeOf = require('image-size')
  14. const mustache = require('mustache')
  15. const mkdirp = require('mkdirp')
  16. const moment = require('moment')
  17. const gm = config.imagemagick != undefined ? require('gm').subClass({
  18. imageMagick: true,
  19. appPath: config.imagemagick
  20. }) : require('gm')
  21. class executor extends EventEmitter {
  22. async readFile(file) {
  23. return new Promise(resolve => {
  24. var data = []
  25. if (fs.existsSync(file)) {
  26. let reader = readline.createInterface({
  27. input: fs.createReadStream(file)
  28. })
  29. reader.on('line', line => {
  30. data.push(line)
  31. })
  32. reader.on('close', () => {
  33. resolve(data)
  34. })
  35. } else {
  36. resolve(data)
  37. }
  38. })
  39. }
  40. async checkFile(url, client) {
  41. let size = sizeOf(await client.download(url))
  42. if (size.width == 0 || size.height == 0) {
  43. throw 'invalid image data:' + url
  44. }
  45. }
  46. async addWatermark(image, file, student, index, showMarker) {
  47. let fontFile = config.watermark.fontFile
  48. let color = config.watermark.color
  49. let imgData = gm(image)
  50. let size = sizeOf(image)
  51. //添加第一页的得分明细
  52. if (index == 1) {
  53. //初始坐标
  54. let x = 30
  55. let y = 10
  56. //最大宽/高限制
  57. let fontSize = 30
  58. let maxX = size.width / 2 - x * 2
  59. let height = fontSize + 10
  60. //计算总分
  61. let totalScore = (parseFloat(student.objectiveScore) || 0) + (parseFloat(student.subjectiveScore) || 0)
  62. //显示总分明细
  63. imgData.font(fontFile, fontSize).fill(color)
  64. imgData.drawText(x, y += height, '成绩明细')
  65. imgData.drawText(x, y += height, '总分=(客观+主观) | ' + totalScore + '=' + student.objectiveScore + '+' + student.subjectiveScore)
  66. //显示客观题明细
  67. if (student.objectiveScoreDetail && student.objectiveScoreDetail.length > 0) {
  68. let lines = []
  69. let array = []
  70. //前置提示文字的字符数
  71. let count = 10
  72. lines.push(array)
  73. for (let i = 0; i < student.objectiveScoreDetail.length; i++) {
  74. let detail = student.objectiveScoreDetail[i]
  75. let content = detail.answer + ':' + detail.score
  76. //超长后另起一行显示客观题
  77. if ((count + content.length) * fontSize * 0.7 > maxX) {
  78. array = []
  79. lines.push(array)
  80. count = 10
  81. }
  82. array.push(content)
  83. count += content.length
  84. }
  85. //显示所有行的客观题明细
  86. for (let l = 0; l < lines.length; l++) {
  87. imgData.drawText(x, y += height, '客观题识别结果 | ' + lines[l].join(';'))
  88. }
  89. }
  90. //显示主观题明细
  91. if (student.subjectiveScoreDetail && student.subjectiveScoreDetail.length > 0) {
  92. let title = '主观题号 | 分数'
  93. if (showMarker) {
  94. title += ' | 评卷员'
  95. }
  96. let startY = y
  97. let width = title.length * fontSize
  98. imgData.drawText(x, y += height, title)
  99. for (let i = 0; i < student.subjectiveScoreDetail.length; i++) {
  100. let detail = student.subjectiveScoreDetail[i]
  101. //超过最大高度了则另起一列
  102. if ((y + height + 15) > size.height) {
  103. y = startY
  104. x += width
  105. imgData.drawText(x, y += height, title)
  106. }
  107. let content = detail.mainNumber + '-' + detail.subNumber + ' : ' + detail.score
  108. if (showMarker) {
  109. content = content + ' ' + (detail.marker || detail.markerName || '')
  110. }
  111. width = Math.max(width, content.length * fontSize)
  112. imgData.drawText(x, y += height, content)
  113. }
  114. }
  115. }
  116. //显示评卷标记
  117. if (student.tags != undefined && student.tags[index] != undefined) {
  118. let fontSize = 60
  119. let height = fontSize + 10
  120. imgData.font(fontFile, fontSize).fill(color)
  121. let tags = student.tags[index]
  122. for (let i = 0; i < tags.length; i++) {
  123. let tag = tags[i]
  124. if (tag.content != undefined) {
  125. let top = tag.top
  126. for (let j = 0; j < tag.content.length; j++) {
  127. imgData.drawText(tag.left, top, tag.content[j])
  128. top += height
  129. }
  130. }
  131. }
  132. }
  133. return new Promise((resolve, reject) => {
  134. imgData.write(file, error => {
  135. if (error) {
  136. logger.error('add watermark error: ' + file)
  137. logger.error(error)
  138. reject(error)
  139. } else {
  140. resolve()
  141. }
  142. })
  143. })
  144. }
  145. async downloadFile(append, remoteTemplate, localTemplate, data, dir, client, bucket, index, watermark, showMarker) {
  146. data.index = index
  147. let remote = mustache.render(remoteTemplate, data)
  148. let local = path.join(dir, mustache.render(localTemplate, data))
  149. mkdirp.sync(path.dirname(local))
  150. //续传模式下,判断目标文件是否存在,存在则直接跳过
  151. if (append && fs.existsSync(local)) {
  152. return Promise.resolve()
  153. } else {
  154. let imgData
  155. if (config.localStore != undefined && config.localStore.length > 0) {
  156. let cache = path.join(config.localStore, bucket, remote)
  157. if (fs.existsSync(cache)) {
  158. imgData = fs.readFileSync(cache)
  159. }
  160. }
  161. if (imgData == undefined) {
  162. try {
  163. imgData = await client.download(remote)
  164. } catch (err) {
  165. if (err.code === 404) {
  166. //文件不存在,记录日志并跳过
  167. downloadLogger.error('404 ' + bucket + ' ' + remote)
  168. return Promise.resolve()
  169. } else {
  170. logger.error(err)
  171. return Promise.reject(err)
  172. }
  173. }
  174. }
  175. //是否需要添加分数水印
  176. if (watermark) {
  177. return this.addWatermark(imgData, local, data, index, showMarker)
  178. } else {
  179. return new Promise((resolve, reject) => {
  180. fs.writeFile(local, imgData, err => {
  181. if (err) {
  182. logger.error('write image file error: ' + local)
  183. logger.error(err)
  184. reject(err)
  185. } else {
  186. resolve()
  187. }
  188. })
  189. })
  190. }
  191. }
  192. }
  193. async downloadSheet(dir, template, append, failover, watermark, showMarker) {
  194. let bucket = env.server.bucketPrefix + '-sheet'
  195. let client = upyun(bucket, config.upyun.operator, config.upyun.password)
  196. if (env.server.upyunDomain && env.server.upyunDomain != '') {
  197. //局域网模式修改图片服务器地址
  198. client.setDomain(env.server.upyunDomain)
  199. }
  200. try {
  201. let totalCount = await api.countStudents(env.examId, true, undefined)
  202. this.emit('total', totalCount)
  203. let count = 0
  204. let pageNumber = 0
  205. this.emit('count', 0)
  206. for (;;) {
  207. pageNumber++
  208. let array = await api.getStudents(env.examId, pageNumber, 100, true, undefined, watermark === true, watermark === true)
  209. if (array == undefined || array.length == 0) {
  210. break
  211. }
  212. for (let i = 0; i < array.length; i++) {
  213. let promises = []
  214. let student = array[i]
  215. student.examId = env.examId
  216. for (let i = 1; i <= student.sheetCount; i++) {
  217. promises.push(this.downloadFile(append, config.imageUrl.sheet, template, student, dir, client, bucket, i, watermark, showMarker))
  218. }
  219. try {
  220. //等待所有图片下载完毕
  221. await Promise.all(promises)
  222. count++
  223. this.emit('count', count)
  224. } catch (err) {
  225. //判断是否异常终止
  226. if (failover) {
  227. throw err
  228. } else {
  229. logger.error('download sheet error:' + err)
  230. logger.error(err)
  231. continue
  232. }
  233. }
  234. }
  235. }
  236. this.emit('finish')
  237. } catch (error) {
  238. logger.error('download sheet error:' + error)
  239. logger.error(error)
  240. this.emit('error', error)
  241. }
  242. }
  243. async downloadPackage(dir, template, append, failover) {
  244. let bucket = env.server.bucketPrefix + '-package'
  245. let client = upyun(bucket, config.upyun.operator, config.upyun.password)
  246. if (env.server.upyunDomain && env.server.upyunDomain != '') {
  247. client.setDomain(env.server.upyunDomain)
  248. }
  249. try {
  250. let array = await api.getPackages(env.examId, true)
  251. this.emit('total', array.length)
  252. let count = 0
  253. this.emit('count', 0)
  254. for (let i = 0; i < array.length; i++) {
  255. let p = array[i]
  256. p.examId = env.examId
  257. for (let i = 1; i <= p.picCount; i++) {
  258. try {
  259. await this.downloadFile(append, config.imageUrl.package, template, p, dir, client, bucket, i)
  260. } catch (err) {
  261. //判断是否异常终止
  262. if (failover) {
  263. throw err
  264. } else {
  265. logger.error('download package error: ' + err)
  266. logger.error(err)
  267. continue
  268. }
  269. }
  270. }
  271. count++
  272. this.emit('count', count)
  273. }
  274. this.emit('finish')
  275. } catch (error) {
  276. logger.error('download package error: ' + error)
  277. logger.error(error)
  278. this.emit('error', error)
  279. }
  280. }
  281. async checkSlice(dir, concurrent) {
  282. let bucket = env.server.bucketPrefix + '-slice'
  283. let client = upyun(bucket, config.upyun.operator, config.upyun.password)
  284. if (env.server.upyunDomain && env.server.upyunDomain != '') {
  285. //局域网模式修改图片服务器地址
  286. client.setDomain(env.server.upyunDomain)
  287. }
  288. try {
  289. let logFile = path.join(dir, 'result.txt')
  290. fs.writeFileSync(logFile, moment().format('YYYY-MM-DD HH:mm:ss') + ', examId=' + env.examId + '\r\n')
  291. let totalCount = await api.countStudents(env.examId, true, false)
  292. this.emit('total', totalCount)
  293. let self = this
  294. let count = 0
  295. let pageNumber = 0
  296. let pool = PromisePool.create(concurrent, function(student) {
  297. student.examId = env.examId
  298. student.promises = []
  299. for (let i = 1; i <= student.sliceCount; i++) {
  300. student.index = i
  301. let url = mustache.render(config.imageUrl.slice, student)
  302. student.promises.push(new Promise(resolved => {
  303. self.checkFile(url, client).then(() => {
  304. //fs.appendFileSync(logFile, url + ': success\r\n')
  305. }).catch(() => {
  306. fs.appendFileSync(logFile, url + ': error\r\n')
  307. }).finally(() => {
  308. resolved()
  309. })
  310. }))
  311. }
  312. return Promise.all(student.promises)
  313. })
  314. pool.on('count', function(offset) {
  315. self.emit('count', count + offset)
  316. })
  317. this.emit('count', 0)
  318. for (;;) {
  319. pageNumber++
  320. let array = await api.getStudents(env.examId, pageNumber, 200, true, false, false, false)
  321. if (array == undefined || array.length == 0) {
  322. break
  323. }
  324. await pool.start(array)
  325. count += array.length
  326. }
  327. this.emit('finish')
  328. fs.appendFileSync(logFile, moment().format('YYYY-MM-DD HH:mm:ss') + ', examId=' + env.examId)
  329. } catch (error) {
  330. logger.error('check slice error:' + error)
  331. logger.error(error)
  332. this.emit('error', error)
  333. }
  334. }
  335. async checkSliceSerial(dir) {
  336. let bucket = env.server.bucketPrefix + '-slice'
  337. let client = upyun(bucket, config.upyun.operator, config.upyun.password)
  338. if (env.server.upyunDomain && env.server.upyunDomain != '') {
  339. //局域网模式修改图片服务器地址
  340. client.setDomain(env.server.upyunDomain)
  341. }
  342. try {
  343. let logFile = path.join(dir, 'result.txt')
  344. fs.writeFileSync(logFile, moment().format('YYYY-MM-DD HH:mm:ss') + ', examId=' + env.examId + '\r\n')
  345. let totalCount = await api.countStudents(env.examId, true, false)
  346. this.emit('total', totalCount)
  347. let self = this
  348. let count = 0
  349. let pageNumber = 0
  350. this.emit('count', 0)
  351. for (;;) {
  352. pageNumber++
  353. let array = await api.getStudents(env.examId, pageNumber, 100, true, false, false, false)
  354. if (array == undefined || array.length == 0) {
  355. break
  356. }
  357. for (let i = 0; i < array.length; i++) {
  358. let student = array[i]
  359. student.examId = env.examId
  360. student.promises = []
  361. for (let i = 1; i <= student.sliceCount; i++) {
  362. student.index = i
  363. let url = mustache.render(config.imageUrl.slice, student)
  364. student.promises.push(new Promise(resolved => {
  365. self.checkFile(url, client).then(() => {
  366. //fs.appendFileSync(logFile, url + ': success\r\n')
  367. }).catch(() => {
  368. fs.appendFileSync(logFile, url + ': error\r\n')
  369. }).finally(() => {
  370. resolved()
  371. })
  372. }))
  373. }
  374. await Promise.all(student.promises)
  375. count++
  376. this.emit('count', count)
  377. }
  378. }
  379. this.emit('finish')
  380. fs.appendFileSync(logFile, moment().format('YYYY-MM-DD HH:mm:ss') + ', examId=' + env.examId)
  381. } catch (error) {
  382. logger.error('check slice error:' + error)
  383. logger.error(error)
  384. this.emit('error', error)
  385. }
  386. }
  387. }
  388. module.exports = function() {
  389. return new executor()
  390. }