├── .cc-metadata.yml
├── .editorconfig
├── .github
└── CODEOWNERS
├── .gitignore
├── .prettierrc.js
├── .yarnrc
├── LICENSE
├── README.md
├── api
├── _fonts
│ ├── RobotoCondensed-Bold.woff2
│ ├── RobotoCondensed-Regular.woff2
│ ├── SourceSansPro-Bold.woff2
│ └── SourceSansPro-Regular.woff2
├── _lib
│ ├── chromium.ts
│ ├── options.ts
│ ├── parser.ts
│ ├── sanitizer.ts
│ ├── template.ts
│ └── types.ts
├── index.ts
└── tsconfig.json
├── package-lock.json
├── package.json
├── public
├── favicon.png
├── index.html
├── robots.txt
├── style.css
└── tweet.png
├── vercel.json
├── web
├── index.ts
└── tsconfig.json
└── yarn.lock
/.cc-metadata.yml:
--------------------------------------------------------------------------------
1 | # Whether this GitHub repo is engineering related
2 | engineering_project: true
3 | # Name of the repository/project in English
4 | english_name: CC Open Graph Image Generator
5 | # All technologies used
6 | technologies: JavaScript
7 | # Whether this repository should be featured on the CC Open Source site
8 | featured: false
9 | # Slack channel name
10 | slack: "cc-developers"
11 |
--------------------------------------------------------------------------------
/.editorconfig:
--------------------------------------------------------------------------------
1 | root = true
2 | [*]
3 | charset = utf-8
4 | indent_style = space
5 | indent_size = 2
6 | end_of_line = lf
7 | insert_final_newline = true
8 | trim_trailing_whitespace = true
9 |
--------------------------------------------------------------------------------
/.github/CODEOWNERS:
--------------------------------------------------------------------------------
1 | # https://help.github.com/en/articles/about-code-owners
2 | * @creativecommons/technology
3 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .now
2 | .vercel
3 | node_modules
4 | api/dist
5 | public/dist
6 |
7 |
--------------------------------------------------------------------------------
/.prettierrc.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | trailingComma: "es5",
3 | tabWidth: 2,
4 | semi: false,
5 | singleQuote: true,
6 | };
7 |
--------------------------------------------------------------------------------
/.yarnrc:
--------------------------------------------------------------------------------
1 | save-prefix ""
2 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2020 Creative Commons
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Open Graph Image Generator
2 |
3 | Serverless service that generates dynamic Open Graph images that you can embed in your `` tags.
4 |
5 | For each keystroke, headless chromium is used to render an HTML page and take a screenshot of the result which gets cached.
6 |
7 | ## Code of conduct
8 |
9 | [`CODE_OF_CONDUCT.md`][org-coc]:
10 | > The Creative Commons team is committed to fostering a welcoming community.
11 | > This project and all other Creative Commons open source projects are governed
12 | > by our [Code of Conduct][code_of_conduct]. Please report unacceptable
13 | > behavior to [conduct@creativecommons.org](mailto:conduct@creativecommons.org)
14 | > per our [reporting guidelines][reporting_guide].
15 |
16 | [org-coc]: https://github.com/creativecommons/.github/blob/main/CODE_OF_CONDUCT.md
17 | [code_of_conduct]: https://opensource.creativecommons.org/community/code-of-conduct/
18 | [reporting_guide]: https://opensource.creativecommons.org/community/code-of-conduct/enforcement/
19 |
20 |
21 | ## Contributing
22 |
23 | See [`CONTRIBUTING.md`][org-contrib].
24 |
25 | [org-contrib]: https://github.com/creativecommons/.github/blob/main/CONTRIBUTING.md
26 |
27 | ## What is an Open Graph Image?
28 |
29 | Have you ever posted a hyperlink to Twitter, Facebook, or Slack and seen an image popup?
30 | How did your social network know how to "unfurl" the URL and get an image?
31 | The answer is in your `
`.
32 |
33 | The [Open Graph protocol](http://ogp.me) says you can put a `` tag in the `` of a webpage to define this image.
34 |
35 | It looks like the following:
36 |
37 | ```html
38 |
39 | Title
40 |
44 |
45 | ```
46 |
47 | ## Why use this service?
48 |
49 | The short answer is that it would take a long time to painstakingly design an image for every single blog post and every single documentation page. And we don't want the exact same image for every blog post because that wouldn't make the article stand out when it was shared to Twitter.
50 |
51 | That's where `cc-og-image.vercel.app` comes in. We can simply pass the title of our blog post to our generator service and it will generate the image for us on the fly!
52 |
53 | It looks like the following:
54 |
55 | ```html
56 |
57 | Hello World
58 |
62 |
63 | ```
64 |
65 | Now try changing the text `Hello%20World` to the title of your choosing and watch the magic happen ✨
66 |
67 | ## Deploy your own
68 |
69 | You'll want to fork this repository and deploy your own image generator.
70 |
71 | 1. Click the fork button at the top right of GitHub
72 | 2. Clone the repo to your local machine with `git clone URL_OF_FORKED_REPO_HERE`
73 | 3. Change directory with `cd og-image`
74 | 4. Make changes by swapping out images, changing colors, etc (see [contributing](https://github.com/vercel/og-image/blob/main/CONTRIBUTING.md) for more info)
75 | 5. Hobby plan users will need to remove all configuration inside `vercel.json` besides `rewrites`
76 | 6. Run locally with `vercel dev` and visit [localhost:3000](http://localhost:3000) (if nothing happens, run `npm install -g vercel`)
77 | 7. Deploy to the cloud by running `vercel` and you'll get a unique URL
78 | 8. Setup [GitHub](https://vercel.com/github) to autodeploy on push
79 |
80 | If you are using a paid plan, you can do a one-click deploy with the button below.
81 |
82 | [](https://vercel.com/new/project?template=vercel/og-image)
83 |
84 | Once you have an image generator that sparks joy, you can setup [automatic GitHub](https://vercel.com/github) deployments so that pushing to main will deploy to production! 🚀
85 |
86 | ## Credits
87 |
88 | This is a fork of the lovely [Open Graph Image Generator](https://github.com/vercel/og-image) created by [Vercel](https:/vercel.com). Special thanks to their clearly-written and reuse-friendly repository!
89 |
--------------------------------------------------------------------------------
/api/_fonts/RobotoCondensed-Bold.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/creativecommons/og-image-generator/ac3a11ce906de174dfc15ce2cecad42fd6642ad3/api/_fonts/RobotoCondensed-Bold.woff2
--------------------------------------------------------------------------------
/api/_fonts/RobotoCondensed-Regular.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/creativecommons/og-image-generator/ac3a11ce906de174dfc15ce2cecad42fd6642ad3/api/_fonts/RobotoCondensed-Regular.woff2
--------------------------------------------------------------------------------
/api/_fonts/SourceSansPro-Bold.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/creativecommons/og-image-generator/ac3a11ce906de174dfc15ce2cecad42fd6642ad3/api/_fonts/SourceSansPro-Bold.woff2
--------------------------------------------------------------------------------
/api/_fonts/SourceSansPro-Regular.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/creativecommons/og-image-generator/ac3a11ce906de174dfc15ce2cecad42fd6642ad3/api/_fonts/SourceSansPro-Regular.woff2
--------------------------------------------------------------------------------
/api/_lib/chromium.ts:
--------------------------------------------------------------------------------
1 | import { launch, Page } from "puppeteer-core";
2 | import { getOptions } from "./options";
3 | import { FileType } from "./types";
4 | let _page: Page | null;
5 |
6 | async function getPage(isDev: boolean) {
7 | if (_page) {
8 | return _page;
9 | }
10 | const options = await getOptions(isDev);
11 | const browser = await launch(options);
12 | _page = await browser.newPage();
13 | return _page;
14 | }
15 |
16 | export async function getScreenshot(html: string, type: FileType, isDev: boolean) {
17 | console.log({ html });
18 | const page = await getPage(isDev);
19 | await page.setViewport({ width: 2048, height: 1170 });
20 | await page.setContent(html);
21 | const file = await page.screenshot({ type });
22 | return file;
23 | }
24 |
--------------------------------------------------------------------------------
/api/_lib/options.ts:
--------------------------------------------------------------------------------
1 | import chrome from 'chrome-aws-lambda';
2 | const exePath = process.platform === 'win32'
3 | ? 'C:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe'
4 | : process.platform === 'linux'
5 | ? '/usr/bin/google-chrome'
6 | : '/Applications/Google Chrome.app/Contents/MacOS/Google Chrome';
7 |
8 | interface Options {
9 | args: string[];
10 | executablePath: string;
11 | headless: boolean;
12 | }
13 |
14 | export async function getOptions(isDev: boolean) {
15 | let options: Options;
16 | if (isDev) {
17 | options = {
18 | args: [],
19 | executablePath: exePath,
20 | headless: true
21 | };
22 | } else {
23 | options = {
24 | args: chrome.args,
25 | executablePath: await chrome.executablePath,
26 | headless: chrome.headless,
27 | };
28 | }
29 | return options;
30 | }
31 |
--------------------------------------------------------------------------------
/api/_lib/parser.ts:
--------------------------------------------------------------------------------
1 | import { IncomingMessage } from 'http'
2 | import { parse } from 'url'
3 | import { ParsedRequest, Theme } from './types'
4 |
5 | export function parseRequest(req: IncomingMessage) {
6 | console.log('HTTP ' + req.url)
7 | const { pathname, query } = parse(req.url || '/', true)
8 | const { fontFamily, fontSize, images, widths, heights, theme, md, imageObj } = query || {}
9 |
10 | if (Array.isArray(imageObj)) {
11 | throw new Error('Expected a single image Object');
12 | }
13 | if (Array.isArray(fontFamily)) {
14 | throw new Error('Expected a single fontFamily')
15 | }
16 | if (Array.isArray(fontSize)) {
17 | throw new Error('Expected a single fontSize')
18 | }
19 | if (Array.isArray(theme)) {
20 | throw new Error('Expected a single theme')
21 | }
22 |
23 | let parsedImages = JSON.parse(imageObj || '{}');
24 | if (Object.keys(parsedImages).length === 0) {
25 | console.log('Legacy image format');
26 | parsedImages.images = getArray(images);
27 | parsedImages.widths = getArray(widths);
28 | parsedImages.heights = getArray(heights);
29 | }
30 |
31 | const arr = (pathname || '/').slice(1).split('.')
32 | let extension = ''
33 | let text = ''
34 | if (arr.length === 0) {
35 | text = ''
36 | } else if (arr.length === 1) {
37 | text = arr[0]
38 | } else {
39 | extension = arr.pop() as string
40 | text = arr.join('.')
41 | }
42 |
43 | const parsedRequest: ParsedRequest = {
44 | fileType: extension === 'jpeg' ? extension : 'png',
45 | text: decodeURIComponent(text),
46 | theme: theme === 'dark' ? 'dark' : 'light',
47 | md: md === '1' || md === 'true',
48 | fontFamily: fontFamily === 'roboto-condensed' ? 'Roboto Condensed' : 'Source Sans Pro',
49 | fontSize: fontSize || '96px',
50 | images: parsedImages.images,
51 | widths: parsedImages.widths,
52 | heights: parsedImages.heights,
53 | }
54 | parsedRequest.images = getDefaultImages(
55 | parsedRequest.images,
56 | parsedRequest.theme
57 | )
58 | return parsedRequest
59 | }
60 |
61 | function getArray(stringOrArray: string[] | string | undefined): string[] {
62 | if (typeof stringOrArray === 'undefined') {
63 | return []
64 | } else if (Array.isArray(stringOrArray)) {
65 | return stringOrArray
66 | } else {
67 | return [stringOrArray]
68 | }
69 | }
70 |
71 | function getDefaultImages(images: string[], theme: Theme): string[] {
72 | const defaultImage =
73 | theme === 'light'
74 | ? 'https://cc-vocabulary.netlify.app/logos/cc/lettermark.svg#lettermark'
75 | : 'https://cc-vocabulary.netlify.app/logos/cc/lettermark.svg#lettermark'
76 |
77 | if (!images || !images[0]) {
78 | return [defaultImage]
79 | }
80 |
81 | return images
82 | }
83 |
--------------------------------------------------------------------------------
/api/_lib/sanitizer.ts:
--------------------------------------------------------------------------------
1 | const entityMap: { [key: string]: string } = {
2 | "&": "&",
3 | "<": "<",
4 | ">": ">",
5 | '"': '"',
6 | "'": ''',
7 | "/": '/'
8 | };
9 |
10 | export function sanitizeHtml(html: string) {
11 | return String(html).replace(/[&<>"'\/]/g, key => entityMap[key]);
12 | }
13 |
14 |
--------------------------------------------------------------------------------
/api/_lib/template.ts:
--------------------------------------------------------------------------------
1 | import { readFileSync } from 'fs'
2 | import marked from 'marked'
3 | import { sanitizeHtml } from './sanitizer'
4 | import { ParsedRequest } from './types'
5 | const twemoji = require('twemoji')
6 | const twOptions = { folder: 'svg', ext: '.svg' }
7 | const emojify = (text: string) => twemoji.parse(text, twOptions)
8 |
9 | const sourceRglr = readFileSync(
10 | `${__dirname}/../_fonts/SourceSansPro-Regular.woff2`
11 | ).toString('base64')
12 | const sourceBold = readFileSync(
13 | `${__dirname}/../_fonts/SourceSansPro-Bold.woff2`
14 | ).toString('base64')
15 | const robotoRglr = readFileSync(
16 | `${__dirname}/../_fonts/RobotoCondensed-Regular.woff2`
17 | ).toString('base64')
18 | const robotoBold = readFileSync(
19 | `${__dirname}/../_fonts/RobotoCondensed-Bold.woff2`
20 | ).toString('base64')
21 |
22 | function getCss(theme: string, fontFamily: string, fontSize: string) {
23 | let background = 'white'
24 | let foreground = 'black'
25 | let bgImage =
26 | ''
27 |
28 | if (theme === 'dark') {
29 | background = 'black'
30 | foreground = 'white'
31 | bgImage =
32 | ''
33 | }
34 |
35 | let headingStyles = `
36 | font-weight: normal;
37 | text-transform: none;`
38 |
39 | if (fontFamily === 'Roboto Condensed') {
40 | headingStyles = `
41 | font-weight: bold;
42 | text-transform: uppercase;`
43 | }
44 |
45 | return `
46 | @font-face {
47 | font-family: 'Source Sans Pro';
48 | font-style: normal;
49 | font-weight: normal;
50 | src: url(data:font/woff2;charset=utf-8;base64,${sourceRglr}) format('woff2');
51 | }
52 |
53 | @font-face {
54 | font-family: 'Source Sans Pro';
55 | font-style: normal;
56 | font-weight: bold;
57 | src: url(data:font/woff2;charset=utf-8;base64,${sourceBold}) format('woff2');
58 | }
59 |
60 | @font-face {
61 | font-family: 'Roboto Condensed';
62 | font-style: normal;
63 | font-weight: normal;
64 | src: url(data:font/woff2;charset=utf-8;base64,${robotoRglr}) format('woff2');
65 | }
66 |
67 | @font-face {
68 | font-family: 'Roboto Condensed';
69 | font-style: normal;
70 | font-weight: bold;
71 | src: url(data:font/woff2;charset=utf-8;base64,${robotoBold}) format('woff2');
72 | }
73 |
74 | body {
75 | background: ${background};
76 | background-image: url(${bgImage});
77 | background-size: 300px 300px;
78 | height: 100vh;
79 | display: flex;
80 | text-align: center;
81 | align-items: center;
82 | justify-content: center;
83 | }
84 |
85 | code {
86 | color: #D400FF;
87 | font-family: 'Vera';
88 | white-space: pre-wrap;
89 | letter-spacing: -5px;
90 | }
91 |
92 | code:before, code:after {
93 | content: '\`';
94 | }
95 |
96 | .logo-wrapper {
97 | display: flex;
98 | align-items: center;
99 | align-content: center;
100 | justify-content: center;
101 | justify-items: center;
102 | }
103 |
104 | .logo {
105 | margin: 0 75px;
106 | max-width: 100%;
107 | }
108 |
109 | .dark-svg {
110 | filter: invert(100%) sepia(0%) saturate(0%) hue-rotate(12deg) brightness(103%) contrast(103%);
111 | }
112 |
113 | .plus {
114 | color: #BBB;
115 | font-family: Times New Roman, Verdana;
116 | font-size: 100px;
117 | }
118 |
119 | .spacer {
120 | margin: 150px;
121 | }
122 |
123 | .emoji {
124 | height: 1em;
125 | width: 1em;
126 | margin: 0 .05em 0 .1em;
127 | vertical-align: -0.1em;
128 | }
129 |
130 | .heading {
131 | font-family: '${sanitizeHtml(fontFamily)}', sans-serif;
132 | font-size: ${sanitizeHtml(fontSize)};
133 | color: ${foreground};
134 | ${headingStyles}
135 | line-height: 1.3;
136 | letter-spacing: 0.02rem;
137 | }`
138 | }
139 |
140 | export function getHtml(parsedReq: ParsedRequest) {
141 | const {
142 | text,
143 | theme,
144 | md,
145 | fontFamily,
146 | fontSize,
147 | images,
148 | widths,
149 | heights,
150 | } = parsedReq
151 | return `
152 |
153 |
154 | Generated Image
155 |
156 |
159 |
160 |