CertListPage.js 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  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 {Link} from "react-router-dom";
  16. import {Button, Table} from "antd";
  17. import moment from "moment";
  18. import * as Setting from "./Setting";
  19. import * as CertBackend from "./backend/CertBackend";
  20. import i18next from "i18next";
  21. import BaseListPage from "./BaseListPage";
  22. import PopconfirmModal from "./common/modal/PopconfirmModal";
  23. class CertListPage extends BaseListPage {
  24. newCert() {
  25. const randomName = Setting.getRandomName();
  26. return {
  27. owner: "admin", // this.props.account.certname,
  28. name: `cert_${randomName}`,
  29. createdTime: moment().format(),
  30. displayName: `New Cert - ${randomName}`,
  31. scope: "JWT",
  32. type: "x509",
  33. cryptoAlgorithm: "RS256",
  34. bitSize: 4096,
  35. expireInYears: 20,
  36. certificate: "",
  37. privateKey: "",
  38. };
  39. }
  40. addCert() {
  41. const newCert = this.newCert();
  42. CertBackend.addCert(newCert)
  43. .then((res) => {
  44. if (res.status === "ok") {
  45. this.props.history.push({pathname: `/certs/${newCert.name}`, mode: "add"});
  46. Setting.showMessage("success", i18next.t("general:Successfully added"));
  47. } else {
  48. Setting.showMessage("error", `${i18next.t("general:Failed to add")}: ${res.msg}`);
  49. }
  50. })
  51. .catch(error => {
  52. Setting.showMessage("error", `${i18next.t("general:Failed to connect to server")}: ${error}`);
  53. });
  54. }
  55. deleteCert(i) {
  56. CertBackend.deleteCert(this.state.data[i])
  57. .then((res) => {
  58. if (res.status === "ok") {
  59. Setting.showMessage("success", i18next.t("general:Successfully deleted"));
  60. this.setState({
  61. data: Setting.deleteRow(this.state.data, i),
  62. pagination: {total: this.state.pagination.total - 1},
  63. });
  64. } else {
  65. Setting.showMessage("error", `${i18next.t("general:Failed to delete")}: ${res.msg}`);
  66. }
  67. })
  68. .catch(error => {
  69. Setting.showMessage("error", `${i18next.t("general:Failed to connect to server")}: ${error}`);
  70. });
  71. }
  72. renderTable(certs) {
  73. const columns = [
  74. {
  75. title: i18next.t("general:Name"),
  76. dataIndex: "name",
  77. key: "name",
  78. width: "120px",
  79. fixed: "left",
  80. sorter: true,
  81. ...this.getColumnSearchProps("name"),
  82. render: (text, record, index) => {
  83. return (
  84. <Link to={`/certs/${text}`}>
  85. {text}
  86. </Link>
  87. );
  88. },
  89. },
  90. {
  91. title: i18next.t("general:Created time"),
  92. dataIndex: "createdTime",
  93. key: "createdTime",
  94. width: "180px",
  95. sorter: true,
  96. render: (text, record, index) => {
  97. return Setting.getFormattedDate(text);
  98. },
  99. },
  100. {
  101. title: i18next.t("general:Display name"),
  102. dataIndex: "displayName",
  103. key: "displayName",
  104. // width: '100px',
  105. sorter: true,
  106. ...this.getColumnSearchProps("displayName"),
  107. },
  108. {
  109. title: i18next.t("provider:Scope"),
  110. dataIndex: "scope",
  111. key: "scope",
  112. filterMultiple: false,
  113. filters: [
  114. {text: "JWT", value: "JWT"},
  115. ],
  116. width: "110px",
  117. sorter: true,
  118. },
  119. {
  120. title: i18next.t("provider:Type"),
  121. dataIndex: "type",
  122. key: "type",
  123. filterMultiple: false,
  124. filters: [
  125. {text: "x509", value: "x509"},
  126. ],
  127. width: "110px",
  128. sorter: true,
  129. },
  130. {
  131. title: i18next.t("cert:Crypto algorithm"),
  132. dataIndex: "cryptoAlgorithm",
  133. key: "cryptoAlgorithm",
  134. filterMultiple: false,
  135. filters: [
  136. {text: "RS256", value: "RS256"},
  137. ],
  138. width: "190px",
  139. sorter: true,
  140. },
  141. {
  142. title: i18next.t("cert:Bit size"),
  143. dataIndex: "bitSize",
  144. key: "bitSize",
  145. width: "130px",
  146. sorter: true,
  147. ...this.getColumnSearchProps("bitSize"),
  148. },
  149. {
  150. title: i18next.t("cert:Expire in years"),
  151. dataIndex: "expireInYears",
  152. key: "expireInYears",
  153. width: "170px",
  154. sorter: true,
  155. ...this.getColumnSearchProps("expireInYears"),
  156. },
  157. {
  158. title: i18next.t("general:Action"),
  159. dataIndex: "",
  160. key: "op",
  161. width: "170px",
  162. fixed: (Setting.isMobile()) ? "false" : "right",
  163. render: (text, record, index) => {
  164. return (
  165. <div>
  166. <Button style={{marginTop: "10px", marginBottom: "10px", marginRight: "10px"}} type="primary" onClick={() => this.props.history.push(`/certs/${record.name}`)}>{i18next.t("general:Edit")}</Button>
  167. <PopconfirmModal
  168. title={i18next.t("general:Sure to delete") + `: ${record.name} ?`}
  169. onConfirm={() => this.deleteCert(index)}
  170. >
  171. </PopconfirmModal>
  172. </div>
  173. );
  174. },
  175. },
  176. ];
  177. const paginationProps = {
  178. total: this.state.pagination.total,
  179. showQuickJumper: true,
  180. showSizeChanger: true,
  181. showTotal: () => i18next.t("general:{total} in total").replace("{total}", this.state.pagination.total),
  182. };
  183. return (
  184. <div>
  185. <Table scroll={{x: "max-content"}} columns={columns} dataSource={certs} rowKey="name" size="middle" bordered pagination={paginationProps}
  186. title={() => (
  187. <div>
  188. {i18next.t("general:Certs")}&nbsp;&nbsp;&nbsp;&nbsp;
  189. <Button type="primary" size="small" onClick={this.addCert.bind(this)}>{i18next.t("general:Add")}</Button>
  190. </div>
  191. )}
  192. loading={this.state.loading}
  193. onChange={this.handleTableChange}
  194. />
  195. </div>
  196. );
  197. }
  198. fetch = (params = {}) => {
  199. let field = params.searchedColumn, value = params.searchText;
  200. const sortField = params.sortField, sortOrder = params.sortOrder;
  201. if (params.category !== undefined && params.category !== null) {
  202. field = "category";
  203. value = params.category;
  204. } else if (params.type !== undefined && params.type !== null) {
  205. field = "type";
  206. value = params.type;
  207. }
  208. this.setState({loading: true});
  209. CertBackend.getCerts("admin", params.pagination.current, params.pagination.pageSize, field, value, sortField, sortOrder)
  210. .then((res) => {
  211. if (res.status === "ok") {
  212. this.setState({
  213. loading: false,
  214. data: res.data,
  215. pagination: {
  216. ...params.pagination,
  217. total: res.data2,
  218. },
  219. searchText: params.searchText,
  220. searchedColumn: params.searchedColumn,
  221. });
  222. } else {
  223. if (Setting.isResponseDenied(res)) {
  224. this.setState({
  225. loading: false,
  226. isAuthorized: false,
  227. });
  228. }
  229. }
  230. });
  231. };
  232. }
  233. export default CertListPage;