Compare commits
30 Commits
Author | SHA1 | Date | |
---|---|---|---|
170c532443 | |||
|
f71c5996ea | ||
|
a1a701428b | ||
be64023660 | |||
|
aff875eb11 | ||
c9ca2b3858 | |||
|
869dedc580 | ||
0ed7416b49 | |||
|
c20634ea80 | ||
541c744c60 | |||
27e9f08a92 | |||
16775ae2ab | |||
40a3dc9339 | |||
ebae5c29cd | |||
4e2f8f6466 | |||
40840390ae | |||
27dd190fbd | |||
3ea919e06c | |||
52e9be31c3 | |||
b587bc4ceb | |||
c53b7deacb | |||
efd43e42a8 | |||
2624771842 | |||
4430e8c666 | |||
2ba0cf7548 | |||
30939d9b16 | |||
366919b6ef | |||
6253de5599 | |||
3f6003edd7 | |||
e74d0280be |
1
.gitignore
vendored
1
.gitignore
vendored
@ -2,4 +2,3 @@
|
|||||||
vendor
|
vendor
|
||||||
logs
|
logs
|
||||||
cache
|
cache
|
||||||
log
|
|
202
LICENSE
Normal file
202
LICENSE
Normal file
@ -0,0 +1,202 @@
|
|||||||
|
|
||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work.
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following
|
||||||
|
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||||
|
replaced with your own identifying information. (Don't include
|
||||||
|
the brackets!) The text should be enclosed in the appropriate
|
||||||
|
comment syntax for the file format. We also recommend that a
|
||||||
|
file or class name and description of purpose be included on the
|
||||||
|
same "printed page" as the copyright notice for easier
|
||||||
|
identification within third-party archives.
|
||||||
|
|
||||||
|
Copyright (c) 2020 lixunhuan Authors. All Rights Reserved.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
21
config.go
21
config.go
@ -2,21 +2,22 @@ package logger
|
|||||||
|
|
||||||
import "fmt"
|
import "fmt"
|
||||||
|
|
||||||
type mode int
|
type mode string
|
||||||
|
|
||||||
var (
|
const (
|
||||||
Dev mode = 0
|
Dev mode = "development"
|
||||||
Prod mode = 1
|
Prod mode = "production"
|
||||||
)
|
)
|
||||||
|
|
||||||
// LogConfig 日志配置
|
// LogConfig 日志配置
|
||||||
type LogConfig struct {
|
type LogConfig struct {
|
||||||
Mode mode // dev, prod
|
Mode mode `env:"LOG_MODE" envDefault:"production"` // dev, prod
|
||||||
LokiEnable bool
|
LokiEnable bool `env:"LOG_LOKI_ENABLE"` // 是否启用Loki
|
||||||
FileEnable bool
|
FileEnable bool `env:"LOG_FILE_ENABLE"` // 是否输出到文件
|
||||||
LokiHost string
|
LokiHost string `env:"LOG_LOKI_HOST"` // Loki地址
|
||||||
LokiPort int
|
LokiPort int `env:"LOG_LOKI_PORT"` // Loki端口
|
||||||
LokiName string // Loki的job和source名称
|
LokiSource string `env:"LOG_LOKI_SOURCE_NAME"` // Loki的source名称
|
||||||
|
LokiJob string `env:"LOG_LOKI_JOB_NAME"` // Loki的job名称
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c LogConfig) getLokiPushURL() string {
|
func (c LogConfig) getLokiPushURL() string {
|
||||||
|
4
file.go
4
file.go
@ -5,13 +5,15 @@ import (
|
|||||||
"github.com/natefinch/lumberjack"
|
"github.com/natefinch/lumberjack"
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
"go.uber.org/zap/zapcore"
|
"go.uber.org/zap/zapcore"
|
||||||
|
"os"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// 初始化LokiCore,使日志可以输出到文件
|
// 初始化LokiCore,使日志可以输出到文件
|
||||||
func initFileCore() zapcore.Core {
|
func initFileCore() zapcore.Core {
|
||||||
|
hostname, _ := os.Hostname()
|
||||||
lumberJackLogger := &lumberjack.Logger{
|
lumberJackLogger := &lumberjack.Logger{
|
||||||
Filename: "logs/runtime.log", // 日志文件的位置
|
Filename: fmt.Sprintf("logs/runtime-%s.log", hostname), // 日志文件的位置
|
||||||
MaxSize: 10, // 最大10M
|
MaxSize: 10, // 最大10M
|
||||||
MaxBackups: 5, // 保留旧文件的最大个数
|
MaxBackups: 5, // 保留旧文件的最大个数
|
||||||
MaxAge: 30, // 保留旧文件的最大天数
|
MaxAge: 30, // 保留旧文件的最大天数
|
||||||
|
82
go.mod
82
go.mod
@ -1,45 +1,69 @@
|
|||||||
module gitee.ltd/lxh/logger
|
module gitee.ltd/lxh/logger
|
||||||
|
|
||||||
go 1.17
|
go 1.20
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/go-kit/kit v0.10.0
|
github.com/bytedance/sonic v1.12.5
|
||||||
github.com/lixh00/loki-client-go v1.0.1
|
github.com/caarlos0/env/v6 v6.10.1
|
||||||
|
github.com/go-kit/kit v0.12.0
|
||||||
|
github.com/grafana/loki-client-go v0.0.0-20240913122146-e119d400c3a5
|
||||||
github.com/natefinch/lumberjack v2.0.0+incompatible
|
github.com/natefinch/lumberjack v2.0.0+incompatible
|
||||||
github.com/prometheus/common v0.30.0
|
github.com/panjf2000/ants/v2 v2.10.0
|
||||||
go.uber.org/zap v1.19.1
|
github.com/prometheus/common v0.37.0
|
||||||
|
go.uber.org/zap v1.23.0
|
||||||
|
gorm.io/driver/mysql v1.3.2
|
||||||
|
gorm.io/gorm v1.24.0
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/BurntSushi/toml v0.4.1 // indirect
|
github.com/BurntSushi/toml v0.4.1 // indirect
|
||||||
github.com/beorn7/perks v1.0.1 // indirect
|
github.com/beorn7/perks v1.0.1 // indirect
|
||||||
github.com/cespare/xxhash/v2 v2.1.1 // indirect
|
github.com/bytedance/sonic/loader v0.2.0 // indirect
|
||||||
github.com/go-logfmt/logfmt v0.5.0 // indirect
|
github.com/cespare/xxhash/v2 v2.2.0 // indirect
|
||||||
github.com/gogo/protobuf v1.3.1 // indirect
|
github.com/cloudwego/base64x v0.1.4 // indirect
|
||||||
github.com/golang/protobuf v1.4.3 // indirect
|
github.com/cloudwego/iasm v0.2.0 // indirect
|
||||||
github.com/golang/snappy v0.0.2 // indirect
|
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||||
|
github.com/dennwc/varint v1.0.0 // indirect
|
||||||
|
github.com/go-kit/log v0.2.1 // indirect
|
||||||
|
github.com/go-logfmt/logfmt v0.5.1 // indirect
|
||||||
|
github.com/go-sql-driver/mysql v1.6.0 // indirect
|
||||||
|
github.com/gogo/protobuf v1.3.2 // indirect
|
||||||
|
github.com/golang/protobuf v1.5.4 // indirect
|
||||||
|
github.com/golang/snappy v0.0.4 // indirect
|
||||||
|
github.com/grafana/loki/pkg/push v0.0.0-20240912152814-63e84b476a9a // indirect
|
||||||
|
github.com/grafana/regexp v0.0.0-20220304095617-2e8d9baf4ac2 // indirect
|
||||||
|
github.com/jinzhu/inflection v1.0.0 // indirect
|
||||||
|
github.com/jinzhu/now v1.1.4 // indirect
|
||||||
github.com/jpillora/backoff v1.0.0 // indirect
|
github.com/jpillora/backoff v1.0.0 // indirect
|
||||||
github.com/json-iterator/go v1.1.11 // indirect
|
github.com/json-iterator/go v1.1.12 // indirect
|
||||||
github.com/matttproud/golang_protobuf_extensions v1.0.1 // indirect
|
github.com/klauspost/cpuid/v2 v2.0.9 // indirect
|
||||||
github.com/mitchellh/mapstructure v1.2.2 // indirect
|
github.com/matttproud/golang_protobuf_extensions v1.0.2 // indirect
|
||||||
|
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
||||||
github.com/modern-go/reflect2 v1.0.1 // indirect
|
github.com/modern-go/reflect2 v1.0.2 // indirect
|
||||||
github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f // indirect
|
github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f // indirect
|
||||||
github.com/pkg/errors v0.9.1 // indirect
|
github.com/pkg/errors v0.9.1 // indirect
|
||||||
github.com/prometheus/client_golang v1.11.0 // indirect
|
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||||
github.com/prometheus/client_model v0.2.0 // indirect
|
github.com/prometheus/client_golang v1.13.0 // indirect
|
||||||
github.com/prometheus/procfs v0.6.0 // indirect
|
github.com/prometheus/client_model v0.3.0 // indirect
|
||||||
github.com/prometheus/prometheus v1.8.2-0.20201028100903-3245b3267b24 // indirect
|
github.com/prometheus/procfs v0.8.0 // indirect
|
||||||
go.uber.org/atomic v1.7.0 // indirect
|
github.com/prometheus/prometheus v0.35.0 // indirect
|
||||||
go.uber.org/multierr v1.6.0 // indirect
|
github.com/stretchr/testify v1.8.2 // indirect
|
||||||
golang.org/x/net v0.0.0-20210525063256-abc453219eb5 // indirect
|
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
||||||
golang.org/x/oauth2 v0.0.0-20210514164344-f6687ab2804c // indirect
|
go.uber.org/atomic v1.10.0 // indirect
|
||||||
golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40 // indirect
|
go.uber.org/goleak v1.1.12 // indirect
|
||||||
golang.org/x/text v0.3.6 // indirect
|
go.uber.org/multierr v1.8.0 // indirect
|
||||||
google.golang.org/appengine v1.6.6 // indirect
|
golang.org/x/arch v0.0.0-20210923205945-b76863e36670 // indirect
|
||||||
google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d // indirect
|
golang.org/x/exp v0.0.0-20240325151524-a685a6edb6d8 // indirect
|
||||||
google.golang.org/grpc v1.32.0 // indirect
|
golang.org/x/net v0.33.0 // indirect
|
||||||
google.golang.org/protobuf v1.26.0-rc.1 // indirect
|
golang.org/x/oauth2 v0.18.0 // indirect
|
||||||
gopkg.in/natefinch/lumberjack.v2 v2.0.0 // indirect
|
golang.org/x/sync v0.10.0 // indirect
|
||||||
|
golang.org/x/sys v0.28.0 // indirect
|
||||||
|
golang.org/x/text v0.21.0 // indirect
|
||||||
|
google.golang.org/appengine v1.6.8 // indirect
|
||||||
|
google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1 // indirect
|
||||||
|
google.golang.org/grpc v1.64.1 // indirect
|
||||||
|
google.golang.org/protobuf v1.36.5 // indirect
|
||||||
gopkg.in/yaml.v2 v2.4.0 // indirect
|
gopkg.in/yaml.v2 v2.4.0 // indirect
|
||||||
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
)
|
)
|
||||||
|
93
gorm.go
Normal file
93
gorm.go
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
package logger
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
gl "gorm.io/gorm/logger"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var gormZap *zap.SugaredLogger
|
||||||
|
|
||||||
|
// 基于Gorm的日志实现
|
||||||
|
type gormLogger struct {
|
||||||
|
gl.Config
|
||||||
|
}
|
||||||
|
|
||||||
|
// LogMode 实现LogMode接口
|
||||||
|
func (l *gormLogger) LogMode(level gl.LogLevel) gl.Interface {
|
||||||
|
nl := *l
|
||||||
|
nl.LogLevel = level
|
||||||
|
return &nl
|
||||||
|
}
|
||||||
|
|
||||||
|
// Info 实现Info接口
|
||||||
|
func (l gormLogger) Info(ctx context.Context, msg string, data ...interface{}) {
|
||||||
|
if l.LogLevel >= gl.Info {
|
||||||
|
// // 去掉第一行
|
||||||
|
// msg = strings.Join(strings.Split(msg, "\n")[1:], " ")
|
||||||
|
// gormZap.Info(msg)
|
||||||
|
//
|
||||||
|
// l.Printf(msg, append([]interface{}{utils.FileWithLineNum()}, data...)...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Warn 实现Warn接口
|
||||||
|
func (l gormLogger) Warn(ctx context.Context, msg string, data ...interface{}) {
|
||||||
|
if l.LogLevel >= gl.Warn {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 实现Error接口
|
||||||
|
func (l gormLogger) Error(ctx context.Context, msg string, data ...interface{}) {
|
||||||
|
if l.LogLevel >= gl.Error {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Trace 实现Trace接口
|
||||||
|
func (l gormLogger) Trace(ctx context.Context, begin time.Time, fc func() (string, int64), err error) {
|
||||||
|
if l.LogLevel <= gl.Silent {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
elapsed := time.Since(begin)
|
||||||
|
sql, rows := fc()
|
||||||
|
msg := fmt.Sprintf("[%v] [rows:%v] %s", elapsed.String(), rows, sql)
|
||||||
|
if rows == -1 {
|
||||||
|
msg = fmt.Sprintf("[%s] [-] %s", elapsed.String(), sql)
|
||||||
|
}
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case err != nil && l.LogLevel >= gl.Error && (!errors.Is(err, gl.ErrRecordNotFound) || !l.IgnoreRecordNotFoundError):
|
||||||
|
gormZap.Errorf("%s -> %s", err.Error(), sql)
|
||||||
|
case elapsed > l.SlowThreshold && l.SlowThreshold != 0 && l.LogLevel >= gl.Warn:
|
||||||
|
slowLog := fmt.Sprintf("SLOW SQL >= %v", l.SlowThreshold)
|
||||||
|
gormZap.Warnf("%v -> %v", slowLog, sql)
|
||||||
|
case l.LogLevel == gl.Info:
|
||||||
|
gormZap.Info(msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGormLoggerWithConfig ...
|
||||||
|
func NewGormLoggerWithConfig(config gl.Config) gl.Interface {
|
||||||
|
return &gormLogger{config}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultGormLogger 默认的日志实现
|
||||||
|
func DefaultGormLogger() gl.Interface {
|
||||||
|
// 默认日志级别为Info,如果是生产环境,就是Error
|
||||||
|
logLevel := gl.Info
|
||||||
|
if config.Mode == Prod {
|
||||||
|
logLevel = gl.Error
|
||||||
|
}
|
||||||
|
return &gormLogger{gl.Config{
|
||||||
|
SlowThreshold: time.Second, // Slow SQL threshold
|
||||||
|
IgnoreRecordNotFoundError: false, // 忽略没找到结果的错误
|
||||||
|
LogLevel: logLevel, // Log level
|
||||||
|
Colorful: false, // Disable color
|
||||||
|
}}
|
||||||
|
}
|
45
gorm_test.go
Normal file
45
gorm_test.go
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
package logger
|
||||||
|
|
||||||
|
import (
|
||||||
|
"gitee.ltd/lxh/logger/log"
|
||||||
|
"gorm.io/driver/mysql"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
gl "gorm.io/gorm/logger"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGormLogger(t *testing.T) {
|
||||||
|
dsn := "saas:saas123@tcp(10.11.0.10:3307)/saas_tenant?charset=utf8mb4&parseTime=True&loc=Local"
|
||||||
|
|
||||||
|
engine, err := gorm.Open(mysql.Open(dsn), &gorm.Config{Logger: DefaultGormLogger()})
|
||||||
|
if err != nil {
|
||||||
|
log.Panicf("mysql connect error: %s", err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
var count int64
|
||||||
|
if err := engine.Table("t_tenant").Count(&count).Error; err != nil {
|
||||||
|
t.Log(err)
|
||||||
|
}
|
||||||
|
t.Logf("count: %d", count)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGormLoggerWithConfig(t *testing.T) {
|
||||||
|
dsn := "saas:saas123@tcp(10.11.0.10:3307)/saas_tenant?charset=utf8mb4&parseTime=True&loc=Local"
|
||||||
|
|
||||||
|
engine, err := gorm.Open(mysql.Open(dsn), &gorm.Config{Logger: NewGormLoggerWithConfig(gl.Config{
|
||||||
|
SlowThreshold: time.Second, // Slow SQL threshold
|
||||||
|
IgnoreRecordNotFoundError: false, // 忽略没找到结果的错误
|
||||||
|
LogLevel: gl.Warn, // Log level
|
||||||
|
Colorful: false, // Disable color
|
||||||
|
})})
|
||||||
|
if err != nil {
|
||||||
|
log.Panicf("mysql connect error: %s", err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
var count int64
|
||||||
|
if err := engine.Table("t_tenant1").Count(&count).Error; err != nil {
|
||||||
|
t.Log(err)
|
||||||
|
}
|
||||||
|
t.Logf("count: %d", count)
|
||||||
|
}
|
75
log/say.go
Normal file
75
log/say.go
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
package log
|
||||||
|
|
||||||
|
import "go.uber.org/zap"
|
||||||
|
|
||||||
|
// Debug uses fmt.Sprint to construct and log a message.
|
||||||
|
func Debug(args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Debug(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Info uses fmt.Sprint to construct and log a message.
|
||||||
|
func Info(args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Info(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Warn uses fmt.Sprint to construct and log a message.
|
||||||
|
func Warn(args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Warn(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error uses fmt.Sprint to construct and log a message.
|
||||||
|
func Error(args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Error(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Panic uses fmt.Sprint to construct and log a message, then panics.
|
||||||
|
func Panic(args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Panic(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fatal uses fmt.Sprint to construct and log a message, then calls os.Exit.
|
||||||
|
func Fatal(args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Fatal(args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Debugf uses fmt.Sprintf to log a templated message.
|
||||||
|
func Debugf(template string, args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Debugf(template, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Infof uses fmt.Sprintf to log a templated message.
|
||||||
|
func Infof(template string, args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Infof(template, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Warnf uses fmt.Sprintf to log a templated message.
|
||||||
|
func Warnf(template string, args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Warnf(template, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Errorf uses fmt.Sprintf to log a templated message.
|
||||||
|
func Errorf(template string, args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Errorf(template, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Panicf uses fmt.Sprintf to log a templated message, then panics.
|
||||||
|
func Panicf(template string, args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Panicf(template, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fatalf uses fmt.Sprintf to log a templated message, then calls os.Exit.
|
||||||
|
func Fatalf(template string, args ...interface{}) {
|
||||||
|
defer zap.S().Sync()
|
||||||
|
zap.S().Fatalf(template, args...)
|
||||||
|
}
|
26
logger.go
26
logger.go
@ -1,16 +1,30 @@
|
|||||||
package logger
|
package logger
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/caarlos0/env/v6"
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
"go.uber.org/zap/zapcore"
|
"go.uber.org/zap/zapcore"
|
||||||
)
|
)
|
||||||
|
|
||||||
var config LogConfig
|
var config LogConfig
|
||||||
var Say *zap.SugaredLogger
|
var initialized bool
|
||||||
|
|
||||||
// 避免异常,在第一次调用时初始化一个只打印到控制台的logger
|
// 避免异常,在第一次调用时初始化一个只打印到控制台的logger
|
||||||
func init() {
|
func init() {
|
||||||
InitLogger(LogConfig{Mode: Dev, LokiEnable: false, FileEnable: false})
|
if !initialized {
|
||||||
|
// 从环境变量读取配置
|
||||||
|
var c LogConfig
|
||||||
|
if err := env.Parse(&c); err != nil {
|
||||||
|
fmt.Println("日志配置解析错误: " + err.Error())
|
||||||
|
c = LogConfig{Mode: Dev, LokiEnable: false, FileEnable: false}
|
||||||
|
}
|
||||||
|
// 如果值错了,直接默认为Prod
|
||||||
|
if c.Mode != Dev && c.Mode != Prod {
|
||||||
|
c.Mode = Prod
|
||||||
|
}
|
||||||
|
InitLogger(c)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// InitLogger 初始化日志工具
|
// InitLogger 初始化日志工具
|
||||||
@ -32,6 +46,10 @@ func InitLogger(c LogConfig) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 增加 caller 信息
|
// 增加 caller 信息
|
||||||
logger := zap.New(zapcore.NewTee(cores...), zap.AddCaller())
|
// AddCallerSkip 输出的文件名和行号是调用封装函数的位置,而不是调用日志函数的位置
|
||||||
Say = logger.Sugar()
|
logger := zap.New(zapcore.NewTee(cores...), zap.AddCaller(), zap.AddCallerSkip(1))
|
||||||
|
initialized = true
|
||||||
|
// 给GORM单独生成一个
|
||||||
|
gormZap = zap.New(zapcore.NewTee(cores...), zap.AddCaller(), zap.AddCallerSkip(3)).Sugar()
|
||||||
|
zap.ReplaceGlobals(logger)
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,39 @@
|
|||||||
package logger
|
package logger
|
||||||
|
|
||||||
import "testing"
|
import (
|
||||||
|
"gitee.ltd/lxh/logger/log"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
func TestLogger(t *testing.T) {
|
func TestLogger(t *testing.T) {
|
||||||
InitLogger(LogConfig{Mode: Dev, LokiEnable: false, FileEnable: true})
|
InitLogger(LogConfig{Mode: Dev, LokiEnable: false, FileEnable: true})
|
||||||
Say.Debug("芜湖")
|
log.Debug("芜湖")
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLogger1(t *testing.T) {
|
||||||
|
log.Info("我是测试消息")
|
||||||
|
time.Sleep(5 * time.Second)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLogger2(t *testing.T) {
|
||||||
|
InitLogger(LogConfig{Mode: Dev, LokiEnable: false, FileEnable: true})
|
||||||
|
log.Info("我是测试消息")
|
||||||
|
//time.Sleep(5 * time.Second)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestLogger3(t *testing.T) {
|
||||||
|
InitLogger(LogConfig{
|
||||||
|
Mode: Dev,
|
||||||
|
LokiEnable: true,
|
||||||
|
FileEnable: false,
|
||||||
|
LokiHost: "",
|
||||||
|
LokiPort: 0,
|
||||||
|
LokiSource: "test-logger",
|
||||||
|
LokiJob: "test-logger",
|
||||||
|
})
|
||||||
|
|
||||||
|
log.Info("这是info日志")
|
||||||
|
log.Debug("这是debug日志")
|
||||||
|
log.Warn("这是warn日志")
|
||||||
}
|
}
|
||||||
|
28
loki.go
28
loki.go
@ -1,11 +1,12 @@
|
|||||||
package logger
|
package logger
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"github.com/bytedance/sonic"
|
||||||
"github.com/go-kit/kit/log"
|
"github.com/go-kit/kit/log"
|
||||||
"github.com/lixh00/loki-client-go/loki"
|
"github.com/grafana/loki-client-go/loki"
|
||||||
|
"github.com/panjf2000/ants/v2"
|
||||||
"github.com/prometheus/common/model"
|
"github.com/prometheus/common/model"
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
"go.uber.org/zap/zapcore"
|
"go.uber.org/zap/zapcore"
|
||||||
@ -32,7 +33,7 @@ func initLokiCore() zapcore.Core {
|
|||||||
// 格式相关的配置
|
// 格式相关的配置
|
||||||
encoderConfig := zap.NewProductionEncoderConfig()
|
encoderConfig := zap.NewProductionEncoderConfig()
|
||||||
// 修改时间戳的格式
|
// 修改时间戳的格式
|
||||||
encoderConfig.EncodeTime = customTimeEncoder
|
encoderConfig.EncodeTime = customTimeEncoder // zapcore.EpochNanosTimeEncoder
|
||||||
// 日志级别使用大写
|
// 日志级别使用大写
|
||||||
encoderConfig.EncodeLevel = zapcore.CapitalLevelEncoder
|
encoderConfig.EncodeLevel = zapcore.CapitalLevelEncoder
|
||||||
// 将日志级别设置为 DEBUG
|
// 将日志级别设置为 DEBUG
|
||||||
@ -43,7 +44,7 @@ func initLokiCore() zapcore.Core {
|
|||||||
func initLokiClient() {
|
func initLokiClient() {
|
||||||
// 如果Loki配置错误,返回一个nil
|
// 如果Loki配置错误,返回一个nil
|
||||||
if config.LokiHost == "" || config.LokiPort < 1 {
|
if config.LokiHost == "" || config.LokiPort < 1 {
|
||||||
panic(errors.New("Loki配置错误"))
|
panic(errors.New("loki配置错误"))
|
||||||
}
|
}
|
||||||
// 初始化配置
|
// 初始化配置
|
||||||
cfg, _ := loki.NewDefaultConfig(config.getLokiPushURL())
|
cfg, _ := loki.NewDefaultConfig(config.getLokiPushURL())
|
||||||
@ -64,22 +65,25 @@ func (c lokiWriter) Write(p []byte) (int, error) {
|
|||||||
Msg string `json:"msg"` // 日志内容
|
Msg string `json:"msg"` // 日志内容
|
||||||
}
|
}
|
||||||
var li logInfo
|
var li logInfo
|
||||||
err := json.Unmarshal(p, &li)
|
err := sonic.Unmarshal(p, &li)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
label := model.LabelSet{"job": model.LabelValue(config.LokiName)}
|
label := model.LabelSet{"job": model.LabelValue(config.LokiJob)}
|
||||||
label["source"] = model.LabelValue(config.LokiName)
|
label["source"] = model.LabelValue(config.LokiSource)
|
||||||
label["level"] = model.LabelValue(li.Level)
|
label["level"] = model.LabelValue(li.Level)
|
||||||
label["caller"] = model.LabelValue(li.Caller)
|
label["caller"] = model.LabelValue(li.Caller)
|
||||||
// 异步推送消息到服务器
|
|
||||||
go func() {
|
_ = ants.Submit(func() {
|
||||||
err = lokiClient.Handle(label, time.Now().Local(), li.Msg)
|
t, e := time.ParseInLocation("2006-01-02 15:04:05.000", li.Ts, time.Local)
|
||||||
if err != nil {
|
if e != nil {
|
||||||
|
t = time.Now().Local()
|
||||||
|
}
|
||||||
|
if err = lokiClient.Handle(label, t, li.Msg); err != nil {
|
||||||
fmt.Printf("日志推送到Loki失败: %v\n", err.Error())
|
fmt.Printf("日志推送到Loki失败: %v\n", err.Error())
|
||||||
}
|
}
|
||||||
}()
|
})
|
||||||
|
|
||||||
return 0, nil
|
return 0, nil
|
||||||
}
|
}
|
||||||
|
18
readme.md
18
readme.md
@ -5,10 +5,24 @@
|
|||||||
```go
|
```go
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import "gitee.ltd/lxh/logger"
|
import (
|
||||||
|
"gitee.ltd/lxh/logger"
|
||||||
|
"gitee.ltd/lxh/logger/log"
|
||||||
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
logger.InitLogger(logger.LogConfig{Mode: logger.Dev, LokiEnable: false, FileEnable: true})
|
logger.InitLogger(logger.LogConfig{Mode: logger.Dev, LokiEnable: false, FileEnable: true})
|
||||||
logger.Say.Debug("芜湖")
|
log.Debug("芜湖")
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### 环境变量
|
||||||
|
```shell
|
||||||
|
export LOG_MODE=0 # development | production
|
||||||
|
export LOG_LOKI_ENABLE=1 # 是否启用Loki 0: disable, 1: enable
|
||||||
|
export LOG_FILE_ENABLE=0 # 是否启用输出到文件 0: disable, 1: enable
|
||||||
|
export LOG_LOKI_HOST=10.0.0.31 # Loki地址
|
||||||
|
export LOG_LOKI_PORT=3100 # Loki端口
|
||||||
|
export LOG_LOKI_SOURCE_NAME=tests # Loki Source 名称
|
||||||
|
export LOG_LOKI_JOB_NAME=testj # Loki Job 名称
|
||||||
|
```
|
Loading…
x
Reference in New Issue
Block a user