diff --git a/db/redis.go b/db/redis.go index d3c34eb..7832f41 100644 --- a/db/redis.go +++ b/db/redis.go @@ -6,6 +6,7 @@ import ( "errors" "fmt" "io" + "maps" "os" "slices" "strconv" @@ -29,7 +30,7 @@ import ( │NO │ KEY │ VALUE │ PURPOSE │ └───┴──────────────────┴──────────┴──────────────────────────────────────────────────┘ ┌───┬──────────────────┬──────────┬──────────────────────────────────────────────────┐ - │ 1 │ CPE#Cache#Titles │ JSON │ TO CACHE CPE#Titles │ + │ 1 │ CPE#Titles │ JSON │ Get ALL Titles │ └───┴──────────────────┴──────────┴──────────────────────────────────────────────────┘ - Sets @@ -45,8 +46,6 @@ import ( ├─────────────────────────────┼───────────────────────┼────────────────────────────────────┤ │ CPE#DeprecatedCPEs │ CPEURI │ Get DeprecatedCPEs │ ├─────────────────────────────┼───────────────────────┼────────────────────────────────────┤ - │ CPE#Titles │ Title │ Get ALL Titles │ - ├─────────────────────────────┼───────────────────────┼────────────────────────────────────┤ │ CPE#Title#{title} │ CPEURI │ Get CPEURI by title │ └─────────────────────────────┴───────────────────────┴────────────────────────────────────┘ @@ -256,20 +255,38 @@ func (r *RedisDriver) GetSimilarCpesByTitle(query string, n int, algorithm edlib } ctx := context.Background() + t, err := r.conn.Type(ctx, titleListKey).Result() + if err != nil { + return nil, xerrors.Errorf("Failed to TYPE CPE#Titles. err: %w", err) + } + var ts []string - bs, err := r.conn.Get(ctx, titleListCacheKey).Bytes() - if err == nil { + switch t { + case "string": + bs, err := r.conn.Get(ctx, titleListKey).Bytes() + if err != nil { + return nil, xerrors.Errorf("Failed to Get Titles. err: %w", err) + } if err := json.Unmarshal(bs, &ts); err != nil { return nil, xerrors.Errorf("Failed to Unmarshal JSON. err: %w", err) } - } else { - if !xerrors.Is(err, redis.Nil) { - return nil, xerrors.Errorf("Failed to Get Titles. err: %w", err) - } - ts, err = r.conn.SMembers(ctx, titleListKey).Result() - if err != nil { - return nil, xerrors.Errorf("Failed to SMembers Titles. err: %w", err) + case "set": // backward compatibility: https://github.com/vulsio/go-cpe-dictionary/pull/186 + bs, err := r.conn.Get(ctx, titleListCacheKey).Bytes() + if err == nil { + if err := json.Unmarshal(bs, &ts); err != nil { + return nil, xerrors.Errorf("Failed to Unmarshal JSON. err: %w", err) + } + } else { + if !xerrors.Is(err, redis.Nil) { + return nil, xerrors.Errorf("Failed to Get Titles. err: %w", err) + } + ts, err = r.conn.SMembers(ctx, titleListKey).Result() + if err != nil { + return nil, xerrors.Errorf("Failed to SMembers Titles. err: %w", err) + } } + default: + return nil, xerrors.Errorf("unexpected CPE#Titles type. expected: %q, actual: %q", []string{"string", "set"}, t) } if len(ts) < n { @@ -310,7 +327,6 @@ func (r *RedisDriver) InsertCpes(fetchType models.FetchType, cpes models.Fetched "VendorProducts": {}, "DeprecatedVendorProducts": {}, "DeprecatedCPEs": {}, - "Titles": {}, "Title": {}, } oldDepsStr, err := r.conn.HGet(ctx, depKey, string(fetchType)).Result() @@ -323,7 +339,6 @@ func (r *RedisDriver) InsertCpes(fetchType models.FetchType, cpes models.Fetched "VendorProducts": {}, "DeprecatedVendorProducts": {}, "DeprecatedCPEs": {}, - "Titles": {}, "Title": {} }` } @@ -332,6 +347,50 @@ func (r *RedisDriver) InsertCpes(fetchType models.FetchType, cpes models.Fetched return xerrors.Errorf("Failed to unmarshal JSON. err: %w", err) } + titles := make(map[string]struct{}) + exists, err := r.conn.Exists(ctx, titleListKey).Result() + if err != nil { + return xerrors.Errorf("Failed to Exists CPE#Titles. err: %w", err) + } + if exists > 0 { + t, err := r.conn.Type(ctx, titleListKey).Result() + if err != nil { + return xerrors.Errorf("Failed to TYPE CPE#Titles. err: %w", err) + } + + var ts []string + switch t { + case "string": + bs, err := r.conn.Get(ctx, titleListKey).Bytes() + if err != nil { + return xerrors.Errorf("Failed to Get Titles. err: %w", err) + } + if err := json.Unmarshal(bs, &ts); err != nil { + return xerrors.Errorf("Failed to Unmarshal JSON. err: %w", err) + } + case "set": // backward compatibility: https://github.com/vulsio/go-cpe-dictionary/pull/186 + bs, err := r.conn.Get(ctx, titleListCacheKey).Bytes() + if err == nil { + if err := json.Unmarshal(bs, &ts); err != nil { + return xerrors.Errorf("Failed to Unmarshal JSON. err: %w", err) + } + } else { + if !xerrors.Is(err, redis.Nil) { + return xerrors.Errorf("Failed to Get Titles. err: %w", err) + } + ts, err = r.conn.SMembers(ctx, titleListKey).Result() + if err != nil { + return xerrors.Errorf("Failed to SMembers Titles. err: %w", err) + } + } + default: + return xerrors.Errorf("unexpected CPE#Titles type. expected: %q, actual: %q", []string{"string", "set"}, t) + } + for _, t := range ts { + titles[t] = struct{}{} + } + } + bar := pb.StartNew(len(cpes.CPEs) + len(cpes.Deprecated) + 1).SetWriter(func() io.Writer { if viper.GetBool("log-json") { return io.Discard @@ -357,17 +416,18 @@ func (r *RedisDriver) InsertCpes(fetchType models.FetchType, cpes models.Fetched vendorProductStr := fmt.Sprintf("%s%s%s", c.Vendor, vpSeparator, c.Product) if c.Deprecated { _ = pipe.SAdd(ctx, deprecatedCPEsKey, c.CpeURI) - newDeps["DeprecatedCPEs"][c.CpeURI] = map[string]struct{}{} + newDeps["DeprecatedCPEs"][c.CpeURI] = nil delete(oldDeps["DeprecatedCPEs"], c.CpeURI) _ = pipe.SAdd(ctx, deprecatedVPListKey, vendorProductStr) - newDeps["DeprecatedVendorProducts"][vendorProductStr] = map[string]struct{}{} + newDeps["DeprecatedVendorProducts"][vendorProductStr] = nil delete(oldDeps["DeprecatedVendorProducts"], vendorProductStr) } else { _ = pipe.SAdd(ctx, vpListKey, vendorProductStr) - newDeps["VendorProducts"][vendorProductStr] = map[string]struct{}{} + newDeps["VendorProducts"][vendorProductStr] = nil delete(oldDeps["VendorProducts"], vendorProductStr) } + _ = pipe.SAdd(ctx, fmt.Sprintf(vpKeyFormat, c.Vendor, c.Product), c.CpeURI) if _, ok := newDeps["VP"][vendorProductStr]; !ok { newDeps["VP"][vendorProductStr] = map[string]struct{}{} @@ -379,9 +439,8 @@ func (r *RedisDriver) InsertCpes(fetchType models.FetchType, cpes models.Fetched delete(oldDeps["VP"], vendorProductStr) } } - _ = pipe.SAdd(ctx, titleListKey, c.Title) - newDeps["Titles"][c.Title] = map[string]struct{}{} - delete(oldDeps["Titles"], c.Title) + + titles[c.Title] = struct{}{} _ = pipe.SAdd(ctx, fmt.Sprintf(titleKeyFormat, c.Title), c.CpeURI) if _, ok := newDeps["Title"][c.Title]; !ok { @@ -402,16 +461,12 @@ func (r *RedisDriver) InsertCpes(fetchType models.FetchType, cpes models.Fetched } } - ts, err := r.conn.SMembers(ctx, titleListKey).Result() - if err != nil { - return xerrors.Errorf("Failed to SMembers Titles. err: %w", err) - } - bs, err := json.Marshal(ts) + bs, err := json.Marshal(slices.Collect(maps.Keys(titles))) if err != nil { return xerrors.Errorf("Failed to Marshal JSON. err: %w", err) } - if err := r.conn.Set(ctx, titleListCacheKey, string(bs), 0).Err(); err != nil { - return xerrors.Errorf("Failed to SET Titles Cache. err: %w", err) + if err := r.conn.Set(ctx, titleListKey, string(bs), 0).Err(); err != nil { + return xerrors.Errorf("Failed to SET Titles. err: %w", err) } bar.Increment() @@ -433,9 +488,7 @@ func (r *RedisDriver) InsertCpes(fetchType models.FetchType, cpes models.Fetched for cpeURI := range oldDeps["DeprecatedCPEs"] { _ = pipe.SRem(ctx, deprecatedCPEsKey, cpeURI) } - for title := range oldDeps["Titles"] { - _ = pipe.SRem(ctx, titleListKey, title) - } + _ = pipe.Del(ctx, titleListCacheKey) for title, cpeURIs := range oldDeps["Title"] { for cpeURI := range cpeURIs { _ = pipe.SRem(ctx, fmt.Sprintf(titleKeyFormat, title), cpeURI)