mirror of
https://github.com/minio/minio.git
synced 2025-04-04 20:00:31 -04:00
With this change, MinIO's ILM supports transitioning objects to a remote tier. This change includes support for Azure Blob Storage, AWS S3 compatible object storage incl. MinIO and Google Cloud Storage as remote tier storage backends. Some new additions include: - Admin APIs remote tier configuration management - Simple journal to track remote objects to be 'collected' This is used by object API handlers which 'mutate' object versions by overwriting/replacing content (Put/CopyObject) or removing the version itself (e.g DeleteObjectVersion). - Rework of previous ILM transition to fit the new model In the new model, a storage class (a.k.a remote tier) is defined by the 'remote' object storage type (one of s3, azure, GCS), bucket name and a prefix. * Fixed bugs, review comments, and more unit-tests - Leverage inline small object feature - Migrate legacy objects to the latest object format before transitioning - Fix restore to particular version if specified - Extend SharedDataDirCount to handle transitioned and restored objects - Restore-object should accept version-id for version-suspended bucket (#12091) - Check if remote tier creds have sufficient permissions - Bonus minor fixes to existing error messages Co-authored-by: Poorna Krishnamoorthy <poorna@minio.io> Co-authored-by: Krishna Srinivas <krishna@minio.io> Signed-off-by: Harshavardhana <harsha@minio.io>
173 lines
4.9 KiB
JavaScript
173 lines
4.9 KiB
JavaScript
/*
|
|
* Copyright (c) 2015-2021 MinIO, Inc.
|
|
*
|
|
* This file is part of MinIO Object Storage stack
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import React from "react"
|
|
import { connect } from "react-redux"
|
|
import logo from "../../img/logo.svg"
|
|
import Alert from "../alert/Alert"
|
|
import * as actionsAlert from "../alert/actions"
|
|
import InputGroup from "./InputGroup"
|
|
import web from "../web"
|
|
import { Redirect } from "react-router-dom"
|
|
import qs from "query-string"
|
|
import { getRandomString } from "../utils"
|
|
import storage from "local-storage-fallback"
|
|
import jwtDecode from "jwt-decode"
|
|
import { buildOpenIDAuthURL, OPEN_ID_NONCE_KEY } from './utils'
|
|
|
|
export class OpenIDLogin extends React.Component {
|
|
constructor(props) {
|
|
super(props)
|
|
this.state = {
|
|
clientID: "",
|
|
discoveryDoc: {}
|
|
}
|
|
this.clientIDChange = this.clientIDChange.bind(this)
|
|
this.handleSubmit = this.handleSubmit.bind(this)
|
|
}
|
|
|
|
clientIDChange(e) {
|
|
this.setState({
|
|
clientID: e.target.value
|
|
})
|
|
}
|
|
|
|
handleSubmit(event) {
|
|
event.preventDefault()
|
|
const { showAlert } = this.props
|
|
let message = ""
|
|
if (this.state.clientID === "") {
|
|
message = "Client ID cannot be empty"
|
|
}
|
|
if (message) {
|
|
showAlert("danger", message)
|
|
return
|
|
}
|
|
|
|
if (this.state.discoveryDoc && this.state.discoveryDoc.authorization_endpoint) {
|
|
const redirectURI = window.location.href.split("#")[0]
|
|
|
|
// Store nonce in localstorage to check again after the redirect
|
|
const nonce = getRandomString(16)
|
|
storage.setItem(OPEN_ID_NONCE_KEY, nonce)
|
|
|
|
const authURL = buildOpenIDAuthURL(
|
|
this.state.discoveryDoc.authorization_endpoint,
|
|
this.state.discoveryDoc.scopes_supported,
|
|
redirectURI,
|
|
this.state.clientID,
|
|
nonce
|
|
)
|
|
window.location = authURL
|
|
}
|
|
}
|
|
|
|
componentWillMount() {
|
|
const { clearAlert } = this.props
|
|
// Clear out any stale message in the alert of previous page
|
|
clearAlert()
|
|
document.body.classList.add("is-guest")
|
|
|
|
web.GetDiscoveryDoc().then(({ DiscoveryDoc }) => {
|
|
this.setState({
|
|
discoveryDoc: DiscoveryDoc
|
|
})
|
|
})
|
|
}
|
|
|
|
componentDidMount() {
|
|
const values = qs.parse(this.props.location.hash)
|
|
if (values.error) {
|
|
this.props.showAlert("danger", values.error_description)
|
|
return
|
|
}
|
|
|
|
if (values.id_token) {
|
|
// Check nonce on the token to prevent replay attacks
|
|
const tokenJSON = jwtDecode(values.id_token)
|
|
if (storage.getItem(OPEN_ID_NONCE_KEY) !== tokenJSON.nonce) {
|
|
this.props.showAlert("danger", "Invalid auth token")
|
|
return
|
|
}
|
|
|
|
web.LoginSTS({ token: values.id_token }).then(() => {
|
|
storage.removeItem(OPEN_ID_NONCE_KEY)
|
|
this.forceUpdate()
|
|
return
|
|
})
|
|
}
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
document.body.classList.remove("is-guest")
|
|
}
|
|
|
|
render() {
|
|
const { clearAlert, alert } = this.props
|
|
if (web.LoggedIn()) {
|
|
return <Redirect to={"/"} />
|
|
}
|
|
let alertBox = <Alert {...alert} onDismiss={clearAlert} />
|
|
// Make sure you don't show a fading out alert box on the initial web-page load.
|
|
if (!alert.message) alertBox = ""
|
|
return (
|
|
<div className="login">
|
|
{alertBox}
|
|
<div className="l-wrap">
|
|
<form onSubmit={this.handleSubmit}>
|
|
<InputGroup
|
|
value={this.state.clientID}
|
|
onChange={this.clientIDChange}
|
|
className="ig-dark"
|
|
label="Client ID"
|
|
id="clientID"
|
|
name="clientID"
|
|
type="text"
|
|
spellCheck="false"
|
|
required="required"
|
|
/>
|
|
<button className="lw-btn" type="submit">
|
|
<i className="fas fa-sign-in-alt" />
|
|
</button>
|
|
</form>
|
|
</div>
|
|
<div className="l-footer">
|
|
<a className="lf-logo" href="">
|
|
<img src={logo} alt="" />
|
|
</a>
|
|
<div className="lf-server">{window.location.host}</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
}
|
|
|
|
const mapDispatchToProps = dispatch => {
|
|
return {
|
|
showAlert: (type, message) =>
|
|
dispatch(actionsAlert.set({ type: type, message: message })),
|
|
clearAlert: () => dispatch(actionsAlert.clear())
|
|
}
|
|
}
|
|
|
|
export default connect(
|
|
state => state,
|
|
mapDispatchToProps
|
|
)(OpenIDLogin)
|