ghost 4.39.0 → 4.41.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (59) hide show
  1. package/Gruntfile.js +1 -1
  2. package/core/built/assets/{ghost-dark-498ff8339a89bb68c3f78f59bee4146e.css → ghost-dark-6fbe502f2bb2cde92e15b2f1a9da57a0.css} +1 -1
  3. package/core/built/assets/{ghost.min-77b93478f83b0def6ddc5a4f23ce963e.css → ghost.min-09301e5bd933cf6d24368e98a4d898a9.css} +1 -1
  4. package/core/built/assets/{ghost.min-e6559d901897066aa6a6d4145e3728ed.js → ghost.min-ee1d1d48a30dbd67513f647f360b39e3.js} +220 -225
  5. package/core/built/assets/img/themes/Headline-c5070cf549e797a6a72b87237caa1617.jpg +0 -0
  6. package/core/built/assets/{vendor.min-c39476bced9adb98ee2b292d01c7a8f4.js → vendor.min-9094db77ba3190cb10876f8e42e1d90d.js} +153 -143
  7. package/core/frontend/public/ghost.min.css +1 -1
  8. package/core/frontend/services/theme-engine/middleware/update-local-template-options.js +3 -1
  9. package/core/server/api/canary/identities.js +0 -1
  10. package/core/server/api/canary/members.js +2 -6
  11. package/core/server/api/canary/utils/serializers/input/tiers.js +17 -0
  12. package/core/server/api/canary/utils/serializers/output/actions.js +2 -2
  13. package/core/server/api/canary/utils/serializers/output/authentication.js +3 -3
  14. package/core/server/api/canary/utils/serializers/output/authors.js +3 -3
  15. package/core/server/api/canary/utils/serializers/output/email-posts.js +2 -2
  16. package/core/server/api/canary/utils/serializers/output/emails.js +3 -3
  17. package/core/server/api/canary/utils/serializers/output/images.js +2 -2
  18. package/core/server/api/canary/utils/serializers/output/integrations.js +5 -6
  19. package/core/server/api/canary/utils/serializers/output/labels.js +3 -3
  20. package/core/server/api/canary/utils/serializers/output/mappers/actions.js +7 -0
  21. package/core/server/api/canary/utils/serializers/output/mappers/emails.js +17 -0
  22. package/core/server/api/canary/utils/serializers/output/mappers/images.js +5 -0
  23. package/core/server/api/canary/utils/serializers/output/mappers/index.js +12 -0
  24. package/core/server/api/canary/utils/serializers/output/mappers/integrations.js +13 -0
  25. package/core/server/api/canary/utils/serializers/output/mappers/labels.js +4 -0
  26. package/core/server/api/canary/utils/serializers/output/mappers/pages.js +11 -0
  27. package/core/server/api/canary/utils/serializers/output/mappers/posts.js +101 -0
  28. package/core/server/api/canary/utils/serializers/output/mappers/settings.js +37 -0
  29. package/core/server/api/canary/utils/serializers/output/mappers/tags.js +11 -0
  30. package/core/server/api/canary/utils/serializers/output/mappers/users.js +12 -0
  31. package/core/server/api/canary/utils/serializers/output/members.js +2 -7
  32. package/core/server/api/canary/utils/serializers/output/pages.js +3 -3
  33. package/core/server/api/canary/utils/serializers/output/posts.js +3 -3
  34. package/core/server/api/canary/utils/serializers/output/preview.js +2 -2
  35. package/core/server/api/canary/utils/serializers/output/settings.js +2 -2
  36. package/core/server/api/canary/utils/serializers/output/tags.js +3 -3
  37. package/core/server/api/canary/utils/serializers/output/users.js +3 -3
  38. package/core/server/data/exporter/table-lists.js +1 -0
  39. package/core/server/data/migrations/versions/4.40/2022-03-07-14-37-add-members-cancel-events-table.js +8 -0
  40. package/core/server/data/migrations/versions/4.40/2022-03-15-06-40-add-offers-admin-integration-permission-roles.js +23 -0
  41. package/core/server/data/migrations/versions/4.40/2022-03-15-06-40-add-tiers-admin-integration-permission-roles.js +20 -0
  42. package/core/server/data/schema/fixtures/fixtures.json +3 -1
  43. package/core/server/data/schema/schema.js +6 -0
  44. package/core/server/models/member-cancel-event.js +28 -0
  45. package/core/server/services/mega/template.js +1 -0
  46. package/core/server/services/members/api.js +1 -0
  47. package/core/server/services/oembed.js +2 -1
  48. package/core/server/services/themes/validate.js +3 -3
  49. package/core/server/web/admin/views/default-prod.html +4 -4
  50. package/core/server/web/admin/views/default.html +4 -4
  51. package/core/server/web/api/app.js +3 -3
  52. package/core/server/web/api/canary/admin/middleware.js +2 -0
  53. package/core/shared/config/defaults.json +2 -2
  54. package/core/shared/config/overrides.json +5 -1
  55. package/core/shared/labs.js +3 -3
  56. package/package.json +23 -23
  57. package/yarn.lock +298 -595
  58. package/core/server/api/canary/utils/serializers/output/utils/mapper.js +0 -213
  59. package/core/server/frontend/ghost.min.css +0 -1
