LdapEditPage.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. // Copyright 2021 The Casdoor Authors. All Rights Reserved.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. import React from "react";
  15. import {Button, Card, Col, Input, InputNumber, Row, Select, Switch} from "antd";
  16. import {EyeInvisibleOutlined, EyeTwoTone} from "@ant-design/icons";
  17. import * as LddpBackend from "./backend/LdapBackend";
  18. import * as OrganizationBackend from "./backend/OrganizationBackend";
  19. import * as Setting from "./Setting";
  20. import i18next from "i18next";
  21. const {Option} = Select;
  22. class LdapEditPage extends React.Component {
  23. constructor(props) {
  24. super(props);
  25. this.state = {
  26. ldapId: props.match.params.ldapId,
  27. organizationName: props.match.params.organizationName,
  28. ldap: null,
  29. organizations: [],
  30. };
  31. }
  32. UNSAFE_componentWillMount() {
  33. this.getLdap();
  34. this.getOrganizations();
  35. }
  36. getLdap() {
  37. LddpBackend.getLdap(this.state.organizationName, this.state.ldapId)
  38. .then((res) => {
  39. if (res.status === "ok") {
  40. this.setState({
  41. ldap: res.data,
  42. });
  43. } else {
  44. Setting.showMessage("error", res.msg);
  45. }
  46. });
  47. }
  48. getOrganizations() {
  49. OrganizationBackend.getOrganizations("admin")
  50. .then((res) => {
  51. this.setState({
  52. organizations: (res.msg === undefined) ? res : [],
  53. });
  54. });
  55. }
  56. updateLdapField(key, value) {
  57. this.setState((prevState) => {
  58. prevState.ldap[key] = value;
  59. return prevState;
  60. });
  61. }
  62. renderAutoSyncWarn() {
  63. if (this.state.ldap.autoSync > 0) {
  64. return (
  65. <span style={{
  66. color: "#faad14",
  67. marginLeft: "20px",
  68. }}>{i18next.t("ldap:The Auto Sync option will sync all users to specify organization")}</span>
  69. );
  70. }
  71. }
  72. renderLdap() {
  73. return (
  74. <Card size="small" title={
  75. <div>
  76. {i18next.t("ldap:Edit LDAP")}&nbsp;&nbsp;&nbsp;&nbsp;
  77. <Button onClick={() => this.submitLdapEdit()}>{i18next.t("general:Save")}</Button>
  78. <Button style={{marginLeft: "20px"}} type="primary" onClick={() => this.submitLdapEdit(true)}>{i18next.t("general:Save & Exit")}</Button>
  79. <Button style={{marginLeft: "20px"}}
  80. onClick={() => Setting.goToLink(`/ldap/sync/${this.state.organizationName}/${this.state.ldapId}`)}>
  81. {i18next.t("general:Sync")} LDAP
  82. </Button>
  83. </div>
  84. } style={{marginLeft: "5px"}} type="inner">
  85. <Row style={{marginTop: "10px"}}>
  86. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  87. {Setting.getLabel(i18next.t("general:Organization"), i18next.t("general:Organization - Tooltip"))} :
  88. </Col>
  89. <Col span={21}>
  90. <Select virtual={false} style={{width: "100%"}} disabled={!Setting.isAdminUser(this.props.account)}
  91. value={this.state.ldap.owner} onChange={(value => {
  92. this.updateLdapField("owner", value);
  93. })}>
  94. {
  95. this.state.organizations.map((organization, index) => <Option key={index}
  96. value={organization.name}>{organization.name}</Option>)
  97. }
  98. </Select>
  99. </Col>
  100. </Row>
  101. <Row style={{marginTop: "20px"}}>
  102. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  103. {Setting.getLabel(i18next.t("general:ID"), i18next.t("general:ID - Tooltip"))} :
  104. </Col>
  105. <Col span={21}>
  106. <Input value={this.state.ldap.id} disabled={true} />
  107. </Col>
  108. </Row>
  109. <Row style={{marginTop: "20px"}}>
  110. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  111. {Setting.getLabel(i18next.t("ldap:Server name"), i18next.t("ldap:Server name - Tooltip"))} :
  112. </Col>
  113. <Col span={21}>
  114. <Input value={this.state.ldap.serverName} onChange={e => {
  115. this.updateLdapField("serverName", e.target.value);
  116. }} />
  117. </Col>
  118. </Row>
  119. <Row style={{marginTop: "20px"}}>
  120. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  121. {Setting.getLabel(i18next.t("ldap:Server host"), i18next.t("ldap:Server host - Tooltip"))} :
  122. </Col>
  123. <Col span={21}>
  124. <Input value={this.state.ldap.host} onChange={e => {
  125. this.updateLdapField("host", e.target.value);
  126. }} />
  127. </Col>
  128. </Row>
  129. <Row style={{marginTop: "20px"}}>
  130. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  131. {Setting.getLabel(i18next.t("ldap:Server port"), i18next.t("ldap:Server port - Tooltip"))} :
  132. </Col>
  133. <Col span={21}>
  134. <InputNumber min={0} max={65535} formatter={value => value.replace(/\$\s?|(,*)/g, "")}
  135. value={this.state.ldap.port} onChange={value => {
  136. this.updateLdapField("port", value);
  137. }} />
  138. </Col>
  139. </Row>
  140. <Row style={{marginTop: "20px"}} >
  141. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  142. {Setting.getLabel(i18next.t("ldap:Enable SSL"), i18next.t("ldap:Enable SSL - Tooltip"))} :
  143. </Col>
  144. <Col span={21} >
  145. <Switch checked={this.state.ldap.enableSsl} onChange={checked => {
  146. this.updateLdapField("enableSsl", checked);
  147. }} />
  148. </Col>
  149. </Row>
  150. <Row style={{marginTop: "20px"}}>
  151. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  152. {Setting.getLabel(i18next.t("ldap:Base DN"), i18next.t("ldap:Base DN - Tooltip"))} :
  153. </Col>
  154. <Col span={21}>
  155. <Input value={this.state.ldap.baseDn} onChange={e => {
  156. this.updateLdapField("baseDn", e.target.value);
  157. }} />
  158. </Col>
  159. </Row>
  160. <Row style={{marginTop: "20px"}}>
  161. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  162. {Setting.getLabel(i18next.t("ldap:Search Filter"), i18next.t("ldap:Search Filter - Tooltip"))} :
  163. </Col>
  164. <Col span={21}>
  165. <Input value={this.state.ldap.filter} onChange={e => {
  166. this.updateLdapField("filter", e.target.value);
  167. }} />
  168. </Col>
  169. </Row>
  170. <Row style={{marginTop: "20px"}}>
  171. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  172. {Setting.getLabel(i18next.t("ldap:Filter fields"), i18next.t("ldap:Filter fields - Tooltip"))} :
  173. </Col>
  174. <Col span={21}>
  175. <Select value={this.state.ldap.filterFields ?? []} style={{width: "100%"}} mode={"multiple"} options={[
  176. {value: "uid", label: "uid"},
  177. {value: "mail", label: "Email"},
  178. {value: "mobile", label: "mobile"},
  179. ].map((item) => Setting.getOption(item.label, item.value))} onChange={value => {
  180. this.updateLdapField("filterFields", value);
  181. }} />
  182. </Col>
  183. </Row>
  184. <Row style={{marginTop: "20px"}}>
  185. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  186. {Setting.getLabel(i18next.t("ldap:Admin"), i18next.t("ldap:Admin - Tooltip"))} :
  187. </Col>
  188. <Col span={21}>
  189. <Input value={this.state.ldap.username} onChange={e => {
  190. this.updateLdapField("username", e.target.value);
  191. }} />
  192. </Col>
  193. </Row>
  194. <Row style={{marginTop: "20px"}}>
  195. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  196. {Setting.getLabel(i18next.t("ldap:Admin Password"), i18next.t("ldap:Admin Password - Tooltip"))} :
  197. </Col>
  198. <Col span={21}>
  199. <Input.Password
  200. iconRender={visible => (visible ? <EyeTwoTone /> : <EyeInvisibleOutlined />)} value={this.state.ldap.password}
  201. onChange={e => {
  202. this.updateLdapField("password", e.target.value);
  203. }}
  204. />
  205. </Col>
  206. </Row>
  207. <Row style={{marginTop: "20px"}}>
  208. <Col style={{lineHeight: "32px", textAlign: "right", paddingRight: "25px"}} span={3}>
  209. {Setting.getLabel(i18next.t("ldap:Auto Sync"), i18next.t("ldap:Auto Sync - Tooltip"))} :
  210. </Col>
  211. <Col span={21}>
  212. <InputNumber min={0} formatter={value => value.replace(/\$\s?|(,*)/g, "")} disabled={false}
  213. value={this.state.ldap.autoSync} onChange={value => {
  214. this.updateLdapField("autoSync", value);
  215. }} /><span>&nbsp;mins</span>
  216. {this.renderAutoSyncWarn()}
  217. </Col>
  218. </Row>
  219. </Card>
  220. );
  221. }
  222. submitLdapEdit(willExist) {
  223. LddpBackend.updateLdap(this.state.ldap)
  224. .then((res) => {
  225. if (res.status === "ok") {
  226. Setting.showMessage("success", "Update LDAP server success");
  227. this.setState({
  228. organizationName: this.state.ldap.owner,
  229. });
  230. if (willExist) {
  231. this.props.history.push(`/organizations/${this.state.organizationName}`);
  232. }
  233. } else {
  234. Setting.showMessage("error", res.msg);
  235. }
  236. })
  237. .catch(error => {
  238. Setting.showMessage("error", `Update LDAP server failed: ${error}`);
  239. });
  240. }
  241. render() {
  242. return (
  243. <div>
  244. {
  245. this.state.ldap !== null ? this.renderLdap() : null
  246. }
  247. <div style={{marginTop: "20px", marginLeft: "40px"}}>
  248. <Button size="large" onClick={() => this.submitLdapEdit()}>{i18next.t("general:Save")}</Button>
  249. <Button style={{marginLeft: "20px"}} type="primary" size="large" onClick={() => this.submitLdapEdit(true)}>{i18next.t("general:Save & Exit")}</Button>
  250. </div>
  251. </div>
  252. );
  253. }
  254. }
  255. export default LdapEditPage;