WebdavService.js 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. /*
  2. * WebdavService.js
  3. *
  4. * @author realor
  5. */
  6. import { FileService, Metadata, Result } from './FileService.js'
  7. import { ServiceManager } from './ServiceManager.js'
  8. import { WebUtils } from '../utils/WebUtils.js'
  9. class WebdavService extends FileService {
  10. constructor(name, description, url, username, password) {
  11. super(name, description, url, username, password)
  12. }
  13. open(path, readyCallback, progressCallback) {
  14. const OK = Result.OK
  15. const ERROR = Result.ERROR
  16. const COLLECTION = Metadata.COLLECTION
  17. const FILE = Metadata.FILE
  18. let url = this.getUrl(path)
  19. let baseUri = url
  20. let index = baseUri.indexOf('://')
  21. if (index !== -1) {
  22. baseUri = baseUri.substring(index + 3)
  23. index = baseUri.indexOf('/')
  24. if (index !== -1) {
  25. baseUri = baseUri.substring(index)
  26. if (baseUri.lastIndexOf('/') !== baseUri.length - 1) {
  27. baseUri += '/'
  28. }
  29. }
  30. }
  31. // **** HTTP PROPFIND REQUEST ****
  32. let request = new XMLHttpRequest()
  33. request.onerror = () => {
  34. // ERROR
  35. readyCallback(new Result(ERROR, 'Connection error'))
  36. }
  37. request.onload = () => {
  38. if (request.status === 200 || request.status === 207) {
  39. try {
  40. // OK
  41. let xml = request.responseXML
  42. let multiNode = xml.childNodes[0]
  43. let responseNodes = multiNode.childNodes
  44. let metadata = new Metadata()
  45. let entries = []
  46. for (let i = 0; i < responseNodes.length; i++) {
  47. let responseNode = responseNodes[i]
  48. if (responseNode.localName === 'response') {
  49. let hrefNode = responseNode.querySelector('href')
  50. let hrefValue = hrefNode.textContent
  51. let fileName = hrefValue.substring(baseUri.length)
  52. let isCollectionNode = responseNode.querySelector('propstat prop resourcetype collection') !== null
  53. let contentLengthNode = responseNode.querySelector('propstat prop getcontentlength')
  54. let lastModifiedNode = responseNode.querySelector('propstat prop getlastmodified')
  55. let fileSize = contentLengthNode ? parseInt(contentLengthNode.textContent) : 0
  56. let lastModified = lastModifiedNode ? parseInt(lastModifiedNode.textContent) : 0
  57. if (fileName.indexOf('/') === 0) fileName = fileName.substring(1)
  58. if (fileName.length === 0) {
  59. // requested resource
  60. let index = hrefValue.lastIndexOf('/')
  61. metadata.name = hrefValue.substring(index + 1)
  62. metadata.description = metadata.name
  63. metadata.type = isCollectionNode ? COLLECTION : FILE
  64. metadata.size = fileSize
  65. metadata.lastModified = lastModified
  66. } else {
  67. let entry = new Metadata(fileName, fileName, isCollectionNode ? COLLECTION : FILE, fileSize, lastModified)
  68. entries.push(entry)
  69. }
  70. }
  71. }
  72. if (metadata.type === COLLECTION) {
  73. readyCallback(new Result(OK, '', path, metadata, entries, null))
  74. } // download file
  75. else {
  76. let request = new XMLHttpRequest()
  77. request.open('GET', url, true)
  78. request.onload = () => {
  79. if (request.status === 200) {
  80. if (progressCallback) {
  81. progressCallback({ progress: 100, message: 'Download completed.' })
  82. }
  83. setTimeout(() => readyCallback(new Result(OK, '', path, metadata, null, request.response)), 100)
  84. }
  85. }
  86. request.onerror = error => {
  87. readyCallback(new Result(ERROR, error))
  88. }
  89. if (progressCallback) {
  90. request.onprogress = event => {
  91. let progress = Math.round((100 * event.loaded) / event.total)
  92. let message = 'Downloading file...'
  93. progressCallback({ progress: progress, message: message })
  94. }
  95. }
  96. WebUtils.setBasicAuthorization(request, this.username, this.password)
  97. request.send()
  98. }
  99. } catch (ex) {
  100. readyCallback(new Result(ERROR, ex))
  101. }
  102. } else {
  103. readyCallback(this.createError("Can't open", request.status))
  104. }
  105. }
  106. request.open('PROPFIND', url, true)
  107. request.setRequestHeader('depth', '1')
  108. WebUtils.setBasicAuthorization(request, this.username, this.password)
  109. request.send()
  110. }
  111. save(path, data, readyCallback, progressCallback) {
  112. const OK = Result.OK
  113. const ERROR = Result.ERROR
  114. const url = this.getUrl(path)
  115. const request = new XMLHttpRequest()
  116. request.onerror = error => {
  117. // ERROR
  118. readyCallback(new Result(ERROR, 'Connection error'))
  119. }
  120. request.onload = () => {
  121. if (request.status === 200) {
  122. readyCallback(new Result(OK))
  123. } else {
  124. readyCallback(this.createError('Save failed', request.status))
  125. }
  126. }
  127. if (progressCallback) {
  128. request.onprogress = event => {
  129. let progress = Math.round((100 * event.loaded) / event.total)
  130. let message = 'Uploading file...'
  131. progressCallback({ progress: progress, message: message })
  132. }
  133. }
  134. request.open('PUT', url, true)
  135. WebUtils.setBasicAuthorization(request, this.username, this.password)
  136. request.send(data)
  137. }
  138. remove(path, readyCallback, progressCallback) {
  139. const OK = Result.OK
  140. const ERROR = Result.ERROR
  141. const url = this.getUrl(path)
  142. const request = new XMLHttpRequest()
  143. request.onerror = error => {
  144. // ERROR
  145. readyCallback(new Result(ERROR, 'Connection error'))
  146. }
  147. request.onload = () => {
  148. if (request.status === 200) {
  149. readyCallback(new Result(OK))
  150. } else {
  151. readyCallback(this.createError('Delete failed', request.status))
  152. }
  153. }
  154. request.open('DELETE', url, true)
  155. WebUtils.setBasicAuthorization(request, this.username, this.password)
  156. request.send()
  157. }
  158. makeCollection(path, readyCallback, progressCallback) {
  159. const OK = Result.OK
  160. const ERROR = Result.ERROR
  161. const url = this.getUrl(path)
  162. const request = new XMLHttpRequest()
  163. request.onerror = error => {
  164. readyCallback(new Result(ERROR, 'Connection error'))
  165. }
  166. request.onload = () => {
  167. if (request.status === 200 || request.status === 201) {
  168. readyCallback(new Result(OK))
  169. } else {
  170. readyCallback(this.createError('Folder creation failed', request.status))
  171. }
  172. }
  173. request.open('MKCOL', url, true)
  174. WebUtils.setBasicAuthorization(request, this.username, this.password)
  175. request.send()
  176. }
  177. createError(message, status) {
  178. let statusMessage = WebUtils.getHttpStatusMessage(status)
  179. if (statusMessage.length > 0) {
  180. message += ': ' + statusMessage
  181. }
  182. message += ' (HTTP ' + status + ').'
  183. let resultStatus
  184. switch (status) {
  185. case 401:
  186. resultStatus = Result.INVALID_CREDENTIALS
  187. break
  188. case 403:
  189. resultStatus = Result.FORBIDDEN
  190. break
  191. default:
  192. resultStatus = Result.ERROR
  193. }
  194. return new Result(resultStatus, message)
  195. }
  196. getUrl(path) {
  197. let url = this.url || ''
  198. if (url && url.endsWith('/')) {
  199. url = url.substring(0, url.length - 1)
  200. }
  201. if (path && !path.startsWith('/')) {
  202. path = '/' + path
  203. }
  204. return url + path
  205. }
  206. }
  207. ServiceManager.addClass(WebdavService)
  208. export { WebdavService }