@@ -1,213 +0,0 @@
1
- const _ = require('lodash');
2
- const utils = require('../../../index');
3
- const url = require('./url');
4
- const date = require('./date');
5
- const gating = require('./post-gating');
6
- const clean = require('./clean');
7
- const extraAttrs = require('./extra-attrs');
8
- const postsMetaSchema = require('../../../../../../data/schema').tables.posts_meta;
9
- const mega = require('../../../../../../services/mega');
10
- const labsService = require('../../../../../../../shared/labs');
11
-
12
- const getPostServiceInstance = require('../../../../../../services/posts/posts-service');
13
- const postsService = getPostServiceInstance('canary');
14
-
15
- const mapUser = (model, frame) => {
16
- const jsonModel = model.toJSON ? model.toJSON(frame.options) : model;
17
-
18
- url.forUser(model.id, jsonModel, frame.options);
19
-
20
- clean.author(jsonModel, frame);
21
-
22
- return jsonModel;
23
- };
24
-
25
- const mapTag = (model, frame) => {
26
- const jsonModel = model.toJSON ? model.toJSON(frame.options) : model;
27
-
28
- url.forTag(model.id, jsonModel, frame.options);
29
- clean.tag(jsonModel, frame);
30
-
31
- return jsonModel;
32
- };
33
-
34
- const mapPost = async (model, frame, options = {}) => {
35
- const {tiers: tiersData} = options || {};
36
- const extendedOptions = Object.assign(_.cloneDeep(frame.options), {
37
- extraProperties: ['canonical_url']
38
- });
39
-
40
- const jsonModel = model.toJSON(extendedOptions);
41
-
42
- url.forPost(model.id, jsonModel, frame);
43
-
44
- extraAttrs.forPost(frame, model, jsonModel);
45
-
46
- // Attach tiers to custom nql visibility filter
47
- if (labsService.isSet('multipleProducts')
48
- && jsonModel.visibility
49
- ) {
50
- if (['members', 'public'].includes(jsonModel.visibility) && jsonModel.tiers) {
51
- jsonModel.tiers = tiersData || [];
52
- }
53
-
54
- if (jsonModel.visibility === 'paid' && jsonModel.tiers) {
55
- jsonModel.tiers = tiersData ? tiersData.filter(t => t.type === 'paid') : [];
56
- }
57
-
58
- if (!['members', 'public', 'paid', 'tiers'].includes(jsonModel.visibility)) {
59
- const tiers = await postsService.getProductsFromVisibilityFilter(jsonModel.visibility);
60
-
61
- jsonModel.visibility = 'tiers';
62
- jsonModel.tiers = tiers;
63
- }
64
- }
65
-
66
- if (utils.isContentAPI(frame)) {
67
- // Content api v2 still expects page prop
68
- if (jsonModel.type === 'page') {
69
- jsonModel.page = true;
70
- }
71
- date.forPost(jsonModel);
72
- gating.forPost(jsonModel, frame);
73
- }
74
-
75
- // Transforms post/page metadata to flat structure
76
- let metaAttrs = _.keys(_.omit(postsMetaSchema, ['id', 'post_id']));
77
- _(metaAttrs).filter((k) => {
78
- return (!frame.options.columns || (frame.options.columns && frame.options.columns.includes(k)));
79
- }).each((attr) => {
80
- // NOTE: the default of `email_only` is `false` which is why we default to `false` instead of `null`
81
- // The undefined value is possible because `posts_meta` table is lazily created only one of the
82
- // values is assigned.
83
- const defaultValue = (attr === 'email_only') ? false : null;
84
- jsonModel[attr] = _.get(jsonModel.posts_meta, attr) || defaultValue;
85
- });
86
- delete jsonModel.posts_meta;
87
-
88
- clean.post(jsonModel, frame);
89
-
90
- if (frame.options && frame.options.withRelated) {
91
- frame.options.withRelated.forEach((relation) => {
92
- // @NOTE: this block also decorates primary_tag/primary_author objects as they
93
- // are being passed by reference in tags/authors. Might be refactored into more explicit call
94
- // in the future, but is good enough for current use-case
95
- if (relation === 'tags' && jsonModel.tags) {
96
- jsonModel.tags = jsonModel.tags.map(tag => mapTag(tag, frame));
97
- }
98
-
99
- if (relation === 'authors' && jsonModel.authors) {
100
- jsonModel.authors = jsonModel.authors.map(author => mapUser(author, frame));
101
- }
102
-
103
- if (relation === 'email' && jsonModel.email) {
104
- jsonModel.email = mapEmail(jsonModel.email, frame);
105
- }
106
-
107
- if (relation === 'email' && _.isEmpty(jsonModel.email)) {
108
- jsonModel.email = null;
109
- }
110
- });
111
- }
112
-
113
- return jsonModel;
114
- };
115
-
116
- const mapPage = async (model, frame, options) => {
117
- const jsonModel = await mapPost(model, frame, options);
118
-
119
- delete jsonModel.email_subject;
120
- delete jsonModel.email_recipient_filter;
121
- delete jsonModel.email_only;
122
-
123
- return jsonModel;
124
- };
125
-
126
- const mapSettings = (attrs, frame) => {
127
- url.forSettings(attrs);
128
- extraAttrs.forSettings(attrs, frame);
129
-
130
- // NOTE: The cleanup of deprecated ghost_head/ghost_foot has to happen here
131
- // because codeinjection_head/codeinjection_foot are assigned on a previous
132
- // `forSettings` step. This logic can be rewritten once we get rid of deprecated
133
- // fields completely.
134
- if (_.isArray(attrs)) {
135
- const keysToFilter = ['ghost_head', 'ghost_foot'];
136
-
137
- // NOTE: to support edits of deprecated 'slack' setting artificial 'slack_url' and 'slack_username'
138
- // were added to the request body in the input serializer. These should not be returned in response
139
- // body unless directly requested
140
- if (frame.original.body && frame.original.body.settings) {
141
- const requestedEditSlackUrl = frame.original.body.settings.find(s => s.key === 'slack_url');
142
- const requestedEditSlackUsername = frame.original.body.settings.find(s => s.key === 'slack_username');
143
-
144
- if (!requestedEditSlackUrl) {
145
- keysToFilter.push('slack_url');
146
- }
147
-
148
- if (!requestedEditSlackUsername) {
149
- keysToFilter.push('slack_username');
150
- }
151
- }
152
-
153
- attrs = _.filter(attrs, attr => !(keysToFilter.includes(attr.key)));
154
- }
155
-
156
- return attrs;
157
- };
158
-
159
- const mapIntegration = (model, frame) => {
160
- const jsonModel = model.toJSON ? model.toJSON(frame.options) : model;
161
-
162
- if (jsonModel.api_keys) {
163
- jsonModel.api_keys.forEach((key) => {
164
- if (key.type === 'admin') {
165
- key.secret = `${key.id}:${key.secret}`;
166
- }
167
- });
168
- }
169
-
170
- return jsonModel;
171
- };
172
-
173
- const mapImage = (path) => {
174
- return url.forImage(path);
175
- };
176
-
177
- const mapAction = (model, frame) => {
178
- const attrs = model.toJSON(frame.options);
179
- clean.action(attrs);
180
- return attrs;
181
- };
182
-
183
- const mapLabel = (model, frame) => {
184
- const jsonModel = model.toJSON ? model.toJSON(frame.options) : model;
185
- return jsonModel;
186
- };
187
-
188
- const mapEmail = (model, frame) => {
189
- const jsonModel = model.toJSON ? model.toJSON(frame.options) : model;
190
-
191
- // Ensure we're not outputting unwanted replacement strings when viewing email contents
192
- // TODO: extract this to a utility, it's duplicated in the email-preview API controller
193
- const replacements = mega.postEmailSerializer.parseReplacements(jsonModel);
194
- replacements.forEach((replacement) => {
195
- jsonModel[replacement.format] = jsonModel[replacement.format].replace(
196
- replacement.match,
197
- replacement.fallback || ''
198
- );
199
- });
200
-
201
- return jsonModel;
202
- };
203
-
204
- module.exports.mapPost = mapPost;
205
- module.exports.mapPage = mapPage;
206
- module.exports.mapUser = mapUser;
207
- module.exports.mapTag = mapTag;
208
- module.exports.mapLabel = mapLabel;
209
- module.exports.mapIntegration = mapIntegration;
210
- module.exports.mapSettings = mapSettings;
211
- module.exports.mapImage = mapImage;
212
- module.exports.mapAction = mapAction;
213
- module.exports.mapEmail = mapEmail;
@@ -1 +0,0 @@
1
- /*! normalize.css v3.0.3 | MIT License | github.com/necolas/normalize.css */html{-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%;font-family:sans-serif}body{margin:0}article,aside,details,figcaption,figure,footer,header,hgroup,main,menu,nav,section,summary{display:block}audio,canvas,progress,video{display:inline-block;vertical-align:baseline}audio:not([controls]){display:none;height:0}[hidden],template{display:none}a{background-color:transparent}a:active,a:hover{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:700}dfn{font-style:italic}h1{font-size:2em;margin:.67em 0}mark{background:#ff0;color:#000}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-.5em}sub{bottom:-.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:1em 40px}hr{box-sizing:content-box;height:0}pre{overflow:auto}code,kbd,pre,samp{font-family:monospace,monospace;font-size:1em}button,input,optgroup,select,textarea{color:inherit;font:inherit;margin:0}button{overflow:visible}button,select{text-transform:none}button,html input[type=button],input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}input{line-height:normal}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=number]::-webkit-inner-spin-button,input[type=number]::-webkit-outer-spin-button{height:auto}input[type=search]{-webkit-appearance:textfield;box-sizing:content-box}input[type=search]::-webkit-search-cancel-button,input[type=search]::-webkit-search-decoration{-webkit-appearance:none}fieldset{border:1px solid silver;margin:0 2px;padding:.35em .625em .75em}legend{border:0;padding:0}textarea{overflow:auto}optgroup{font-weight:700}table{border-collapse:collapse;border-spacing:0}td,th{padding:0}.black{color:#15171a}.darkgrey{color:#394047}.midgrey{color:#7c8b9a}.lightgrey{color:#ced4d9}.blue{color:#14b8ff}.red{color:#f50b23}.orange{color:#ffb41f}.green{color:#30cf43}.darkgrey-hover:hover{color:#394047}.midgrey-hover:hover{color:#7c8b9a}.lightgrey-hover:hover{color:#ced4d9}.blue-hover:hover{color:#14b8ff}.red-hover:hover{color:#f50b23}.orange-hover:hover{color:#ffb41f}.green-hover:hover{color:#30cf43}*,:after,:before{box-sizing:border-box}html{-webkit-tap-highlight-color:rgba(0,0,0,0);font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,Cantarell,Fira Sans,Droid Sans,Helvetica Neue,sans-serif;font-size:62.5%;letter-spacing:.2px;line-height:1.65;overflow:hidden}body,html{height:100%;width:100%}body{color:#343f44;font-size:1.4rem;overflow:auto;overflow-x:hidden}.gh-view{-ms-flex-positive:1;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;flex-grow:1}h1,h2{text-rendering:optimizeLegibility;color:#343f44;font-size:2.9rem;line-height:1.15em;margin:0 0 .3em;text-indent:-1px}@media (max-width:500px){h1{font-size:2.4rem}}.gh-input{-webkit-appearance:none;border:1px solid #d6e3eb;border-radius:4px;color:#4b5b62;display:block;font-size:1.6rem;font-weight:300;height:40px;line-height:1em;padding:10px 12px;transition:border-color .15s linear;-webkit-user-select:text;-moz-user-select:text;-ms-user-select:text;user-select:text;width:100%}.gh-input:focus{border-color:#b4cbda;outline:0}.gh-btn{fill:#829aa8;-webkit-font-smoothing:subpixel-antialiased;border:1px solid #d6e3eb;border-radius:5px;color:#829aa8;display:inline-block;outline:none;text-decoration:none!important;transition:all .2s ease;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.gh-btn span{border-radius:4px;display:block;font-size:1.3rem;font-weight:400;height:33px;letter-spacing:.2px;line-height:33px;padding:0 12px;text-align:center}.gh-btn:hover{border-color:#b4cbda}.gh-btn-hover-blue:hover{border-color:#3eb0ef;color:#3eb0ef}.gh-btn-blue{fill:#fff;background:linear-gradient(#3da1d6,#2288bf);border:0;box-shadow:0 1px 0 rgba(0,0,0,.12);color:#fff;padding:1px;text-shadow:0 -1px 0 rgba(0,0,0,.1);transition:none!important}.gh-btn-blue span{background:linear-gradient(#4ab6f0,#2fa5e4 60%,#2fa5e4 90%,#38a9e5);box-shadow:inset 0 1px 0 hsla(0,0%,100%,.1)}.gh-btn-blue:active,.gh-btn-blue:focus{background:#1e78a9}.gh-btn-blue:active span,.gh-btn-blue:focus span{background:#29a0e0;box-shadow:none}.gh-btn-block{display:block;width:100%}.gh-input-icon{display:block;position:relative}.gh-input-icon svg{fill:color(var(--midgrey) l(15%));height:14px;left:10px;position:absolute;top:50%;transform:translateY(-7px);width:auto;z-index:2}.gh-input-icon input{padding-left:35px}.gh-app{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;height:100%;overflow:hidden}.gh-viewport{max-height:100%;overflow:hidden}.gh-main,.gh-viewport{-ms-flex-positive:1;display:-ms-flexbox;display:flex;flex-grow:1}.gh-main{background:#fff;overflow-y:auto;position:relative}.gh-flow{background:linear-gradient(315deg,#efefef,#fff);min-height:100%;overflow-y:auto}.gh-flow,.gh-flow-content-wrap{display:flex;flex-direction:column;flex-grow:1}.gh-flow-content-wrap{align-items:center;flex-shrink:0;justify-content:center;margin:0 24px;padding-bottom:8vh}.gh-flow-content-wrap .site-icon{border-radius:3px;height:70px;width:70px}.gh-flow-content{background:#fff;border-radius:3px;box-shadow:0 2.8px 2.2px rgba(0,0,0,.02),0 6.7px 5.3px rgba(0,0,0,.02),0 12.5px 10px rgba(0,0,0,.02),0 22.3px 17.9px rgba(0,0,0,.03),0 41.8px 33.4px rgba(0,0,0,.03),0 100px 80px rgba(0,0,0,.05);color:var(--darkgrey);display:flex;flex-direction:column;font-size:1.9rem;font-weight:300;line-height:1.5em;margin:4rem 0 6rem;max-width:520px;padding:40px;width:100%}.gh-flow-content.unsubscribe{align-items:center;justify-content:center;margin:4rem 0;max-width:560px;min-height:200px;text-align:center}@media (max-width:500px){.gh-flow-content{background:transparent;box-shadow:none;padding:0}}.gh-flow-content header{align-items:center;display:flex;flex-direction:column}.gh-flow-content h1{color:#15171a;font-size:4.1rem;font-weight:700;line-height:1.15em;margin-bottom:24px}.gh-flow-content.unsubscribe h1{font-size:3.2rem}@media (max-width:600px){.gh-flow-content h1,.gh-flow-content.unsubscribe h1{font-size:6vw}}.gh-flow-content.unsubscribe p{color:#394047;font-size:1.8rem;margin:0 0 .4em}@media (max-width:500px){.gh-flow-content.unsubscribe p{font-size:1.6rem;line-height:1.5}}.gh-flow-content .gh-btn{display:block;margin:20px auto 0;max-width:400px}.gh-flow-content .form-group{margin-bottom:2.5rem;position:relative}.gh-flow-content .form-group.error .gh-input{border-color:#f50b23;box-shadow:0 0 0 3px rgba(239,24,24,.15)}.gh-flow-content .main-error{color:#7c8b9a;font-size:1.35rem;font-weight:400;margin:0;position:absolute;right:0;text-align:center;user-select:text}.gh-flow-em{font-weight:500}.unsubscribe-footer{font-size:1.5rem;text-align:center}@media (max-width:500px){.unsubscribe-footer{font-size:1.4rem;line-height:1.4em;padding:0 24px}}.unsubscribe-footer p{color:#7c8b9a;margin:0 0 .4rem}.unsubscribe-footer a{color:#15171a;text-decoration:none}.unsubscribe-footer a:hover{text-decoration:underline}.gh-signin{margin-bottom:1.5rem}.gh-signin .gh-input,.gh-signin .gh-input:-webkit-autofill:first-line{border-radius:8px;font-size:1.8rem;height:54px;padding:12px 16px}.gh-signin .gh-input::placeholder{color:#abb4be;font-weight:400;opacity:1}.gh-signin .gh-input::-webkit-input-placeholder{color:#abb4be;font-weight:400}.gh-signin .gh-input:-ms-input-placeholder{color:#abb4be;font-weight:400}.gh-signin .gh-input::-moz-placeholder{color:#abb4be;font-weight:400;opacity:1}.gh-signin .gh-input:focus{border-color:#30cf43;box-shadow:0 0 0 3px rgba(26,170,96,.15)}.gh-signin .gh-btn{-webkit-font-smoothing:subpixel-antialiased;background:#15171a;border-radius:8px;font-weight:300;height:54px;line-height:54px;margin:0;margin-top:32px;max-width:unset;transition:all .4s ease;width:100%}.gh-signin .gh-btn span{color:#fff;font-size:1.8rem}.error-content{flex-grow:1;justify-content:center;padding:8vw;user-select:text}.error-content,.error-details{align-items:center;display:flex}.error-details{margin-bottom:4rem}.error-ghost{height:115px;margin:15px}@media (max-width:630px){.error-ghost{display:none}}.error-code{color:#c5d2d9;font-size:10vw;font-weight:600;letter-spacing:-.4vw;line-height:.9em;margin:0}.error-description{border:none;color:#54666d;font-size:2.3rem;font-weight:300;line-height:1.3em;margin:0;padding:0}.error-message{align-items:center;display:flex;flex-direction:column;margin:15px}.error-message a{font-size:1.4rem;line-height:1;margin:8px 0}.error-link{background-color:transparent;color:#5ba4e5;text-decoration:none;transition:background .3s,color .3s}.error-stack{background-color:hsla(0,0%,100%,.3);margin:1rem auto;max-width:800px;padding:2rem}.error-stack-list{list-style-type:none;margin:0;padding:0}.error-stack-list li{display:block}.error-stack-list li:before{color:#bbb;content:"\21AA";display:inline-block;font-size:1.2rem;margin-right:.5rem}.error-stack-function{font-weight:700}