2016-06-07 11:59:17 +02:00
|
|
|
/*
|
2021-04-02 01:50:11 +02:00
|
|
|
Copyright 2021 GitHub Inc.
|
2016-06-07 11:59:17 +02:00
|
|
|
See https://github.com/github/gh-ost/blob/master/LICENSE
|
|
|
|
*/
|
|
|
|
|
|
|
|
package logic
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"fmt"
|
2016-09-12 12:38:14 +02:00
|
|
|
"io"
|
2016-06-07 11:59:17 +02:00
|
|
|
"net"
|
|
|
|
"os"
|
2016-09-12 12:38:14 +02:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"sync/atomic"
|
2016-06-07 11:59:17 +02:00
|
|
|
|
|
|
|
"github.com/github/gh-ost/go/base"
|
|
|
|
)
|
|
|
|
|
2016-09-12 12:38:14 +02:00
|
|
|
type printStatusFunc func(PrintStatusRule, io.Writer)
|
2016-06-07 11:59:17 +02:00
|
|
|
|
|
|
|
// Server listens for requests on a socket file or via TCP
|
|
|
|
type Server struct {
|
|
|
|
migrationContext *base.MigrationContext
|
|
|
|
unixListener net.Listener
|
|
|
|
tcpListener net.Listener
|
2016-09-12 12:38:14 +02:00
|
|
|
hooksExecutor *HooksExecutor
|
|
|
|
printStatus printStatusFunc
|
2016-06-07 11:59:17 +02:00
|
|
|
}
|
|
|
|
|
2017-08-08 13:36:54 -07:00
|
|
|
func NewServer(migrationContext *base.MigrationContext, hooksExecutor *HooksExecutor, printStatus printStatusFunc) *Server {
|
2016-06-07 11:59:17 +02:00
|
|
|
return &Server{
|
2017-08-08 13:36:54 -07:00
|
|
|
migrationContext: migrationContext,
|
2016-09-12 12:38:14 +02:00
|
|
|
hooksExecutor: hooksExecutor,
|
|
|
|
printStatus: printStatus,
|
2016-06-07 11:59:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *Server) BindSocketFile() (err error) {
|
|
|
|
if this.migrationContext.ServeSocketFile == "" {
|
|
|
|
return nil
|
|
|
|
}
|
2016-07-22 17:34:18 +02:00
|
|
|
if this.migrationContext.DropServeSocket && base.FileExists(this.migrationContext.ServeSocketFile) {
|
2016-06-07 11:59:17 +02:00
|
|
|
os.Remove(this.migrationContext.ServeSocketFile)
|
|
|
|
}
|
|
|
|
this.unixListener, err = net.Listen("unix", this.migrationContext.ServeSocketFile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-10-07 11:10:36 -04:00
|
|
|
this.migrationContext.Log.Infof("Listening on unix socket file: %s", this.migrationContext.ServeSocketFile)
|
2016-06-07 11:59:17 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-08-11 09:01:14 +02:00
|
|
|
func (this *Server) RemoveSocketFile() (err error) {
|
2019-10-07 11:10:36 -04:00
|
|
|
this.migrationContext.Log.Infof("Removing socket file: %s", this.migrationContext.ServeSocketFile)
|
2016-08-11 09:01:14 +02:00
|
|
|
return os.Remove(this.migrationContext.ServeSocketFile)
|
|
|
|
}
|
|
|
|
|
2016-06-07 11:59:17 +02:00
|
|
|
func (this *Server) BindTCPPort() (err error) {
|
|
|
|
if this.migrationContext.ServeTCPPort == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
this.tcpListener, err = net.Listen("tcp", fmt.Sprintf(":%d", this.migrationContext.ServeTCPPort))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-10-07 11:10:36 -04:00
|
|
|
this.migrationContext.Log.Infof("Listening on tcp port: %d", this.migrationContext.ServeTCPPort)
|
2016-06-07 11:59:17 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-06-19 17:55:37 +02:00
|
|
|
// Serve begins listening & serving on whichever device was configured
|
2016-06-07 11:59:17 +02:00
|
|
|
func (this *Server) Serve() (err error) {
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
conn, err := this.unixListener.Accept()
|
|
|
|
if err != nil {
|
2019-10-07 11:10:36 -04:00
|
|
|
this.migrationContext.Log.Errore(err)
|
2016-06-07 11:59:17 +02:00
|
|
|
}
|
|
|
|
go this.handleConnection(conn)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
2016-06-07 14:24:30 +02:00
|
|
|
if this.tcpListener == nil {
|
|
|
|
return
|
|
|
|
}
|
2016-06-07 11:59:17 +02:00
|
|
|
for {
|
|
|
|
conn, err := this.tcpListener.Accept()
|
|
|
|
if err != nil {
|
2019-10-07 11:10:36 -04:00
|
|
|
this.migrationContext.Log.Errore(err)
|
2016-06-07 11:59:17 +02:00
|
|
|
}
|
|
|
|
go this.handleConnection(conn)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *Server) handleConnection(conn net.Conn) (err error) {
|
2017-04-13 08:27:42 +03:00
|
|
|
if conn != nil {
|
|
|
|
defer conn.Close()
|
|
|
|
}
|
2016-06-07 11:59:17 +02:00
|
|
|
command, _, err := bufio.NewReader(conn).ReadLine()
|
2017-04-13 08:27:42 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
return this.onServerCommand(string(command), bufio.NewWriter(conn))
|
|
|
|
}
|
|
|
|
|
|
|
|
// onServerCommand responds to a user's interactive command
|
|
|
|
func (this *Server) onServerCommand(command string, writer *bufio.Writer) (err error) {
|
|
|
|
defer writer.Flush()
|
|
|
|
|
|
|
|
printStatusRule, err := this.applyServerCommand(command, writer)
|
|
|
|
if err == nil {
|
|
|
|
this.printStatus(printStatusRule, writer)
|
|
|
|
} else {
|
|
|
|
fmt.Fprintf(writer, "%s\n", err.Error())
|
|
|
|
}
|
2019-10-07 11:10:36 -04:00
|
|
|
return this.migrationContext.Log.Errore(err)
|
2016-09-12 12:38:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// applyServerCommand parses and executes commands by user
|
|
|
|
func (this *Server) applyServerCommand(command string, writer *bufio.Writer) (printStatusRule PrintStatusRule, err error) {
|
|
|
|
printStatusRule = NoPrintStatusRule
|
|
|
|
|
|
|
|
tokens := strings.SplitN(command, "=", 2)
|
|
|
|
command = strings.TrimSpace(tokens[0])
|
|
|
|
arg := ""
|
|
|
|
if len(tokens) > 1 {
|
|
|
|
arg = strings.TrimSpace(tokens[1])
|
2018-02-25 19:17:37 +02:00
|
|
|
if unquoted, err := strconv.Unquote(arg); err == nil {
|
|
|
|
arg = unquoted
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
}
|
2017-01-29 09:25:29 +02:00
|
|
|
argIsQuestion := (arg == "?")
|
2016-09-12 12:38:14 +02:00
|
|
|
throttleHint := "# Note: you may only throttle for as long as your binary logs are not purged\n"
|
|
|
|
|
|
|
|
if err := this.hooksExecutor.onInteractiveCommand(command); err != nil {
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
|
|
|
|
|
|
|
switch command {
|
|
|
|
case "help":
|
|
|
|
{
|
2019-08-11 15:06:03 +03:00
|
|
|
fmt.Fprint(writer, `available commands:
|
2016-09-12 12:38:14 +02:00
|
|
|
status # Print a detailed status message
|
|
|
|
sup # Print a short status message
|
2021-04-02 01:50:11 +02:00
|
|
|
coordinates # Print the currently inspected coordinates
|
2021-04-02 16:57:13 +02:00
|
|
|
applier # Print the hostname of the applier
|
|
|
|
inspector # Print the hostname of the inspector
|
2016-09-12 12:38:14 +02:00
|
|
|
chunk-size=<newsize> # Set a new chunk-size
|
2017-07-19 16:48:22 +03:00
|
|
|
dml-batch-size=<newsize> # Set a new dml-batch-size
|
2017-11-08 00:44:30 +00:00
|
|
|
nice-ratio=<ratio> # Set a new nice-ratio, immediate sleep after each row-copy operation, float (examples: 0 is aggressive, 0.7 adds 70% runtime, 1.0 doubles runtime, 2.0 triples runtime, ...)
|
2016-09-12 12:38:14 +02:00
|
|
|
critical-load=<load> # Set a new set of max-load thresholds
|
|
|
|
max-lag-millis=<max-lag> # Set a new replication lag threshold
|
|
|
|
replication-lag-query=<query> # Set a new query that determines replication lag (no quotes)
|
|
|
|
max-load=<load> # Set a new set of max-load thresholds
|
|
|
|
throttle-query=<query> # Set a new throttle-query (no quotes)
|
2017-03-26 13:10:34 +03:00
|
|
|
throttle-http=<URL> # Set a new throttle URL
|
2016-09-12 12:38:14 +02:00
|
|
|
throttle-control-replicas=<replicas> # Set a new comma delimited list of throttle control replicas
|
|
|
|
throttle # Force throttling
|
|
|
|
no-throttle # End forced throttling (other throttling may still apply)
|
|
|
|
unpostpone # Bail out a cut-over postpone; proceed to cut-over
|
|
|
|
panic # panic and quit without cleanup
|
|
|
|
help # This message
|
2017-01-29 09:25:29 +02:00
|
|
|
- use '?' (question mark) as argument to get info rather than set. e.g. "max-load=?" will just print out current max-load.
|
2016-09-12 12:38:14 +02:00
|
|
|
`)
|
|
|
|
}
|
|
|
|
case "sup":
|
|
|
|
return ForcePrintStatusOnlyRule, nil
|
|
|
|
case "info", "status":
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
2017-04-28 15:50:51 -07:00
|
|
|
case "coordinates":
|
|
|
|
{
|
|
|
|
if argIsQuestion || arg == "" {
|
|
|
|
fmt.Fprintf(writer, "%+v\n", this.migrationContext.GetRecentBinlogCoordinates())
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
|
|
|
return NoPrintStatusRule, fmt.Errorf("coordinates are read-only")
|
|
|
|
}
|
2021-04-02 16:57:13 +02:00
|
|
|
case "applier":
|
2021-04-03 23:24:29 +02:00
|
|
|
if this.migrationContext.ApplierConnectionConfig != nil && this.migrationContext.ApplierConnectionConfig.ImpliedKey != nil {
|
|
|
|
fmt.Fprintf(writer, "Host: %s, Version: %s\n",
|
|
|
|
this.migrationContext.ApplierConnectionConfig.ImpliedKey.String(),
|
|
|
|
this.migrationContext.ApplierMySQLVersion,
|
|
|
|
)
|
|
|
|
}
|
2021-04-02 16:57:13 +02:00
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
case "inspector":
|
2021-04-03 23:24:29 +02:00
|
|
|
if this.migrationContext.InspectorConnectionConfig != nil && this.migrationContext.InspectorConnectionConfig.ImpliedKey != nil {
|
|
|
|
fmt.Fprintf(writer, "Host: %s, Version: %s\n",
|
|
|
|
this.migrationContext.InspectorConnectionConfig.ImpliedKey.String(),
|
|
|
|
this.migrationContext.InspectorMySQLVersion,
|
|
|
|
)
|
|
|
|
}
|
2021-04-02 01:50:11 +02:00
|
|
|
return NoPrintStatusRule, nil
|
2016-09-12 12:38:14 +02:00
|
|
|
case "chunk-size":
|
|
|
|
{
|
2017-01-29 09:25:29 +02:00
|
|
|
if argIsQuestion {
|
|
|
|
fmt.Fprintf(writer, "%+v\n", atomic.LoadInt64(&this.migrationContext.ChunkSize))
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
if chunkSize, err := strconv.Atoi(arg); err != nil {
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
} else {
|
|
|
|
this.migrationContext.SetChunkSize(int64(chunkSize))
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
}
|
2017-07-19 16:44:18 +03:00
|
|
|
case "dml-batch-size":
|
|
|
|
{
|
|
|
|
if argIsQuestion {
|
|
|
|
fmt.Fprintf(writer, "%+v\n", atomic.LoadInt64(&this.migrationContext.DMLBatchSize))
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
|
|
|
if dmlBatchSize, err := strconv.Atoi(arg); err != nil {
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
} else {
|
|
|
|
this.migrationContext.SetDMLBatchSize(int64(dmlBatchSize))
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
case "max-lag-millis":
|
|
|
|
{
|
2017-01-29 09:25:29 +02:00
|
|
|
if argIsQuestion {
|
|
|
|
fmt.Fprintf(writer, "%+v\n", atomic.LoadInt64(&this.migrationContext.MaxLagMillisecondsThrottleThreshold))
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
if maxLagMillis, err := strconv.Atoi(arg); err != nil {
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
} else {
|
|
|
|
this.migrationContext.SetMaxLagMillisecondsThrottleThreshold(int64(maxLagMillis))
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "replication-lag-query":
|
|
|
|
{
|
2016-12-26 21:38:37 +02:00
|
|
|
return NoPrintStatusRule, fmt.Errorf("replication-lag-query is deprecated. gh-ost uses an internal, subsecond resolution query")
|
2016-09-12 12:38:14 +02:00
|
|
|
}
|
|
|
|
case "nice-ratio":
|
|
|
|
{
|
2017-01-29 09:25:29 +02:00
|
|
|
if argIsQuestion {
|
|
|
|
fmt.Fprintf(writer, "%+v\n", this.migrationContext.GetNiceRatio())
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
if niceRatio, err := strconv.ParseFloat(arg, 64); err != nil {
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
} else {
|
|
|
|
this.migrationContext.SetNiceRatio(niceRatio)
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case "max-load":
|
|
|
|
{
|
2017-01-29 09:25:29 +02:00
|
|
|
if argIsQuestion {
|
|
|
|
maxLoad := this.migrationContext.GetMaxLoad()
|
|
|
|
fmt.Fprintf(writer, "%s\n", maxLoad.String())
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
if err := this.migrationContext.ReadMaxLoad(arg); err != nil {
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
case "critical-load":
|
|
|
|
{
|
2017-01-29 09:25:29 +02:00
|
|
|
if argIsQuestion {
|
|
|
|
criticalLoad := this.migrationContext.GetCriticalLoad()
|
|
|
|
fmt.Fprintf(writer, "%s\n", criticalLoad.String())
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
if err := this.migrationContext.ReadCriticalLoad(arg); err != nil {
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
case "throttle-query":
|
|
|
|
{
|
2017-01-29 09:25:29 +02:00
|
|
|
if argIsQuestion {
|
|
|
|
fmt.Fprintf(writer, "%+v\n", this.migrationContext.GetThrottleQuery())
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
this.migrationContext.SetThrottleQuery(arg)
|
|
|
|
fmt.Fprintf(writer, throttleHint)
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
2017-03-26 13:10:34 +03:00
|
|
|
case "throttle-http":
|
|
|
|
{
|
|
|
|
if argIsQuestion {
|
|
|
|
fmt.Fprintf(writer, "%+v\n", this.migrationContext.GetThrottleHTTP())
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
|
|
|
this.migrationContext.SetThrottleHTTP(arg)
|
|
|
|
fmt.Fprintf(writer, throttleHint)
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
case "throttle-control-replicas":
|
|
|
|
{
|
2017-01-29 09:25:29 +02:00
|
|
|
if argIsQuestion {
|
|
|
|
fmt.Fprintf(writer, "%s\n", this.migrationContext.GetThrottleControlReplicaKeys().ToCommaDelimitedList())
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
if err := this.migrationContext.ReadThrottleControlReplicaKeys(arg); err != nil {
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
|
|
|
fmt.Fprintf(writer, "%s\n", this.migrationContext.GetThrottleControlReplicaKeys().ToCommaDelimitedList())
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
case "throttle", "pause", "suspend":
|
|
|
|
{
|
2019-02-25 14:02:57 +02:00
|
|
|
if arg != "" && arg != this.migrationContext.OriginalTableName {
|
|
|
|
// User explicitly provided table name. This is a courtesy protection mechanism
|
|
|
|
err := fmt.Errorf("User commanded 'throttle' on %s, but migrated table is %s; ignoring request.", arg, this.migrationContext.OriginalTableName)
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
atomic.StoreInt64(&this.migrationContext.ThrottleCommandedByUser, 1)
|
|
|
|
fmt.Fprintf(writer, throttleHint)
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
case "no-throttle", "unthrottle", "resume", "continue":
|
|
|
|
{
|
2019-02-25 14:02:57 +02:00
|
|
|
if arg != "" && arg != this.migrationContext.OriginalTableName {
|
|
|
|
// User explicitly provided table name. This is a courtesy protection mechanism
|
|
|
|
err := fmt.Errorf("User commanded 'no-throttle' on %s, but migrated table is %s; ignoring request.", arg, this.migrationContext.OriginalTableName)
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
2016-09-12 12:38:14 +02:00
|
|
|
atomic.StoreInt64(&this.migrationContext.ThrottleCommandedByUser, 0)
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
case "unpostpone", "no-postpone", "cut-over":
|
|
|
|
{
|
2016-09-12 19:17:36 +02:00
|
|
|
if arg == "" && this.migrationContext.ForceNamedCutOverCommand {
|
|
|
|
err := fmt.Errorf("User commanded 'unpostpone' without specifying table name, but --force-named-cut-over is set")
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
|
|
|
if arg != "" && arg != this.migrationContext.OriginalTableName {
|
2017-11-08 00:47:54 +00:00
|
|
|
// User explicitly provided table name. This is a courtesy protection mechanism
|
2017-11-08 00:49:06 +00:00
|
|
|
err := fmt.Errorf("User commanded 'unpostpone' on %s, but migrated table is %s; ignoring request.", arg, this.migrationContext.OriginalTableName)
|
2016-09-12 19:17:36 +02:00
|
|
|
return NoPrintStatusRule, err
|
2016-09-12 12:38:14 +02:00
|
|
|
}
|
|
|
|
if atomic.LoadInt64(&this.migrationContext.IsPostponingCutOver) > 0 {
|
|
|
|
atomic.StoreInt64(&this.migrationContext.UserCommandedUnpostponeFlag, 1)
|
|
|
|
fmt.Fprintf(writer, "Unpostponed\n")
|
|
|
|
return ForcePrintStatusAndHintRule, nil
|
|
|
|
}
|
|
|
|
fmt.Fprintf(writer, "You may only invoke this when gh-ost is actively postponing migration. At this time it is not.\n")
|
|
|
|
return NoPrintStatusRule, nil
|
|
|
|
}
|
|
|
|
case "panic":
|
|
|
|
{
|
2019-01-14 13:27:44 +02:00
|
|
|
if arg == "" && this.migrationContext.ForceNamedPanicCommand {
|
|
|
|
err := fmt.Errorf("User commanded 'panic' without specifying table name, but --force-named-panic is set")
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
|
|
|
if arg != "" && arg != this.migrationContext.OriginalTableName {
|
|
|
|
// User explicitly provided table name. This is a courtesy protection mechanism
|
|
|
|
err := fmt.Errorf("User commanded 'panic' on %s, but migrated table is %s; ignoring request.", arg, this.migrationContext.OriginalTableName)
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
2019-08-12 12:40:31 -07:00
|
|
|
err := fmt.Errorf("User commanded 'panic'. The migration will be aborted without cleanup. Please drop the gh-ost tables before trying again.")
|
2016-09-12 12:38:14 +02:00
|
|
|
this.migrationContext.PanicAbort <- err
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
err = fmt.Errorf("Unknown command: %s", command)
|
|
|
|
return NoPrintStatusRule, err
|
|
|
|
}
|
|
|
|
return NoPrintStatusRule, nil
|
2016-06-07 11:59:17 +02:00
|
|
|
}
|