bucket.js 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. const assert = require('assert');
  2. const { isArray } = require('./common/utils/isArray');
  3. const { checkBucketName: _checkBucketName } = require('../lib/common/utils/checkBucketName');
  4. const { formatTag } = require('../lib/common/utils/formatTag');
  5. const proto = exports;
  6. function toArray(obj) {
  7. if (!obj) return [];
  8. if (isArray(obj)) return obj;
  9. return [obj];
  10. }
  11. /**
  12. * Bucket opertaions
  13. */
  14. proto.listBuckets = async function listBuckets(query = {}, options = {}) {
  15. // prefix, marker, max-keys
  16. const { subres = {} } = query;
  17. const rest = {};
  18. for (const key in query) {
  19. if (key !== 'subres') {
  20. rest[key] = query[key];
  21. }
  22. }
  23. const params = this._bucketRequestParams(
  24. 'GET',
  25. '',
  26. Object.assign(subres, options.subres),
  27. options
  28. );
  29. params.query = rest;
  30. const result = await this.request(params);
  31. if (result.status === 200) {
  32. const data = await this.parseXML(result.data);
  33. let buckets = data.Buckets || null;
  34. if (buckets) {
  35. if (buckets.Bucket) {
  36. buckets = buckets.Bucket;
  37. }
  38. if (!isArray(buckets)) {
  39. buckets = [buckets];
  40. }
  41. buckets = buckets.map(item => ({
  42. name: item.Name,
  43. region: item.Location,
  44. creationDate: item.CreationDate,
  45. StorageClass: item.StorageClass,
  46. tag: formatTag(item)
  47. }));
  48. }
  49. return {
  50. buckets,
  51. owner: {
  52. id: data.Owner.ID,
  53. displayName: data.Owner.DisplayName
  54. },
  55. isTruncated: data.IsTruncated === 'true',
  56. nextMarker: data.NextMarker || null,
  57. res: result.res
  58. };
  59. }
  60. throw await this.requestError(result);
  61. };
  62. proto.useBucket = function useBucket(name) {
  63. _checkBucketName(name);
  64. return this.setBucket(name);
  65. };
  66. proto.setBucket = function useBucket(name) {
  67. _checkBucketName(name);
  68. this.options.bucket = name;
  69. return this;
  70. };
  71. proto.getBucket = function getBucket() {
  72. return this.options.bucket;
  73. };
  74. proto.getBucketLocation = async function getBucketLocation(name, options) {
  75. _checkBucketName(name);
  76. name = name || this.getBucket();
  77. const params = this._bucketRequestParams('GET', name, 'location', options);
  78. params.successStatuses = [200];
  79. params.xmlResponse = true;
  80. const result = await this.request(params);
  81. return {
  82. location: result.data,
  83. res: result.res
  84. };
  85. };
  86. proto.getBucketInfo = async function getBucketInfo(name, options) {
  87. _checkBucketName(name);
  88. name = name || this.getBucket();
  89. const params = this._bucketRequestParams('GET', name, 'bucketInfo', options);
  90. params.successStatuses = [200];
  91. params.xmlResponse = true;
  92. const result = await this.request(params);
  93. return {
  94. bucket: result.data.Bucket,
  95. res: result.res
  96. };
  97. };
  98. proto.deleteBucket = async function deleteBucket(name, options) {
  99. _checkBucketName(name);
  100. const params = this._bucketRequestParams('DELETE', name, '', options);
  101. const result = await this.request(params);
  102. if (result.status === 200 || result.status === 204) {
  103. return {
  104. res: result.res
  105. };
  106. }
  107. throw await this.requestError(result);
  108. };
  109. // acl
  110. proto.putBucketACL = async function putBucketACL(name, acl, options) {
  111. _checkBucketName(name);
  112. const params = this._bucketRequestParams('PUT', name, 'acl', options);
  113. params.headers = {
  114. 'x-oss-acl': acl
  115. };
  116. params.successStatuses = [200];
  117. const result = await this.request(params);
  118. return {
  119. bucket: (result.headers.location && result.headers.location.substring(1)) || null,
  120. res: result.res
  121. };
  122. };
  123. proto.getBucketACL = async function getBucketACL(name, options) {
  124. _checkBucketName(name);
  125. const params = this._bucketRequestParams('GET', name, 'acl', options);
  126. params.successStatuses = [200];
  127. params.xmlResponse = true;
  128. const result = await this.request(params);
  129. return {
  130. acl: result.data.AccessControlList.Grant,
  131. owner: {
  132. id: result.data.Owner.ID,
  133. displayName: result.data.Owner.DisplayName
  134. },
  135. res: result.res
  136. };
  137. };
  138. // logging
  139. proto.putBucketLogging = async function putBucketLogging(name, prefix, options) {
  140. _checkBucketName(name);
  141. const params = this._bucketRequestParams('PUT', name, 'logging', options);
  142. let xml = `${'<?xml version="1.0" encoding="UTF-8"?>\n<BucketLoggingStatus>\n' +
  143. '<LoggingEnabled>\n<TargetBucket>'}${name}</TargetBucket>\n`;
  144. if (prefix) {
  145. xml += `<TargetPrefix>${prefix}</TargetPrefix>\n`;
  146. }
  147. xml += '</LoggingEnabled>\n</BucketLoggingStatus>';
  148. params.content = xml;
  149. params.mime = 'xml';
  150. params.successStatuses = [200];
  151. const result = await this.request(params);
  152. return {
  153. res: result.res
  154. };
  155. };
  156. proto.getBucketLogging = async function getBucketLogging(name, options) {
  157. _checkBucketName(name);
  158. const params = this._bucketRequestParams('GET', name, 'logging', options);
  159. params.successStatuses = [200];
  160. params.xmlResponse = true;
  161. const result = await this.request(params);
  162. const enable = result.data.LoggingEnabled;
  163. return {
  164. enable: !!enable,
  165. prefix: (enable && enable.TargetPrefix) || null,
  166. res: result.res
  167. };
  168. };
  169. proto.deleteBucketLogging = async function deleteBucketLogging(name, options) {
  170. _checkBucketName(name);
  171. const params = this._bucketRequestParams('DELETE', name, 'logging', options);
  172. params.successStatuses = [204, 200];
  173. const result = await this.request(params);
  174. return {
  175. res: result.res
  176. };
  177. };
  178. proto.putBucketCORS = async function putBucketCORS(name, rules, options) {
  179. _checkBucketName(name);
  180. rules = rules || [];
  181. assert(rules.length, 'rules is required');
  182. rules.forEach((rule) => {
  183. assert(rule.allowedOrigin, 'allowedOrigin is required');
  184. assert(rule.allowedMethod, 'allowedMethod is required');
  185. });
  186. const params = this._bucketRequestParams('PUT', name, 'cors', options);
  187. let xml = '<?xml version="1.0" encoding="UTF-8"?>\n<CORSConfiguration>';
  188. const parseOrigin = (val) => {
  189. xml += `<AllowedOrigin>${val}</AllowedOrigin>`;
  190. };
  191. const parseMethod = (val) => {
  192. xml += `<AllowedMethod>${val}</AllowedMethod>`;
  193. };
  194. const parseHeader = (val) => {
  195. xml += `<AllowedHeader>${val}</AllowedHeader>`;
  196. };
  197. const parseExposeHeader = (val) => {
  198. xml += `<ExposeHeader>${val}</ExposeHeader>`;
  199. };
  200. for (let i = 0, l = rules.length; i < l; i++) {
  201. const rule = rules[i];
  202. xml += '<CORSRule>';
  203. toArray(rule.allowedOrigin).forEach(parseOrigin);
  204. toArray(rule.allowedMethod).forEach(parseMethod);
  205. toArray(rule.allowedHeader).forEach(parseHeader);
  206. toArray(rule.exposeHeader).forEach(parseExposeHeader);
  207. if (rule.maxAgeSeconds) {
  208. xml += `<MaxAgeSeconds>${rule.maxAgeSeconds}</MaxAgeSeconds>`;
  209. }
  210. xml += '</CORSRule>';
  211. }
  212. xml += '</CORSConfiguration>';
  213. params.content = xml;
  214. params.mime = 'xml';
  215. params.successStatuses = [200];
  216. const result = await this.request(params);
  217. return {
  218. res: result.res
  219. };
  220. };
  221. proto.getBucketCORS = async function getBucketCORS(name, options) {
  222. _checkBucketName(name);
  223. const params = this._bucketRequestParams('GET', name, 'cors', options);
  224. params.successStatuses = [200];
  225. params.xmlResponse = true;
  226. const result = await this.request(params);
  227. const rules = [];
  228. if (result.data && result.data.CORSRule) {
  229. let { CORSRule } = result.data;
  230. if (!isArray(CORSRule)) CORSRule = [CORSRule];
  231. CORSRule.forEach((rule) => {
  232. const r = {};
  233. Object.keys(rule).forEach((key) => {
  234. r[key.slice(0, 1).toLowerCase() + key.slice(1, key.length)] = rule[key];
  235. });
  236. rules.push(r);
  237. });
  238. }
  239. return {
  240. rules,
  241. res: result.res
  242. };
  243. };
  244. proto.deleteBucketCORS = async function deleteBucketCORS(name, options) {
  245. _checkBucketName(name);
  246. const params = this._bucketRequestParams('DELETE', name, 'cors', options);
  247. params.successStatuses = [204];
  248. const result = await this.request(params);
  249. return {
  250. res: result.res
  251. };
  252. };
  253. // referer
  254. proto.putBucketReferer = async function putBucketReferer(name, allowEmpty, referers, options) {
  255. _checkBucketName(name);
  256. const params = this._bucketRequestParams('PUT', name, 'referer', options);
  257. let xml = '<?xml version="1.0" encoding="UTF-8"?>\n<RefererConfiguration>\n';
  258. xml += ` <AllowEmptyReferer>${allowEmpty ? 'true' : 'false'}</AllowEmptyReferer>\n`;
  259. if (referers && referers.length > 0) {
  260. xml += ' <RefererList>\n';
  261. for (let i = 0; i < referers.length; i++) {
  262. xml += ` <Referer>${referers[i]}</Referer>\n`;
  263. }
  264. xml += ' </RefererList>\n';
  265. } else {
  266. xml += ' <RefererList />\n';
  267. }
  268. xml += '</RefererConfiguration>';
  269. params.content = xml;
  270. params.mime = 'xml';
  271. params.successStatuses = [200];
  272. const result = await this.request(params);
  273. return {
  274. res: result.res
  275. };
  276. };
  277. proto.getBucketReferer = async function getBucketReferer(name, options) {
  278. _checkBucketName(name);
  279. const params = this._bucketRequestParams('GET', name, 'referer', options);
  280. params.successStatuses = [200];
  281. params.xmlResponse = true;
  282. const result = await this.request(params);
  283. let referers = result.data.RefererList.Referer || null;
  284. if (referers) {
  285. if (!isArray(referers)) {
  286. referers = [referers];
  287. }
  288. }
  289. return {
  290. allowEmpty: result.data.AllowEmptyReferer === 'true',
  291. referers,
  292. res: result.res
  293. };
  294. };
  295. proto.deleteBucketReferer = async function deleteBucketReferer(name, options) {
  296. _checkBucketName(name);
  297. return await this.putBucketReferer(name, true, null, options);
  298. };
  299. // private apis
  300. proto._bucketRequestParams = function _bucketRequestParams(method, bucket, subres, options) {
  301. return {
  302. method,
  303. bucket,
  304. subres,
  305. timeout: options && options.timeout,
  306. ctx: options && options.ctx
  307. };
  308. };