addSeccurityConfigs_test.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493
  1. /**
  2. * Licensed to the Apache Software Foundation (ASF) under one
  3. * or more contributor license agreements. See the NOTICE file
  4. * distributed with this work for additional information
  5. * regarding copyright ownership. The ASF licenses this file
  6. * to you under the Apache License, Version 2.0 (the
  7. * "License"); you may not use this file except in compliance
  8. * with the License. You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. */
  18. var App = require('app');
  19. var stackDescriptorData = require('test/mock_data_setup/stack_descriptors');
  20. var stackDescriptor = stackDescriptorData.Versions.kerberos_descriptor;
  21. require('mixins/wizard/addSecurityConfigs');
  22. describe('App.AddSecurityConfigs', function () {
  23. var controller = Em.Object.extend(App.AddSecurityConfigs,{}).create({
  24. content: {},
  25. enableSubmit: function () {
  26. this._super();
  27. },
  28. secureMapping: [],
  29. secureProperties: []
  30. });
  31. describe('#secureServices', function() {
  32. it('content.services is correct', function() {
  33. controller.set('content.services', [{}]);
  34. expect(controller.get('secureServices')).to.eql([{}]);
  35. controller.reopen({
  36. secureServices: []
  37. });
  38. });
  39. });
  40. describe('#loadUiSideSecureConfigs()', function() {
  41. beforeEach(function(){
  42. sinon.stub(controller, 'checkServiceForConfigValue', function() {
  43. return 'value2';
  44. });
  45. sinon.stub(controller, 'setConfigValue', Em.K);
  46. sinon.stub(controller, 'formatConfigName', Em.K);
  47. sinon.stub(App.Service, 'find').returns([{serviceName: 'SOME_SERVICE'}]);
  48. });
  49. afterEach(function(){
  50. controller.checkServiceForConfigValue.restore();
  51. controller.setConfigValue.restore();
  52. controller.formatConfigName.restore();
  53. App.Service.find.restore();
  54. });
  55. it('secureMapping is empty', function() {
  56. controller.set('secureMapping', []);
  57. expect(controller.loadUiSideSecureConfigs()).to.be.empty;
  58. });
  59. it('Config does not have dependedServiceName', function() {
  60. controller.set('secureMapping', [{
  61. name: 'config1',
  62. value: 'value1',
  63. filename: 'file1',
  64. serviceName: 'SOME_SERVICE',
  65. foreignKey: null
  66. }]);
  67. expect(controller.loadUiSideSecureConfigs()).to.eql([{
  68. "id": "site property",
  69. "name": 'config1',
  70. "value": 'value1',
  71. "filename": 'file1'
  72. }]);
  73. });
  74. it('Config has dependedServiceName', function() {
  75. controller.set('secureMapping', [{
  76. name: 'config1',
  77. value: 'value1',
  78. filename: 'file1',
  79. foreignKey: null,
  80. serviceName: 'SOME_SERVICE',
  81. dependedServiceName: 'SOME_SERVICE'
  82. }]);
  83. expect(controller.loadUiSideSecureConfigs()).to.eql([{
  84. "id": "site property",
  85. "name": 'config1',
  86. "value": 'value2',
  87. "filename": 'file1'
  88. }]);
  89. });
  90. it('Config has non-existent serviceName', function() {
  91. controller.set('secureMapping', [{
  92. name: 'config1',
  93. value: 'value1',
  94. filename: 'file1',
  95. foreignKey: true,
  96. serviceName: 'NO_SERVICE'
  97. }]);
  98. expect(controller.loadUiSideSecureConfigs()).to.be.empty;
  99. });
  100. it('Config has correct serviceName', function() {
  101. controller.set('secureMapping', [{
  102. name: 'config1',
  103. value: 'value1',
  104. filename: 'file1',
  105. foreignKey: true,
  106. serviceName: 'SOME_SERVICE'
  107. }]);
  108. expect(controller.loadUiSideSecureConfigs()).to.eql([{
  109. "id": "site property",
  110. "name": 'config1',
  111. "value": 'value1',
  112. "filename": 'file1'
  113. }]);
  114. expect(controller.setConfigValue.calledOnce).to.be.true;
  115. expect(controller.formatConfigName.calledOnce).to.be.true;
  116. });
  117. });
  118. describe('#checkServiceForConfigValue()', function() {
  119. it('services is empty', function() {
  120. var services = [];
  121. expect(controller.checkServiceForConfigValue('value1', services)).to.equal('value1');
  122. });
  123. it('Service is loaded', function() {
  124. var services = [{}];
  125. sinon.stub(App.Service, 'find', function () {
  126. return Em.Object.create({isLoaded: false});
  127. });
  128. expect(controller.checkServiceForConfigValue('value1', services)).to.equal('value1');
  129. App.Service.find.restore();
  130. });
  131. it('Service is not loaded', function() {
  132. var services = [{
  133. replace: 'val'
  134. }];
  135. sinon.stub(App.Service, 'find', function () {
  136. return Em.Object.create({isLoaded: false});
  137. });
  138. expect(controller.checkServiceForConfigValue('value1', services)).to.equal('ue1');
  139. App.Service.find.restore();
  140. });
  141. });
  142. describe('#formatConfigName()', function() {
  143. it('config.value is null', function() {
  144. var config = {
  145. value: null
  146. };
  147. expect(controller.formatConfigName([], config)).to.be.false;
  148. });
  149. it('config.name does not contain foreignKey', function() {
  150. var config = {
  151. value: 'value1',
  152. name: 'config1'
  153. };
  154. expect(controller.formatConfigName([], config)).to.be.false;
  155. });
  156. it('globalProperties is empty, use uiConfig', function() {
  157. var config = {
  158. value: 'value1',
  159. name: '<foreignKey[0]>',
  160. foreignKey: ['key1']
  161. };
  162. controller.set('globalProperties', []);
  163. var uiConfig = [{
  164. name: 'key1',
  165. value: 'globalValue1'
  166. }];
  167. expect(controller.formatConfigName(uiConfig, config)).to.be.true;
  168. expect(config._name).to.equal('globalValue1');
  169. });
  170. });
  171. describe('#setConfigValue()', function() {
  172. it('config.value is null', function() {
  173. var config = {
  174. value: null
  175. };
  176. expect(controller.setConfigValue(config)).to.be.false;
  177. });
  178. it('config.value does not match "templateName"', function() {
  179. var config = {
  180. value: ''
  181. };
  182. expect(controller.setConfigValue(config)).to.be.false;
  183. });
  184. it('No such property in global configs', function() {
  185. var config = {
  186. value: '<templateName[0]>',
  187. templateName: ['config1']
  188. };
  189. controller.set('globalProperties', []);
  190. controller.set('configs', []);
  191. expect(controller.setConfigValue(config)).to.be.true;
  192. expect(config.value).to.be.null;
  193. });
  194. });
  195. describe('#addHostConfig()', function() {
  196. afterEach(function () {
  197. App.Service.find.restore();
  198. });
  199. it('No such service loaded', function() {
  200. sinon.stub(App.Service, 'find', function(){
  201. return Em.Object.create({isLoaded: false});
  202. });
  203. expect(controller.addHostConfig('service1', 'comp1', 'config1')).to.be.false;
  204. });
  205. it('No such service in secureServices', function() {
  206. sinon.stub(App.Service, 'find', function(){
  207. return Em.Object.create({isLoaded: true});
  208. });
  209. controller.set('secureServices', []);
  210. expect(controller.addHostConfig('service1', 'comp1', 'config1')).to.be.false;
  211. });
  212. it('Service does not have such host-component', function() {
  213. sinon.stub(App.Service, 'find', function(){
  214. return Em.Object.create({
  215. isLoaded: true,
  216. hostComponents: []
  217. });
  218. });
  219. controller.set('secureServices', [{
  220. serviceName: 'service1'
  221. }]);
  222. expect(controller.addHostConfig('service1', 'comp1', 'config1')).to.be.false;
  223. });
  224. });
  225. describe('#getPrincipalNames()', function() {
  226. beforeEach(function () {
  227. controller.set('globalProperties', []);
  228. controller.set('secureProperties', []);
  229. });
  230. it('globalProperties and secureProperties are empty', function() {
  231. expect(controller.getPrincipalNames()).to.be.empty;
  232. });
  233. it('global property name does not match "principal_name"', function() {
  234. controller.set('globalProperties', [{
  235. name: 'config1'
  236. }]);
  237. expect(controller.getPrincipalNames()).to.be.empty;
  238. });
  239. it('secure property name does not match "principal_name"', function() {
  240. controller.set('secureProperties', [{
  241. name: 'config1'
  242. }]);
  243. expect(controller.getPrincipalNames()).to.be.empty;
  244. });
  245. it('property with such name already exists', function() {
  246. controller.set('globalProperties', [{
  247. name: 'principal_name'
  248. }]);
  249. controller.set('secureProperties', [{
  250. name: 'principal_name'
  251. }]);
  252. expect(controller.getPrincipalNames().mapProperty('name')).to.eql(['principal_name']);
  253. });
  254. });
  255. describe('#loadUsersFromServer()', function() {
  256. it('testMode = true', function() {
  257. controller.set('testModeUsers', [{
  258. name: 'user1',
  259. value: 'value1'
  260. }]);
  261. controller.set('serviceUsers', []);
  262. sinon.stub(App, 'get', function(k) {
  263. if ('testMode' === k) return true;
  264. return Em.get(App, k);
  265. });
  266. controller.loadUsersFromServer();
  267. expect(controller.get('serviceUsers')).to.eql([{
  268. name: 'user1',
  269. value: 'value1',
  270. id: 'puppet var'
  271. }]);
  272. App.get.restore();
  273. });
  274. it('testMode = false', function() {
  275. sinon.stub(App.router, 'set', Em.K);
  276. sinon.stub(App.db, 'getSecureUserInfo', function(){
  277. return [];
  278. });
  279. sinon.stub(App, 'get', function(k) {
  280. if ('testMode' === k) return false;
  281. return Em.get(App, k);
  282. });
  283. controller.loadUsersFromServer();
  284. expect(App.db.getSecureUserInfo.calledOnce).to.be.true;
  285. expect(App.router.set.calledWith('mainAdminSecurityController.serviceUsers', [])).to.be.true;
  286. App.router.set.restore();
  287. App.get.restore();
  288. App.db.getSecureUserInfo.restore();
  289. });
  290. });
  291. describe('#createServicesStackDescriptorConfigs', function() {
  292. var result = controller.createServicesStackDescriptorConfigs(stackDescriptorData);
  293. var propertyValidationTests = [
  294. {
  295. property: 'spnego_keytab',
  296. e: [
  297. { key: 'value', value: '${keytab_dir}/spnego.service.keytab' },
  298. { key: 'serviceName', value: 'Cluster' },
  299. ]
  300. }
  301. ];
  302. it('resulted array should have unique properties by name', function() {
  303. expect(result.mapProperty('name').length).to.be.eql(result.mapProperty('name').uniq().length);
  304. });
  305. propertyValidationTests.forEach(function(test) {
  306. it('property {0} should be created'.format(test.property), function() {
  307. expect(result.findProperty('name', test.property)).to.be.ok;
  308. });
  309. test.e.forEach(function(expected) {
  310. it('property `{0}` should have `{1}` with value `{2}`'.format(test.property, expected.key, expected.value), function() {
  311. expect(result.findProperty('name', test.property)).to.have.deep.property(expected.key, expected.value);
  312. });
  313. });
  314. });
  315. });
  316. describe('#expandKerberosStackDescriptorProps', function() {
  317. var result = controller.expandKerberosStackDescriptorProps(stackDescriptor.properties);
  318. var testCases = [
  319. {
  320. property: 'realm',
  321. e: [
  322. { key: 'isEditable', value: false },
  323. { key: 'serviceName', value: 'Cluster' },
  324. ]
  325. },
  326. {
  327. property: 'keytab_dir',
  328. e: [
  329. { key: 'isEditable', value: true },
  330. { key: 'serviceName', value: 'Cluster' },
  331. ]
  332. }
  333. ];
  334. testCases.forEach(function(test) {
  335. it('property {0} should be created'.format(test.property), function() {
  336. expect(result.findProperty('name', test.property)).to.be.ok;
  337. });
  338. test.e.forEach(function(expected) {
  339. it('property `{0}` should have `{1}` with value `{2}`'.format(test.property, expected.key, expected.value), function() {
  340. expect(result.findProperty('name', test.property)).to.have.deep.property(expected.key, expected.value);
  341. });
  342. });
  343. });
  344. });
  345. describe('#createConfigsByIdentity', function() {
  346. var identitiesData = stackDescriptor.services[0].components[0].identities;
  347. var tests = [
  348. {
  349. property: 'dfs.namenode.kerberos.principal',
  350. e: [
  351. { key: 'value', value: 'nn/_HOST@${realm}' },
  352. ]
  353. },
  354. {
  355. property: 'dfs.web.authentication.kerberos.principal',
  356. e: [
  357. { key: 'observesValueFrom', value: 'spnego_principal' },
  358. { key: 'isEditable', value: false }
  359. ]
  360. }
  361. ];
  362. var properties = controller.createConfigsByIdentities(identitiesData, 'NAMENODE');
  363. tests.forEach(function(test) {
  364. it('property {0} should be created'.format(test.property), function() {
  365. expect(properties.findProperty('name', test.property)).to.be.ok;
  366. });
  367. test.e.forEach(function(expected) {
  368. it('property `{0}` should have `{1}` with value `{2}`'.format(test.property, expected.key, expected.value), function() {
  369. expect(properties.findProperty('name', test.property)).to.have.deep.property(expected.key, expected.value);
  370. });
  371. });
  372. });
  373. });
  374. describe('#parseIdentityObject', function() {
  375. var testCases = [
  376. {
  377. identity: stackDescriptor.services[0].components[0].identities[0],
  378. tests: [
  379. {
  380. property: 'dfs.namenode.kerberos.principal',
  381. e: [
  382. { key: 'filename', value: 'hdfs-site' },
  383. ]
  384. },
  385. {
  386. property: 'dfs.namenode.keytab.file',
  387. e: [
  388. { key: 'value', value: '${keytab_dir}/nn.service.keytab' }
  389. ]
  390. }
  391. ]
  392. },
  393. {
  394. identity: stackDescriptor.services[0].components[0].identities[1],
  395. tests: [
  396. {
  397. property: 'dfs.namenode.kerberos.https.principal',
  398. e: [
  399. { key: 'filename', value: 'hdfs-site' }
  400. ]
  401. }
  402. ]
  403. },
  404. {
  405. identity: stackDescriptor.identities[0],
  406. tests: [
  407. {
  408. property: 'spnego_principal',
  409. e: [
  410. { key: 'displayName', value: 'spnego_principal' },
  411. { key: 'filename', value: 'cluster-env' },
  412. ]
  413. }
  414. ]
  415. },
  416. {
  417. identity: stackDescriptor.identities[0],
  418. tests: [
  419. {
  420. property: 'spnego_keytab',
  421. e: [
  422. { key: 'displayName', value: 'spnego_keytab' },
  423. { key: 'filename', value: 'cluster-env' },
  424. ]
  425. }
  426. ]
  427. }
  428. ];
  429. testCases.forEach(function(testCase) {
  430. testCase.tests.forEach(function(test) {
  431. var result = controller.parseIdentityObject(testCase.identity);
  432. it('property `{0}` should be present'.format(test.property), function() {
  433. expect(result.findProperty('name', test.property)).to.be.ok;
  434. });
  435. test.e.forEach(function(expected) {
  436. it('property `{0}` should have `{1}` with value `{2}`'.format(test.property, expected.key, expected.value), function() {
  437. expect(result.findProperty('name', test.property)).to.have.deep.property(expected.key, expected.value);
  438. });
  439. });
  440. });
  441. });
  442. });
  443. });